WebRTC 系列(四、多人通话,H5、Android、iOS)

news2025/1/11 15:52:15

WebRTC 系列(三、点对点通话,H5、Android、iOS)

 上一篇博客中,我们已经实现了点对点通话,即一对一通话,这一次就接着实现多人通话。多人通话的实现方式呢也有好几种方案,这里我简单介绍两种方案。

一、多人通话方案

1.Mesh

多个客户端之间建立多个 PeerConnection,即如果有三个客户端 A、B、C,A 有两个 PeerConnection 分别与 B、C 通信,B 也是有两个 PeerConnection,分别与 A、C 通信,C 也是有两个 PeerConnection,分别与 A、B 通信,如图:

​​​​​​​​​​​​​​优点:服务端压力小,不需要对音视频数据做处理。
缺点:客户端编解码压力较大,传输的数据与通话人数成正比,兼容性较差。

2.Mixer

客户端只与服务器有一个 PeerConnection,有多个客户端时,服务端增加多个媒体流,由服务端来做媒体数据转发,如图:

优点:客户端只有一个连接,传输数据减少,服务端可对音视频数据预处理,兼容性好。
缺点:服务器压力大,通话人数过多时,服务器如果对音视频数据有预处理,可能导致通话延迟。

3.demo 方案选择

两种方案各有利弊,感觉在实际业务中,第二种方案更合适,毕竟把更多逻辑放在服务端更可控一点,我为了演示简单,就选用了第一种方案,下面就说说第一种方案的话,第一个人、第二个人、第三个人加入房间的流程是什么样的。

第一个人 A 加入房间:

  1. A 发送 join;
  2. 服务器向房间内其他所有人发送 otherJoin;
  3. 房间内没有其他人,结束。

第二个人 B 加入房间:

  1. B 发送 join;
  2. 服务器向房间内其他所有人发送 otherJoin;
  3. A 收到 otherJoin(带有 B 的 userId);
  4. A 检查远端视频渲染控件集合中是否存在 B 的 userId 对应的远端视频渲染控件,没有则创建并初始化,然后保存;
  5. A 检查 PeerConnection 集合中是否存在 B 的 userId 对应的 PeerConnection,没有则创建并添加音轨、视轨,然后保存;
  6. A 通过 PeerConnection 创建 offer,获取 sdp;
  7. A 将 offer sdp 作为参数 setLocalDescription;
  8. A 发送 offer sdp(带有 A 的 userId);
  9. B 收到 offer(带有 A 的 userId);
  10. B 检查远端视频渲染控件集合中是否存在 A 的 userId 对应远端视频渲染控件,没有则创建并初始化,然后保存;
  11. B 检查 PeerConnection 集合中是否存在 A 的 userId 对应的 PeerConnection,没有则创建并添加音轨、视轨,然后保存;
  12. B 将 offer sdp 作为参数 setRemoteDescription;
  13. B 通过 PeerConnection 创建 answer,获取 sdp;
  14. B 将 answer sdp 作为参数 setLocalDescription;
  15. B 发送 answer sdp(带有 B 的 userId);
  16. A 收到 answer sdp(带有 B 的 userId);
  17. A 通过 userId 找到对应 PeerConnection,将 answer sdp 作为参数 setRemoteDescription。

第三个人 C 加入房间:

  1. C 发送 join;
  2. 服务器向房间内其他所有人发送 otherJoin;
  3. A 收到 otherJoin(带有 C 的 userId);
  4. A 检查远端视频渲染控件集合中是否存在 C 的 userId 对应的远端视频渲染控件,没有则创建并初始化,然后保存;
  5. A 检查 PeerConnection 集合中是否存在 C 的 userId 对应的 PeerConnection,没有则创建并添加音轨、视轨,然后保存;
  6. A 通过 PeerConnection 创建 offer,获取 sdp;
  7. A 将 offer sdp 作为参数 setLocalDescription;
  8. A 发送 offer sdp(带有 A 的 userId);
  9. C 收到 offer(带有 A 的 userId);
  10. C 检查远端视频渲染控件集合中是否存在 A 的 userId 对应远端视频渲染控件,没有则创建并初始化,然后保存;
  11. C 检查 PeerConnection 集合中是否存在 A 的 userId 对应的 PeerConnection,没有则创建并添加音轨、视轨,然后保存;
  12. C 将 offer sdp 作为参数 setRemoteDescription;
  13. C 通过 PeerConnection 创建 answer,获取 sdp;
  14. C 将 answer sdp 作为参数 setLocalDescription;
  15. C 发送 answer sdp(带有 C 的 userId);
  16. A 收到 answer sdp(带有 C 的 userId);
  17. A 通过 userId 找到对应 PeerConnection,将 answer sdp 作为参数 setRemoteDescription。
  18. B 收到 otherJoin(带有 C 的 userId);
  19. B 检查远端视频渲染控件集合中是否存在 C 的 userId 对应的远端视频渲染控件,没有则创建并初始化,然后保存;
  20. B 检查 PeerConnection 集合中是否存在 C 的 userId 对应的 PeerConnection,没有则创建并添加音轨、视轨,然后保存;
  21. B 通过 PeerConnection 创建 offer,获取 sdp;
  22. B 将 offer sdp 作为参数 setLocalDescription;
  23. B 发送 offer sdp(带有 B 的 userId);
  24. C 收到 offer(带有 B 的 userId);
  25. C 检查远端视频渲染控件集合中是否存在 B 的 userId 对应远端视频渲染控件,没有则创建并初始化,然后保存;
  26. C 检查 PeerConnection 集合中是否存在 B 的 userId 对应的 PeerConnection,没有则创建并添加音轨、视轨,然后保存;
  27. C 将 offer sdp 作为参数 setRemoteDescription;
  28. C 通过 PeerConnection 创建 answer,获取 sdp;
  29. C 将 answer sdp 作为参数 setLocalDescription;
  30. C 发送 answer sdp(带有 C 的 userId);
  31. B 收到 answer sdp(带有 C 的 userId);
  32. B 通过 userId 找到对应 PeerConnection,将 answer sdp 作为参数 setRemoteDescription。

依此类推,如果还有第四个用户 D 再加入房间的话,D 也会发送 join,然后 A、B、C 也会类似上述 3~17 步处理。

这期间的 onIceCandidate 回调的处理和之前类似,只是将生成的 IceCandidate 对象传递给对方时需要带上发送方(自己)的 userId,便于对方找到对应的 PeerConnection,以及接收方的 userId,便于服务器找到接收方的长连接。

这期间的 onAddStream 回调的处理也和之前类似,只是需要通过对方的 userId 找到对应的远端控件渲染控件。

二、信令服务器

信令服务器的依赖就不重复了,根据上述流程,我们需要引入用户的概念,但暂时我没有引入房间的概念,所以在测试的时候我认为只有一个房间,所有人都加入的同一个房间。

多人通话 WebSocket 服务端代码:

package com.qinshou.webrtcdemo_server;

import com.google.gson.Gson;
import com.google.gson.JsonObject;
import com.google.gson.reflect.TypeToken;

import org.java_websocket.WebSocket;
import org.java_websocket.handshake.ClientHandshake;
import org.java_websocket.server.WebSocketServer;

import java.net.InetSocketAddress;
import java.util.LinkedList;
import java.util.List;
import java.util.Map;

/**
 * Author: MrQinshou
 * Email: cqflqinhao@126.com
 * Date: 2023/2/8 9:33
 * Description: 多人通话 WebSocketServer
 */
public class MultipleWebSocketServerHelper {
    public static class WebSocketBean {
        private String mUserId;
        private WebSocket mWebSocket;

        public WebSocketBean() {
        }

        public WebSocketBean(WebSocket webSocket) {
            mWebSocket = webSocket;
        }

        public String getUserId() {
            return mUserId;
        }

        public void setUserId(String userId) {
            mUserId = userId;
        }

        public WebSocket getWebSocket() {
            return mWebSocket;
        }

        public void setWebSocket(WebSocket webSocket) {
            mWebSocket = webSocket;
        }
    }

    private WebSocketServer mWebSocketServer;
    private final List<WebSocketBean> mWebSocketBeans = new LinkedList<>();
    //    private static final String HOST_NAME = "192.168.1.104";
    private static final String HOST_NAME = "172.16.2.172";
    private static final int PORT = 8888;

    private WebSocketBean getWebSocketBeanByWebSocket(WebSocket webSocket) {
        for (WebSocketBean webSocketBean : mWebSocketBeans) {
            if (webSocket == webSocketBean.getWebSocket()) {
                return webSocketBean;
            }
        }
        return null;
    }

    private WebSocketBean getWebSocketBeanByUserId(String userId) {
        for (WebSocketBean webSocketBean : mWebSocketBeans) {
            if (userId.equals(webSocketBean.getUserId())) {
                return webSocketBean;
            }
        }
        return null;
    }

    private WebSocketBean removeWebSocketBeanByWebSocket(WebSocket webSocket) {
        for (WebSocketBean webSocketBean : mWebSocketBeans) {
            if (webSocket == webSocketBean.getWebSocket()) {
                mWebSocketBeans.remove(webSocketBean);
                return webSocketBean;
            }
        }
        return null;
    }

    public void start() {
        InetSocketAddress inetSocketAddress = new InetSocketAddress(HOST_NAME, PORT);
        mWebSocketServer = new WebSocketServer(inetSocketAddress) {

            @Override
            public void onOpen(WebSocket conn, ClientHandshake handshake) {
                System.out.println("onOpen--->" + conn);
                // 有客户端连接,创建 WebSocketBean,此时仅保存了 WebSocket 连接,但还没有和 userId 绑定
                mWebSocketBeans.add(new WebSocketBean(conn));
            }

            @Override
            public void onClose(WebSocket conn, int code, String reason, boolean remote) {
                System.out.println("onClose--->" + conn);
                WebSocketBean webSocketBean = removeWebSocketBeanByWebSocket(conn);
                if (webSocketBean == null) {
                    return;
                }
                // 通知其他用户有人退出房间
                JsonObject jsonObject = new JsonObject();
                jsonObject.addProperty("msgType", "otherQuit");
                jsonObject.addProperty("userId", webSocketBean.mUserId);
                for (WebSocketBean w : mWebSocketBeans) {
                    if (w != webSocketBean) {
                        w.mWebSocket.send(jsonObject.toString());
                    }
                }
            }

            @Override
            public void onMessage(WebSocket conn, String message) {
                System.out.println("onMessage--->" + message);
                Map<String, String> map = new Gson().fromJson(message, new TypeToken<Map<String, String>>() {
                }.getType());
                String msgType = map.get("msgType");
                if ("join".equals(msgType)) {
                    // 收到加入房间指令
                    String userId = map.get("userId");
                    WebSocketBean webSocketBean = getWebSocketBeanByWebSocket(conn);
                    // WebSocket 连接绑定 userId
                    if (webSocketBean != null) {
                        webSocketBean.setUserId(userId);
                    }
                    // 通知其他用户有其他人加入房间
                    JsonObject jsonObject = new JsonObject();
                    jsonObject.addProperty("msgType", "otherJoin");
                    jsonObject.addProperty("userId", userId);
                    for (WebSocketBean w : mWebSocketBeans) {
                        if (w != webSocketBean && w.getUserId() != null) {
                            w.mWebSocket.send(jsonObject.toString());
                        }
                    }
                    return;
                }
                if ("quit".equals(msgType)) {
                    // 收到退出房间指令
                    String userId = map.get("userId");
                    WebSocketBean webSocketBean = getWebSocketBeanByWebSocket(conn);
                    // WebSocket 连接解绑 userId
                    if (webSocketBean != null) {
                        webSocketBean.setUserId(null);
                    }
                    // 通知其他用户有其他人退出房间
                    JsonObject jsonObject = new JsonObject();
                    jsonObject.addProperty("msgType", "otherQuit");
                    jsonObject.addProperty("userId", userId);
                    for (WebSocketBean w : mWebSocketBeans) {
                        if (w != webSocketBean && w.getUserId() != null) {
                            w.mWebSocket.send(jsonObject.toString());
                        }
                    }
                    return;
                }
                // 其他消息透传
                // 接收方
                String toUserId = map.get("toUserId");
                // 找到接收方对应 WebSocket 连接
                WebSocketBean webSocketBean = getWebSocketBeanByUserId(toUserId);
                if (webSocketBean != null) {
                    webSocketBean.getWebSocket().send(message);
                }
            }

            @Override
            public void onError(WebSocket conn, Exception ex) {
                ex.printStackTrace();
                System.out.println("onError");
            }

            @Override
            public void onStart() {
                System.out.println("onStart");
            }
        };
        mWebSocketServer.start();
    }

    public void stop() {
        if (mWebSocketServer == null) {
            return;
        }
        for (WebSocket webSocket : mWebSocketServer.getConnections()) {
            webSocket.close();
        }
        try {
            mWebSocketServer.stop();
        } catch (InterruptedException e) {
            throw new RuntimeException(e);
        }
        mWebSocketServer = null;
    }

    public static void main(String[] args) {
        new MultipleWebSocketServerHelper().start();
    }
}

三、消息格式

传递的消息的话,相较于点对点通话,sdp 和 iceCandidate 中需要添加 fromUserId 和 toUserId 字段,另外还需要增加 join、otherJoin、quit、ohterQuit 消息:

// sdp
{
    "msgType": "sdp",
    "fromUserId": userId,
    "toUserId": toUserId,
    "type": sessionDescription.type,
    "sdp": sessionDescription.sdp
}
 
// iceCandidate
{
	"msgType": "iceCandidate",
    "fromUserId": userId,
    "toUserId": toUserId,
	"id": iceCandidate.sdpMid,
	"label": iceCandidate.sdpMLineIndex,
	"candidate": iceCandidate.candidate
}

// join
{
    "msgType": "join"
    "userId": userId
}

// otherJoin
{
    "msgType": "otherJoin"
    "userId": userId
}

// quit
{
    "msgType": "quit"
    "userId": userId
}

// otherQuit
{
    "msgType": "otherQuit"
    "userId": userId
}

四、H5

代码与 p2p_demo 其实差不了太多,但是我们创建 PeerConnection 的时机需要根据上面梳理流程进行修改,发送的信令也需要根据上面定义的格式进行修改,布局中将远端视频渲染控件去掉,改成一个远端视频渲染控件的容器,每当有新的连接时创建新的远端视频渲染控件放到容器中,另外,WebSocket 需要额外处理 otherJoin 和 otherQuit 信令。

1.添加依赖

这个跟前两篇的一样,不需要额外引入。

2.multiple_demo.html

<html>

<head>
    <title>Multiple Demo</title>
    <style>
        body {
            overflow: hidden;
            margin: 0px;
            padding: 0px;
        }

        #local_view {
            width: 100%;
            height: 100%;
        }

        #remote_views {
            width: 9%;
            height: 80%;
            position: absolute;
            top: 10%;
            right: 10%;
            bottom: 10%;
            overflow-y: auto;
        }

        .remote_view {
            width: 100%;
            aspect-ratio: 9/16;
        }

        #left {
            width: 10%;
            height: 5%;
            position: absolute;
            left: 10%;
            top: 10%;
        }

        #p_websocket_state,
        #input_server_url,
        .my_button {
            width: 100%;
            height: 100%;
            display: block;
            margin-bottom: 10%;
        }
    </style>
</head>

<body>
    <video id="local_view" width="480" height="270" autoplay controls muted></video>
    <div id="remote_views">
    </div>

    <div id="left">
        <p id="p_websocket_state">WebSocket 已断开</p>
        <input id="input_server_url" type="text" placeholder="请输入服务器地址" value="ws://192.168.1.104:8888"></input>
        <button id="btn_connect" class="my_button" onclick="connect()">连接 WebSocket</button>
        <button id="btn_disconnect" class="my_button" onclick="disconnect()">断开 WebSocket</button>
        <button id="btn_join" class="my_button" onclick="join()">加入房间</button>
        <button id="btn_quit" class="my_button" onclick="quit()">退出房间</button>
    </div>
</body>

<script type="text/javascript">
    /**
     * Author: MrQinshou
     * Email: cqflqinhao@126.com
     * Date: 2023/4/15 11:24
     * Description: 生成 uuid
     */
    function uuid() {
        return 'xxxxxxxx-xxxx-4xxx-yxxx-xxxxxxxxxxxx'.replace(/[xy]/g, function (c) {
            var r = Math.random() * 16 | 0;
            var v = c == 'x' ? r : (r & 0x3 | 0x8);
            return v.toString(16);
        });
    }
</script>

<script type="text/javascript">
    var localView = document.getElementById("local_view");
    var remoteViews = document.getElementById("remote_views");
    var localStream;
    // let userId = uuid();
    let userId = "h5";
    let peerConnectionDict = {};
    let remoteViewDict = {};

    function createPeerConnection(fromUserId) {
        let peerConnection = new RTCPeerConnection();
        peerConnection.oniceconnectionstatechange = function (event) {
            if ("disconnected" == event.target.iceConnectionState) {
                let peerConnection = peerConnectionDict[fromUserId];
                if (peerConnection != null) {
                    peerConnection.close();
                    delete peerConnectionDict[fromUserId];
                }
                let remoteView = remoteViewDict[fromUserId];
                if (remoteView != null) {
                    remoteView.removeAttribute('src');
                    remoteView.load();
                    remoteView.remove();
                    delete remoteViewDict[fromUserId];
                }
            }
        }
        peerConnection.onicecandidate = function (event) {
            console.log("onicecandidate--->" + event.candidate);
            sendIceCandidate(event.candidate, fromUserId);
        }
        peerConnection.ontrack = function (event) {
            console.log("remote ontrack--->" + event.streams);
            let remoteView = remoteViewDict[fromUserId];
            if (remoteView == null) {
                return;
            }
            let streams = event.streams;
            if (streams && streams.length > 0) {
                remoteView.srcObject = streams[0];
            }
        }
        return peerConnection;
    }

    function createOffer(peerConnection, fromUserId) {
        peerConnection.createOffer().then(function (sessionDescription) {
            console.log(fromUserId + " create offer success.");
            peerConnection.setLocalDescription(sessionDescription).then(function () {
                console.log(fromUserId + " set local sdp success.");
                var jsonObject = {
                    "msgType": "sdp",
                    "fromUserId": userId,
                    "toUserId": fromUserId,
                    "type": "offer",
                    "sdp": sessionDescription.sdp
                };
                send(JSON.stringify(jsonObject));
            }).catch(function (error) {
                console.log("error--->" + error);
            })
        }).catch(function (error) {
            console.log("error--->" + error);
        })
    }

    function createAnswer(peerConnection, fromUserId) {
        peerConnection.createAnswer().then(function (sessionDescription) {
            console.log(fromUserId + " create answer success.");
            peerConnection.setLocalDescription(sessionDescription).then(function () {
                console.log(fromUserId + " set local sdp success.");
                var jsonObject = {
                    "msgType": "sdp",
                    "fromUserId": userId,
                    "toUserId": fromUserId,
                    "type": "answer",
                    "sdp": sessionDescription.sdp
                };
                send(JSON.stringify(jsonObject));
            }).catch(function (error) {
                console.log("error--->" + error);
            })
        }).catch(function (error) {
            console.log("error--->" + error);
        })
    }

    function join() {
        var jsonObject = {
            "msgType": "join",
            "userId": userId,
        };
        send(JSON.stringify(jsonObject));
    }

    function quit() {
        var jsonObject = {
            "msgType": "quit",
            "userId": userId,
        };
        send(JSON.stringify(jsonObject));
        for (var key in peerConnectionDict) {
            let peerConnection = peerConnectionDict[key];
            peerConnection.close();
            delete peerConnectionDict[key];
        }
        for (var key in remoteViewDict) {
            let remoteView = remoteViewDict[key];
            remoteView.removeAttribute('src');
            remoteView.load();
            remoteView.remove();
            delete remoteViewDict[key];
        }
    }


    function sendOffer(offer, toUserId) {
        var jsonObject = {
            "msgType": "sdp",
            "fromUserId": userId,
            "toUserId": toUserId,
            "type": "offer",
            "sdp": offer.sdp
        };
        send(JSON.stringify(jsonObject));
    }

    function receivedOffer(jsonObject) {
        let fromUserId = jsonObject["fromUserId"];
        var peerConnection = peerConnectionDict[fromUserId];
        if (peerConnection == null) {
            // 创建 PeerConnection
            peerConnection = createPeerConnection(fromUserId);
            // 为 PeerConnection 添加音轨、视轨
            for (let i = 0; localStream != null && i < localStream.getTracks().length; i++) {
                const track = localStream.getTracks()[i];
                peerConnection.addTrack(track, localStream);
            }
            peerConnectionDict[fromUserId] = peerConnection;
        }
        var remoteView = remoteViewDict[fromUserId];
        if (remoteView == null) {
            remoteView = document.createElement("video");
            remoteView.className = "remote_view";
            remoteView.autoplay = true;
            remoteView.control = true;
            remoteView.muted = true;
            remoteViews.appendChild(remoteView);
            remoteViewDict[fromUserId] = remoteView;
        }
        let options = {
            "type": jsonObject["type"],
            "sdp": jsonObject["sdp"]
        }
        // 将 offer sdp 作为参数 setRemoteDescription
        let sessionDescription = new RTCSessionDescription(options);
        peerConnection.setRemoteDescription(sessionDescription).then(function () {
            console.log(fromUserId + " set remote sdp success.");
            // 通过 PeerConnection 创建 answer,获取 sdp
            peerConnection.createAnswer().then(function (sessionDescription) {
                console.log(fromUserId + " create answer success.");
                // 将 answer sdp 作为参数 setLocalDescription
                peerConnection.setLocalDescription(sessionDescription).then(function () {
                    console.log(fromUserId + " set local sdp success.");
                    // 发送 answer sdp
                    sendAnswer(sessionDescription, fromUserId);
                })
            })
        }).catch(function (error) {
            console.log("error--->" + error);
        });
    }

    function sendAnswer(answer, toUserId) {
        var jsonObject = {
            "msgType": "sdp",
            "fromUserId": userId,
            "toUserId": toUserId,
            "type": "answer",
            "sdp": answer.sdp
        };
        send(JSON.stringify(jsonObject));
    }

    function receivedAnswer(jsonObject) {
        let fromUserId = jsonObject["fromUserId"];
        var peerConnection = peerConnectionDict[fromUserId];
        if (peerConnection == null) {
            // 创建 PeerConnection
            peerConnection = createPeerConnection(fromUserId);
            // 为 PeerConnection 添加音轨、视轨
            for (let i = 0; localStream != null && i < localStream.getTracks().length; i++) {
                const track = localStream.getTracks()[i];
                peerConnection.addTrack(track, localStream);
            }
            peerConnectionDict[fromUserId] = peerConnection;
        }
        var remoteView = remoteViewDict[fromUserId];
        if (remoteView == null) {
            remoteView = document.createElement("video");
            remoteView.className = "remote_view";
            remoteView.autoplay = true;
            remoteView.control = true;
            remoteView.muted = true;
            remoteViews.appendChild(remoteView);
            remoteViewDict[fromUserId] = remoteView;
        }
        let options = {
            "type": jsonObject["type"],
            "sdp": jsonObject["sdp"]
        }
        let sessionDescription = new RTCSessionDescription(options);
        let type = jsonObject["type"];
        peerConnection.setRemoteDescription(sessionDescription).then(function () {
            console.log(fromUserId + " set remote sdp success.");
        }).catch(function (error) {
            console.log("error--->" + error);
        });
    }

    function sendIceCandidate(iceCandidate, toUserId) {
        if (iceCandidate == null) {
            return;
        }
        var jsonObject = {
            "msgType": "iceCandidate",
            "fromUserId": userId,
            "toUserId": toUserId,
            "id": iceCandidate.sdpMid,
            "label": iceCandidate.sdpMLineIndex,
            "candidate": iceCandidate.candidate
        };
        send(JSON.stringify(jsonObject));
    }

    function receivedCandidate(jsonObject) {
        let fromUserId = jsonObject["fromUserId"];
        let peerConnection = peerConnectionDict[fromUserId];
        if (peerConnection == null) {
            return
        }
        let options = {
            "sdpMLineIndex": jsonObject["label"],
            "sdpMid": jsonObject["id"],
            "candidate": jsonObject["candidate"]
        }
        let iceCandidate = new RTCIceCandidate(options);
        peerConnection.addIceCandidate(iceCandidate);
    }

    function receivedOtherJoin(jsonObject) {
        // 创建 PeerConnection
        let userId = jsonObject["userId"];
        var peerConnection = peerConnectionDict[userId];
        if (peerConnection == null) {
            peerConnection = createPeerConnection(userId);
            for (let i = 0; localStream != null && i < localStream.getTracks().length; i++) {
                const track = localStream.getTracks()[i];
                peerConnection.addTrack(track, localStream);
            }
            peerConnectionDict[userId] = peerConnection;
        }
        var remoteView = remoteViewDict[userId];
        if (remoteView == null) {
            remoteView = document.createElement("video");
            remoteView.className = "remote_view";
            remoteView.autoplay = true;
            remoteView.control = true;
            remoteView.muted = true;
            remoteViews.appendChild(remoteView);
            remoteViewDict[userId] = remoteView;
        }
        // 通过 PeerConnection 创建 offer,获取 sdp
        peerConnection.createOffer().then(function (sessionDescription) {
            console.log(userId + " create offer success.");
            // 将 offer sdp 作为参数 setLocalDescription
            peerConnection.setLocalDescription(sessionDescription).then(function () {
                console.log(userId + " set local sdp success.");
                // 发送 offer sdp
                sendOffer(sessionDescription, userId);
            }).catch(function (error) {
                console.log("error--->" + error);
            })
        }).catch(function (error) {
            console.log("error--->" + error);
        });
    }

    function receivedOtherQuit(jsonObject) {
        let userId = jsonObject["userId"];
        let peerConnection = peerConnectionDict[userId];
        if (peerConnection != null) {
            peerConnection.close();
            delete peerConnectionDict[userId];
        }
        let remoteView = remoteViewDict[userId];
        if (remoteView != null) {
            remoteView.removeAttribute('src');
            remoteView.load();
            remoteView.remove();
            delete remoteViewDict[userId];
        }
    }

    navigator.mediaDevices.getUserMedia({ audio: true, video: true }).then(function (mediaStream) {
        // 初始化 PeerConnectionFactory;
        // 创建 EglBase;
        // 创建 PeerConnectionFactory;
        // 创建音轨;
        // 创建视轨;
        localStream = mediaStream;
        // 初始化本地视频渲染控件;
        // 初始化远端视频渲染控件;
        // 开始本地渲染。
        localView.srcObject = mediaStream;
    }).catch(function (error) {
        console.log("error--->" + error);
    })
</script>

<script type="text/javascript">
    var websocket;

    function connect() {
        let inputServerUrl = document.getElementById("input_server_url");
        let pWebsocketState = document.getElementById("p_websocket_state");
        let url = inputServerUrl.value;
        websocket = new WebSocket(url);
        websocket.onopen = function () {
            console.log("onOpen");
            pWebsocketState.innerText = "WebSocket 已连接";
        }
        websocket.onmessage = function (message) {
            console.log("onmessage--->" + message.data);
            let jsonObject = JSON.parse(message.data);
            let msgType = jsonObject["msgType"];
            if ("sdp" == msgType) {
                let type = jsonObject["type"];
                if ("offer" == type) {
                    receivedOffer(jsonObject);
                } else if ("answer" == type) {
                    receivedAnswer(jsonObject);
                }
            } else if ("iceCandidate" == msgType) {
                receivedCandidate(jsonObject);
            } else if ("otherJoin" == msgType) {
                receivedOtherJoin(jsonObject);
            } else if ("otherQuit" == msgType) {
                receivedOtherQuit(jsonObject);
            }
        }
        websocket.onclose = function (error) {
            console.log("onclose--->" + error);
            pWebsocketState.innerText = "WebSocket 已断开";
        }
        websocket.onerror = function (error) {
            console.log("onerror--->" + error);
        }
    }

    function disconnect() {
        websocket.close();
    }

    function send(message) {
        if (!websocket) {
            return;
        }
        websocket.send(message);
    }

</script>

</html>

多人通话至少需要三个端,我们就等所有端都实现了再最后来看效果。

五、Android

1.添加依赖

这个跟前两篇的一样,不需要额外引入。

2.布局

<?xml version="1.0" encoding="utf-8"?>
<androidx.constraintlayout.widget.ConstraintLayout xmlns:android="http://schemas.android.com/apk/res/android"
    xmlns:app="http://schemas.android.com/apk/res-auto"
    xmlns:tools="http://schemas.android.com/tools"
    android:layout_width="match_parent"
    android:layout_height="match_parent"
    android:background="#FF000000"
    android:keepScreenOn="true"
    tools:context=".P2PDemoActivity">

    <org.webrtc.SurfaceViewRenderer
        android:id="@+id/svr_local"
        android:layout_width="match_parent"
        android:layout_height="0dp"
        app:layout_constraintBottom_toBottomOf="parent"
        app:layout_constraintDimensionRatio="9:16"
        app:layout_constraintEnd_toEndOf="parent"
        app:layout_constraintStart_toStartOf="parent"
        app:layout_constraintTop_toTopOf="parent" />

    <androidx.core.widget.NestedScrollView
        android:layout_width="90dp"
        android:layout_height="wrap_content"
        android:layout_marginTop="30dp"
        android:layout_marginEnd="30dp"
        android:layout_marginBottom="30dp"
        app:layout_constraintEnd_toEndOf="parent"
        app:layout_constraintTop_toTopOf="parent">

        <androidx.appcompat.widget.LinearLayoutCompat
            android:id="@+id/ll_remotes"
            android:layout_width="match_parent"
            android:layout_height="wrap_content"
            android:orientation="vertical">

        </androidx.appcompat.widget.LinearLayoutCompat>
    </androidx.core.widget.NestedScrollView>

    <androidx.appcompat.widget.LinearLayoutCompat
        android:layout_width="match_parent"
        android:layout_height="wrap_content"
        android:layout_marginStart="30dp"
        android:layout_marginTop="30dp"
        android:layout_marginEnd="30dp"
        android:orientation="vertical"
        app:layout_constraintStart_toStartOf="parent"
        app:layout_constraintTop_toTopOf="parent">

        <androidx.appcompat.widget.AppCompatTextView
            android:id="@+id/tv_websocket_state"
            android:layout_width="match_parent"
            android:layout_height="wrap_content"
            android:text="WebSocket 已断开"
            android:textColor="#FFFFFFFF" />

        <androidx.appcompat.widget.AppCompatEditText
            android:id="@+id/et_server_url"
            android:layout_width="match_parent"
            android:layout_height="wrap_content"
            android:hint="请输入服务器地址"
            android:textColor="#FFFFFFFF"
            android:textColorHint="#FFFFFFFF" />

        <androidx.appcompat.widget.AppCompatButton
            android:id="@+id/btn_connect"
            android:layout_width="wrap_content"
            android:layout_height="wrap_content"
            android:text="连接 WebSocket"
            android:textAllCaps="false" />

        <androidx.appcompat.widget.AppCompatButton
            android:id="@+id/btn_disconnect"
            android:layout_width="wrap_content"
            android:layout_height="wrap_content"
            android:text="断开 WebSocket"
            android:textAllCaps="false" />

        <androidx.appcompat.widget.AppCompatButton
            android:id="@+id/btn_join"
            android:layout_width="wrap_content"
            android:layout_height="wrap_content"
            android:text="加入房间"
            android:textSize="12sp" />

        <androidx.appcompat.widget.AppCompatButton
            android:id="@+id/btn_quit"
            android:layout_width="wrap_content"
            android:layout_height="wrap_content"
            android:text="退出房间"
            android:textSize="12sp" />
    </androidx.appcompat.widget.LinearLayoutCompat>
</androidx.constraintlayout.widget.ConstraintLayout>

布局中将远端视频渲染控件去掉,改成一个远端视频渲染控件的容器,每当有新的连接时创建新的远端视频渲染控件放到容器中。

3.MultipleDemoActivity.java

package com.qinshou.webrtcdemo_android;

import android.content.Context;
import android.os.Bundle;
import android.text.TextUtils;
import android.view.View;
import android.view.ViewGroup;
import android.widget.EditText;
import android.widget.LinearLayout;
import android.widget.TextView;

import androidx.appcompat.app.AppCompatActivity;
import androidx.appcompat.widget.LinearLayoutCompat;

import org.json.JSONException;
import org.json.JSONObject;
import org.webrtc.AudioSource;
import org.webrtc.AudioTrack;
import org.webrtc.Camera2Capturer;
import org.webrtc.Camera2Enumerator;
import org.webrtc.CameraEnumerator;
import org.webrtc.DataChannel;
import org.webrtc.DefaultVideoDecoderFactory;
import org.webrtc.DefaultVideoEncoderFactory;
import org.webrtc.EglBase;
import org.webrtc.IceCandidate;
import org.webrtc.MediaConstraints;
import org.webrtc.MediaStream;
import org.webrtc.PeerConnection;
import org.webrtc.PeerConnectionFactory;
import org.webrtc.RtpReceiver;
import org.webrtc.SessionDescription;
import org.webrtc.SurfaceTextureHelper;
import org.webrtc.SurfaceViewRenderer;
import org.webrtc.VideoCapturer;
import org.webrtc.VideoDecoderFactory;
import org.webrtc.VideoEncoderFactory;
import org.webrtc.VideoSource;
import org.webrtc.VideoTrack;

import java.util.ArrayList;
import java.util.List;
import java.util.Map;
import java.util.UUID;
import java.util.concurrent.ConcurrentHashMap;

/**
 * Author: MrQinshou
 * Email: cqflqinhao@126.com
 * Date: 2023/3/21 17:22
 * Description: P2P demo
 */
public class MultipleDemoActivity extends AppCompatActivity {
    private static final String TAG = MultipleDemoActivity.class.getSimpleName();
    private static final String AUDIO_TRACK_ID = "ARDAMSa0";
    private static final String VIDEO_TRACK_ID = "ARDAMSv0";
    private static final List<String> STREAM_IDS = new ArrayList<String>() {{
        add("ARDAMS");
    }};
    private static final String SURFACE_TEXTURE_HELPER_THREAD_NAME = "SurfaceTextureHelperThread";
    private static final int WIDTH = 1280;
    private static final int HEIGHT = 720;
    private static final int FPS = 30;

    private EglBase mEglBase;
    private PeerConnectionFactory mPeerConnectionFactory;
    private VideoCapturer mVideoCapturer;
    private AudioTrack mAudioTrack;
    private VideoTrack mVideoTrack;
    private WebSocketClientHelper mWebSocketClientHelper = new WebSocketClientHelper();
//    private String mUserId = UUID.randomUUID().toString();
    private String mUserId = "Android";
    private final Map<String, PeerConnection> mPeerConnectionMap = new ConcurrentHashMap<>();
    private final Map<String, SurfaceViewRenderer> mRemoteViewMap = new ConcurrentHashMap<>();

    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_multiple_demo);
        ((EditText) findViewById(R.id.et_server_url)).setText("ws://192.168.1.104:8888");
        findViewById(R.id.btn_connect).setOnClickListener(new View.OnClickListener() {
            @Override
            public void onClick(View view) {
                String url = ((EditText) findViewById(R.id.et_server_url)).getText().toString().trim();
                mWebSocketClientHelper.connect(url);
            }
        });
        findViewById(R.id.btn_disconnect).setOnClickListener(new View.OnClickListener() {
            @Override
            public void onClick(View view) {
                mWebSocketClientHelper.disconnect();
            }
        });
        findViewById(R.id.btn_join).setOnClickListener(new View.OnClickListener() {
            @Override
            public void onClick(View view) {
                join();
            }
        });
        findViewById(R.id.btn_quit).setOnClickListener(new View.OnClickListener() {
            @Override
            public void onClick(View view) {
                quit();
            }
        });
        mWebSocketClientHelper.setOnWebSocketListener(new WebSocketClientHelper.OnWebSocketClientListener() {
            @Override
            public void onOpen() {
                runOnUiThread(new Runnable() {
                    @Override
                    public void run() {
                        ((TextView) findViewById(R.id.tv_websocket_state)).setText("WebSocket 已连接");
                    }
                });
            }

            @Override
            public void onClose() {
                runOnUiThread(new Runnable() {
                    @Override
                    public void run() {
                        ((TextView) findViewById(R.id.tv_websocket_state)).setText("WebSocket 已断开");
                    }
                });
            }

            @Override
            public void onMessage(String message) {
                ShowLogUtil.debug("message--->" + message);
                try {
                    JSONObject jsonObject = new JSONObject(message);
                    String msgType = jsonObject.optString("msgType");
                    if (TextUtils.equals("sdp", msgType)) {
                        String type = jsonObject.optString("type");
                        if (TextUtils.equals("offer", type)) {
                            receivedOffer(jsonObject);
                        } else if (TextUtils.equals("answer", type)) {
                            receivedAnswer(jsonObject);
                        }
                    } else if (TextUtils.equals("iceCandidate", msgType)) {
                        receivedCandidate(jsonObject);
                    } else if (TextUtils.equals("otherJoin", msgType)) {
                        receivedOtherJoin(jsonObject);
                    } else if (TextUtils.equals("otherQuit", msgType)) {
                        receivedOtherQuit(jsonObject);
                    }
                } catch (JSONException e) {
                    e.printStackTrace();
                }
            }
        });
        // 初始化 PeerConnectionFactory
        initPeerConnectionFactory(MultipleDemoActivity.this);
        // 创建 EglBase
        mEglBase = EglBase.create();
        // 创建 PeerConnectionFactory
        mPeerConnectionFactory = createPeerConnectionFactory(mEglBase);
        // 创建音轨
        mAudioTrack = createAudioTrack(mPeerConnectionFactory);
        // 创建视轨
        mVideoCapturer = createVideoCapturer();
        VideoSource videoSource = createVideoSource(mPeerConnectionFactory, mVideoCapturer);
        mVideoTrack = createVideoTrack(mPeerConnectionFactory, videoSource);
        // 初始化本地视频渲染控件,这个方法非常重要,不初始化会黑屏
        SurfaceViewRenderer svrLocal = findViewById(R.id.svr_local);
        svrLocal.init(mEglBase.getEglBaseContext(), null);
        mVideoTrack.addSink(svrLocal);
        // 开始本地渲染
        // 创建 SurfaceTextureHelper,用来表示 camera 初始化的线程
        SurfaceTextureHelper surfaceTextureHelper = SurfaceTextureHelper.create(SURFACE_TEXTURE_HELPER_THREAD_NAME, mEglBase.getEglBaseContext());
        // 初始化视频采集器
        mVideoCapturer.initialize(surfaceTextureHelper, MultipleDemoActivity.this, videoSource.getCapturerObserver());
        mVideoCapturer.startCapture(WIDTH, HEIGHT, FPS);
    }

    @Override
    protected void onDestroy() {
        super.onDestroy();
        if (mEglBase != null) {
            mEglBase.release();
            mEglBase = null;
        }
        if (mVideoCapturer != null) {
            try {
                mVideoCapturer.stopCapture();
            } catch (InterruptedException e) {
                e.printStackTrace();
            }
            mVideoCapturer.dispose();
            mVideoCapturer = null;
        }
        if (mAudioTrack != null) {
            mAudioTrack.dispose();
            mAudioTrack = null;
        }
        if (mVideoTrack != null) {
            mVideoTrack.dispose();
            mVideoTrack = null;
        }
        for (PeerConnection peerConnection : mPeerConnectionMap.values()) {
            peerConnection.close();
            peerConnection.dispose();
        }
        mPeerConnectionMap.clear();
        SurfaceViewRenderer svrLocal = findViewById(R.id.svr_local);
        svrLocal.release();
        for (SurfaceViewRenderer surfaceViewRenderer : mRemoteViewMap.values()) {
            surfaceViewRenderer.release();
        }
        mRemoteViewMap.clear();
        mWebSocketClientHelper.disconnect();
    }

    private void initPeerConnectionFactory(Context context) {
        PeerConnectionFactory.initialize(PeerConnectionFactory.InitializationOptions.builder(context).createInitializationOptions());
    }

    private PeerConnectionFactory createPeerConnectionFactory(EglBase eglBase) {
        VideoEncoderFactory videoEncoderFactory = new DefaultVideoEncoderFactory(eglBase.getEglBaseContext(), true, true);
        VideoDecoderFactory videoDecoderFactory = new DefaultVideoDecoderFactory(eglBase.getEglBaseContext());
        return PeerConnectionFactory.builder().setVideoEncoderFactory(videoEncoderFactory).setVideoDecoderFactory(videoDecoderFactory).createPeerConnectionFactory();
    }

    private AudioTrack createAudioTrack(PeerConnectionFactory peerConnectionFactory) {
        AudioSource audioSource = peerConnectionFactory.createAudioSource(new MediaConstraints());
        AudioTrack audioTrack = peerConnectionFactory.createAudioTrack(AUDIO_TRACK_ID, audioSource);
        audioTrack.setEnabled(true);
        return audioTrack;
    }

    private VideoCapturer createVideoCapturer() {
        VideoCapturer videoCapturer = null;
        CameraEnumerator cameraEnumerator = new Camera2Enumerator(MultipleDemoActivity.this);
        for (String deviceName : cameraEnumerator.getDeviceNames()) {
            // 前摄像头
            if (cameraEnumerator.isFrontFacing(deviceName)) {
                videoCapturer = new Camera2Capturer(MultipleDemoActivity.this, deviceName, null);
            }
        }
        return videoCapturer;
    }

    private VideoSource createVideoSource(PeerConnectionFactory peerConnectionFactory, VideoCapturer videoCapturer) {
        // 创建视频源
        VideoSource videoSource = peerConnectionFactory.createVideoSource(videoCapturer.isScreencast());
        return videoSource;
    }

    private VideoTrack createVideoTrack(PeerConnectionFactory peerConnectionFactory, VideoSource videoSource) {
        // 创建视轨
        VideoTrack videoTrack = peerConnectionFactory.createVideoTrack(VIDEO_TRACK_ID, videoSource);
        videoTrack.setEnabled(true);
        return videoTrack;
    }

    private PeerConnection createPeerConnection(PeerConnectionFactory peerConnectionFactory, String fromUserId) {
        // 内部会转成 RTCConfiguration
        List<PeerConnection.IceServer> iceServers = new ArrayList<>();
        PeerConnection peerConnection = peerConnectionFactory.createPeerConnection(iceServers, new PeerConnection.Observer() {
            @Override
            public void onSignalingChange(PeerConnection.SignalingState signalingState) {
            }

            @Override
            public void onIceConnectionChange(PeerConnection.IceConnectionState iceConnectionState) {
                ShowLogUtil.debug("onIceConnectionChange--->" + iceConnectionState);
                if (iceConnectionState == PeerConnection.IceConnectionState.DISCONNECTED) {
                    PeerConnection peerConnection = mPeerConnectionMap.get(fromUserId);
                    ShowLogUtil.debug("peerConnection--->" + peerConnection);
                    if (peerConnection != null) {
                        peerConnection.close();
                        mPeerConnectionMap.remove(fromUserId);
                    }
                    runOnUiThread(new Runnable() {
                        @Override
                        public void run() {
                            SurfaceViewRenderer surfaceViewRenderer = mRemoteViewMap.get(fromUserId);
                            if (surfaceViewRenderer != null) {
                                ((ViewGroup) surfaceViewRenderer.getParent()).removeView(surfaceViewRenderer);
                                mRemoteViewMap.remove(fromUserId);
                            }
                        }
                    });
                }
            }

            @Override
            public void onIceConnectionReceivingChange(boolean b) {

            }

            @Override
            public void onIceGatheringChange(PeerConnection.IceGatheringState iceGatheringState) {

            }

            @Override
            public void onIceCandidate(IceCandidate iceCandidate) {
                ShowLogUtil.verbose("onIceCandidate--->" + iceCandidate);
                sendIceCandidate(iceCandidate, fromUserId);
            }

            @Override
            public void onIceCandidatesRemoved(IceCandidate[] iceCandidates) {

            }

            @Override
            public void onAddStream(MediaStream mediaStream) {
                ShowLogUtil.verbose("onAddStream--->" + mediaStream);
                if (mediaStream == null || mediaStream.videoTracks == null || mediaStream.videoTracks.isEmpty()) {
                    return;
                }
                runOnUiThread(new Runnable() {
                    @Override
                    public void run() {
                        SurfaceViewRenderer surfaceViewRenderer = mRemoteViewMap.get(fromUserId);
                        if (surfaceViewRenderer != null) {
                            mediaStream.videoTracks.get(0).addSink(surfaceViewRenderer);
                        }
                    }
                });
            }

            @Override
            public void onRemoveStream(MediaStream mediaStream) {
            }

            @Override
            public void onDataChannel(DataChannel dataChannel) {

            }

            @Override
            public void onRenegotiationNeeded() {

            }

            @Override
            public void onAddTrack(RtpReceiver rtpReceiver, MediaStream[] mediaStreams) {

            }
        });
        return peerConnection;
    }

    private void join() {
        try {
            JSONObject jsonObject = new JSONObject();
            jsonObject.put("msgType", "join");
            jsonObject.put("userId", mUserId);
            mWebSocketClientHelper.send(jsonObject.toString());
        } catch (JSONException e) {
            e.printStackTrace();
        }
    }

    private void quit() {
        try {
            JSONObject jsonObject = new JSONObject();
            jsonObject.put("msgType", "quit");
            jsonObject.put("userId", mUserId);
            mWebSocketClientHelper.send(jsonObject.toString());
        } catch (JSONException e) {
            e.printStackTrace();
        }
        new Thread(new Runnable() {
            @Override
            public void run() {
                for (PeerConnection peerConnection : mPeerConnectionMap.values()) {
                    peerConnection.close();
                }
                mPeerConnectionMap.clear();
            }
        }).start();
        for (SurfaceViewRenderer surfaceViewRenderer : mRemoteViewMap.values()) {
            ((ViewGroup) surfaceViewRenderer.getParent()).removeView(surfaceViewRenderer);
        }
        mRemoteViewMap.clear();
    }

    private void sendOffer(SessionDescription offer, String toUserId) {
        try {
            JSONObject jsonObject = new JSONObject();
            jsonObject.put("msgType", "sdp");
            jsonObject.put("fromUserId", mUserId);
            jsonObject.put("toUserId", toUserId);
            jsonObject.put("type", "offer");
            jsonObject.put("sdp", offer.description);
            mWebSocketClientHelper.send(jsonObject.toString());
        } catch (JSONException e) {
            e.printStackTrace();
        }
    }

    private void receivedOffer(JSONObject jsonObject) {
        String fromUserId = jsonObject.optString("fromUserId");
        PeerConnection peerConnection = mPeerConnectionMap.get(fromUserId);
        if (peerConnection == null) {
            // 创建 PeerConnection
            peerConnection = createPeerConnection(mPeerConnectionFactory, fromUserId);
            // 为 PeerConnection 添加音轨、视轨
            peerConnection.addTrack(mAudioTrack, STREAM_IDS);
            peerConnection.addTrack(mVideoTrack, STREAM_IDS);
            mPeerConnectionMap.put(fromUserId, peerConnection);
        }
        runOnUiThread(new Runnable() {
            @Override
            public void run() {
                SurfaceViewRenderer surfaceViewRenderer = mRemoteViewMap.get(fromUserId);
                if (surfaceViewRenderer == null) {
                    // 初始化 SurfaceViewRender ,这个方法非常重要,不初始化黑屏
                    surfaceViewRenderer = new SurfaceViewRenderer(MultipleDemoActivity.this);
                    surfaceViewRenderer.init(mEglBase.getEglBaseContext(), null);
                    surfaceViewRenderer.setLayoutParams(new LinearLayout.LayoutParams(dp2px(MultipleDemoActivity.this, 90), dp2px(MultipleDemoActivity.this, 160)));
                    LinearLayoutCompat llRemotes = findViewById(R.id.ll_remotes);
                    llRemotes.addView(surfaceViewRenderer);
                    mRemoteViewMap.put(fromUserId, surfaceViewRenderer);
                }
            }
        });
        String type = jsonObject.optString("type");
        String sdp = jsonObject.optString("sdp");
        PeerConnection finalPeerConnection = peerConnection;
        // 将 offer sdp 作为参数 setRemoteDescription
        SessionDescription sessionDescription = new SessionDescription(SessionDescription.Type.fromCanonicalForm(type), sdp);
        peerConnection.setRemoteDescription(new MySdpObserver() {
            @Override
            public void onCreateSuccess(SessionDescription sessionDescription) {
            }

            @Override
            public void onSetSuccess() {
                ShowLogUtil.debug(fromUserId + " set remote sdp success.");
                // 通过 PeerConnection 创建 answer,获取 sdp
                MediaConstraints mediaConstraints = new MediaConstraints();
                finalPeerConnection.createAnswer(new MySdpObserver() {
                    @Override
                    public void onCreateSuccess(SessionDescription sessionDescription) {
                        ShowLogUtil.verbose(fromUserId + "create answer success.");
                        // 将 answer sdp 作为参数 setLocalDescription
                        finalPeerConnection.setLocalDescription(new MySdpObserver() {
                            @Override
                            public void onCreateSuccess(SessionDescription sessionDescription) {

                            }

                            @Override
                            public void onSetSuccess() {
                                ShowLogUtil.verbose(fromUserId + " set local sdp success.");
                                // 发送 answer sdp
                                sendAnswer(sessionDescription, fromUserId);
                            }
                        }, sessionDescription);
                    }

                    @Override
                    public void onSetSuccess() {

                    }
                }, mediaConstraints);
            }
        }, sessionDescription);
    }

    private void sendAnswer(SessionDescription answer, String toUserId) {
        try {
            JSONObject jsonObject = new JSONObject();
            jsonObject.put("msgType", "sdp");
            jsonObject.put("fromUserId", mUserId);
            jsonObject.put("toUserId", toUserId);
            jsonObject.put("type", "answer");
            jsonObject.put("sdp", answer.description);
            mWebSocketClientHelper.send(jsonObject.toString());
        } catch (JSONException e) {
            e.printStackTrace();
        }
    }

    private void receivedAnswer(JSONObject jsonObject) {
        String fromUserId = jsonObject.optString("fromUserId");
        PeerConnection peerConnection = mPeerConnectionMap.get(fromUserId);
        if (peerConnection == null) {
            peerConnection = createPeerConnection(mPeerConnectionFactory, fromUserId);
            peerConnection.addTrack(mAudioTrack, STREAM_IDS);
            peerConnection.addTrack(mVideoTrack, STREAM_IDS);
            mPeerConnectionMap.put(fromUserId, peerConnection);
        }
        runOnUiThread(new Runnable() {
            @Override
            public void run() {
                SurfaceViewRenderer surfaceViewRenderer = mRemoteViewMap.get(fromUserId);
                if (surfaceViewRenderer == null) {
                    // 初始化 SurfaceViewRender ,这个方法非常重要,不初始化黑屏
                    surfaceViewRenderer = new SurfaceViewRenderer(MultipleDemoActivity.this);
                    surfaceViewRenderer.init(mEglBase.getEglBaseContext(), null);
                    surfaceViewRenderer.setLayoutParams(new LinearLayout.LayoutParams(dp2px(MultipleDemoActivity.this, 90), dp2px(MultipleDemoActivity.this, 160)));
                    LinearLayoutCompat llRemotes = findViewById(R.id.ll_remotes);
                    llRemotes.addView(surfaceViewRenderer);
                    mRemoteViewMap.put(fromUserId, surfaceViewRenderer);
                }
            }
        });
        String type = jsonObject.optString("type");
        String sdp = jsonObject.optString("sdp");
        // 收到 answer sdp,将 answer sdp 作为参数 setRemoteDescription
        SessionDescription sessionDescription = new SessionDescription(SessionDescription.Type.fromCanonicalForm(type), sdp);
        peerConnection.setRemoteDescription(new MySdpObserver() {
            @Override
            public void onCreateSuccess(SessionDescription sessionDescription) {
            }

            @Override
            public void onSetSuccess() {
                ShowLogUtil.debug(fromUserId + " set remote sdp success.");
            }
        }, sessionDescription);
    }

    private void sendIceCandidate(IceCandidate iceCandidate, String toUserId) {
        try {
            JSONObject jsonObject = new JSONObject();
            jsonObject.put("msgType", "iceCandidate");
            jsonObject.put("fromUserId", mUserId);
            jsonObject.put("toUserId", toUserId);
            jsonObject.put("id", iceCandidate.sdpMid);
            jsonObject.put("label", iceCandidate.sdpMLineIndex);
            jsonObject.put("candidate", iceCandidate.sdp);
            mWebSocketClientHelper.send(jsonObject.toString());
        } catch (JSONException e) {
            e.printStackTrace();
        }
    }

    private void receivedCandidate(JSONObject jsonObject) {
        String fromUserId = jsonObject.optString("fromUserId");
        PeerConnection peerConnection = mPeerConnectionMap.get(fromUserId);
        if (peerConnection == null) {
            return;
        }
        String id = jsonObject.optString("id");
        int label = jsonObject.optInt("label");
        String candidate = jsonObject.optString("candidate");
        IceCandidate iceCandidate = new IceCandidate(id, label, candidate);
        peerConnection.addIceCandidate(iceCandidate);
    }

    private void receivedOtherJoin(JSONObject jsonObject) throws JSONException {
        String userId = jsonObject.optString("userId");
        PeerConnection peerConnection = mPeerConnectionMap.get(userId);
        if (peerConnection == null) {
            // 创建 PeerConnection
            peerConnection = createPeerConnection(mPeerConnectionFactory, userId);
            // 为 PeerConnection 添加音轨、视轨
            peerConnection.addTrack(mAudioTrack, STREAM_IDS);
            peerConnection.addTrack(mVideoTrack, STREAM_IDS);
            mPeerConnectionMap.put(userId, peerConnection);
        }
        runOnUiThread(new Runnable() {
            @Override
            public void run() {
                SurfaceViewRenderer surfaceViewRenderer = mRemoteViewMap.get(userId);
                if (surfaceViewRenderer == null) {
                    // 初始化 SurfaceViewRender ,这个方法非常重要,不初始化黑屏
                    surfaceViewRenderer = new SurfaceViewRenderer(MultipleDemoActivity.this);
                    surfaceViewRenderer.init(mEglBase.getEglBaseContext(), null);
                    surfaceViewRenderer.setLayoutParams(new LinearLayout.LayoutParams(dp2px(MultipleDemoActivity.this, 90), dp2px(MultipleDemoActivity.this, 160)));
                    LinearLayoutCompat llRemotes = findViewById(R.id.ll_remotes);
                    llRemotes.addView(surfaceViewRenderer);
                    mRemoteViewMap.put(userId, surfaceViewRenderer);
                }
            }
        });
        PeerConnection finalPeerConnection = peerConnection;
        // 通过 PeerConnection 创建 offer,获取 sdp
        MediaConstraints mediaConstraints = new MediaConstraints();
        peerConnection.createOffer(new MySdpObserver() {
            @Override
            public void onCreateSuccess(SessionDescription sessionDescription) {
                ShowLogUtil.verbose(userId + " create offer success.");
                // 将 offer sdp 作为参数 setLocalDescription
                finalPeerConnection.setLocalDescription(new MySdpObserver() {
                    @Override
                    public void onCreateSuccess(SessionDescription sessionDescription) {

                    }

                    @Override
                    public void onSetSuccess() {
                        ShowLogUtil.verbose(userId + " set local sdp success.");
                        // 发送 offer sdp
                        sendOffer(sessionDescription, userId);
                    }
                }, sessionDescription);
            }

            @Override
            public void onSetSuccess() {

            }
        }, mediaConstraints);
    }

    private void receivedOtherQuit(JSONObject jsonObject) throws JSONException {
        String userId = jsonObject.optString("userId");
        PeerConnection peerConnection = mPeerConnectionMap.get(userId);
        if (peerConnection != null) {
            peerConnection.close();
            mPeerConnectionMap.remove(userId);
        }
        runOnUiThread(new Runnable() {
            @Override
            public void run() {
                SurfaceViewRenderer surfaceViewRenderer = mRemoteViewMap.get(userId);
                if (surfaceViewRenderer != null) {
                    ((ViewGroup) surfaceViewRenderer.getParent()).removeView(surfaceViewRenderer);
                    mRemoteViewMap.remove(userId);
                }
            }
        });
    }

    public static int dp2px(Context context, float dp) {
        float density = context.getResources().getDisplayMetrics().density;
        return (int) (dp * density + 0.5f);
    }
}

其中 WebSocketClientHelper 跟之前一样的,其余逻辑跟 H5 是一样的。多人通话至少需要三个端,我们就等所有端都实现了再最后来看效果。

六、iOS

1.添加依赖

这个跟前两篇的一样,不需要额外引入。

2.MultipleDemoViewController.swift

//
//  LocalDemoViewController.swift
//  WebRTCDemo-iOS
//
//  Created by 覃浩 on 2023/3/21.
//

import UIKit
import WebRTC
import SnapKit

class MultipleDemoViewController: UIViewController {
    private static let AUDIO_TRACK_ID = "ARDAMSa0"
    private static let VIDEO_TRACK_ID = "ARDAMSv0"
    private static let STREAM_IDS = ["ARDAMS"]
    private static let WIDTH = 1280
    private static let HEIGHT = 720
    private static let FPS = 30
    
    private var localView: RTCEAGLVideoView!
    private var remoteViews: UIScrollView!
    private var peerConnectionFactory: RTCPeerConnectionFactory!
    private var audioTrack: RTCAudioTrack?
    private var videoTrack: RTCVideoTrack?
    /**
     iOS 需要将 Capturer 保存为全局变量,否则无法渲染本地画面
     */
    private var videoCapturer: RTCVideoCapturer?
    /**
     iOS 需要将远端流保存为全局变量,否则无法渲染远端画面
     */
    private var remoteStreamDict: [String : RTCMediaStream] = [:]
//    private let userId = UUID().uuidString
    private let userId = "iOS"
    private var peerConnectionDict: [String : RTCPeerConnection] = [:]
    private var remoteViewDict: [String : RTCEAGLVideoView] = [:]
    private var lbWebSocketState: UILabel? = nil
    private var tfServerUrl: UITextField? = nil
    private let webSocketHelper = WebSocketClientHelper()
    
    override func viewDidLoad() {
        super.viewDidLoad()
        // 表明 View 不要扩展到整个屏幕,而是在 NavigationBar 下的区域
        edgesForExtendedLayout = UIRectEdge()
        self.view.backgroundColor = UIColor.black
        // WebSocket 状态文本框
        lbWebSocketState = UILabel()
        lbWebSocketState!.textColor = UIColor.white
        lbWebSocketState!.text = "WebSocket 已断开"
        self.view.addSubview(lbWebSocketState!)
        lbWebSocketState!.snp.makeConstraints({ make in
            make.left.equalToSuperview().offset(30)
            make.right.equalToSuperview().offset(-30)
            make.height.equalTo(40)
        })
        // 服务器地址输入框
        tfServerUrl = UITextField()
        tfServerUrl!.textColor = UIColor.white
        tfServerUrl!.text = "ws://192.168.1.104:8888"
        tfServerUrl!.placeholder = "请输入服务器地址"
        tfServerUrl!.delegate = self
        self.view.addSubview(tfServerUrl!)
        tfServerUrl!.snp.makeConstraints({ make in
            make.left.equalToSuperview().offset(30)
            make.right.equalToSuperview().offset(-30)
            make.height.equalTo(20)
            make.top.equalTo(lbWebSocketState!.snp.bottom).offset(10)
        })
        // 连接 WebSocket 按钮
        let btnConnect = UIButton()
        btnConnect.backgroundColor = UIColor.lightGray
        btnConnect.setTitle("连接 WebSocket", for: .normal)
        btnConnect.setTitleColor(UIColor.black, for: .normal)
        btnConnect.addTarget(self, action: #selector(connect), for: .touchUpInside)
        self.view.addSubview(btnConnect)
        btnConnect.snp.makeConstraints({ make in
            make.left.equalToSuperview().offset(30)
            make.width.equalTo(140)
            make.height.equalTo(40)
            make.top.equalTo(tfServerUrl!.snp.bottom).offset(10)
        })
        // 断开 WebSocket 按钮
        let btnDisconnect = UIButton()
        btnDisconnect.backgroundColor = UIColor.lightGray
        btnDisconnect.setTitle("断开 WebSocket", for: .normal)
        btnDisconnect.setTitleColor(UIColor.black, for: .normal)
        btnDisconnect.addTarget(self, action: #selector(disconnect), for: .touchUpInside)
        self.view.addSubview(btnDisconnect)
        btnDisconnect.snp.makeConstraints({ make in
            make.left.equalToSuperview().offset(30)
            make.width.equalTo(140)
            make.height.equalTo(40)
            make.top.equalTo(btnConnect.snp.bottom).offset(10)
        })
        // 呼叫按钮
        let btnCall = UIButton()
        btnCall.backgroundColor = UIColor.lightGray
        btnCall.setTitle("加入房间", for: .normal)
        btnCall.setTitleColor(UIColor.black, for: .normal)
        btnCall.addTarget(self, action: #selector(join), for: .touchUpInside)
        self.view.addSubview(btnCall)
        btnCall.snp.makeConstraints({ make in
            make.left.equalToSuperview().offset(30)
            make.width.equalTo(160)
            make.height.equalTo(40)
            make.top.equalTo(btnDisconnect.snp.bottom).offset(10)
        })
        // 挂断按钮
        let btnHangUp = UIButton()
        btnHangUp.backgroundColor = UIColor.lightGray
        btnHangUp.setTitle("退出房间", for: .normal)
        btnHangUp.setTitleColor(UIColor.black, for: .normal)
        btnHangUp.addTarget(self, action: #selector(quit), for: .touchUpInside)
        self.view.addSubview(btnHangUp)
        btnHangUp.snp.makeConstraints({ make in
            make.left.equalToSuperview().offset(30)
            make.width.equalTo(160)
            make.height.equalTo(40)
            make.top.equalTo(btnCall.snp.bottom).offset(10)
        })
        webSocketHelper.setDelegate(delegate: self)
        // 初始化 PeerConnectionFactory
        initPeerConnectionFactory()
        // 创建 EglBase
        // 创建 PeerConnectionFactory
        peerConnectionFactory = createPeerConnectionFactory()
        // 创建音轨
        audioTrack = createAudioTrack(peerConnectionFactory: peerConnectionFactory)
        // 创建视轨
        videoTrack = createVideoTrack(peerConnectionFactory: peerConnectionFactory)
        let tuple = createVideoCapturer(videoSource: videoTrack!.source)
        let captureDevice = tuple.captureDevice
        videoCapturer = tuple.videoCapture
        // 初始化本地视频渲染控件
        localView = RTCEAGLVideoView()
        localView.delegate = self
        self.view.insertSubview(localView,at: 0)
        localView.snp.makeConstraints({ make in
            make.width.equalToSuperview()
            make.height.equalTo(localView.snp.width).multipliedBy(16.0/9.0)
            make.centerY.equalToSuperview()
        })
        videoTrack?.add(localView!)
        // 开始本地渲染
        (videoCapturer as? RTCCameraVideoCapturer)?.startCapture(with: captureDevice!, format: captureDevice!.activeFormat, fps: MultipleDemoViewController.FPS)
        // 初始化远端视频渲染控件容器
        remoteViews = UIScrollView()
        self.view.insertSubview(remoteViews, aboveSubview: localView)
        remoteViews.snp.makeConstraints { maker in
            maker.width.equalTo(90)
            maker.top.equalToSuperview().offset(30)
            maker.right.equalToSuperview().offset(-30)
            maker.bottom.equalToSuperview().offset(-30)
        }
    }
    
    override func viewDidDisappear(_ animated: Bool) {
        (videoCapturer as? RTCCameraVideoCapturer)?.stopCapture()
        videoCapturer = nil
        for peerConnection in peerConnectionDict.values {
            peerConnection.close()
        }
        peerConnectionDict.removeAll(keepingCapacity: false)
        remoteViewDict.removeAll(keepingCapacity: false)
        remoteStreamDict.removeAll(keepingCapacity: false)
        webSocketHelper.disconnect()
    }
    
    private func initPeerConnectionFactory() {
        RTCPeerConnectionFactory.initialize()
    }
    
    private func createPeerConnectionFactory() -> RTCPeerConnectionFactory {
        var videoEncoderFactory = RTCDefaultVideoEncoderFactory()
        var videoDecoderFactory = RTCDefaultVideoDecoderFactory()
        if TARGET_OS_SIMULATOR != 0 {
            videoEncoderFactory = RTCSimluatorVideoEncoderFactory()
            videoDecoderFactory = RTCSimulatorVideoDecoderFactory()
        }
        return RTCPeerConnectionFactory(encoderFactory: videoEncoderFactory, decoderFactory: videoDecoderFactory)
    }
    
    private func createAudioTrack(peerConnectionFactory: RTCPeerConnectionFactory) -> RTCAudioTrack {
        let mandatoryConstraints : [String : String] = [:]
        let optionalConstraints : [String : String] = [:]
        let audioSource = peerConnectionFactory.audioSource(with: RTCMediaConstraints(mandatoryConstraints: mandatoryConstraints, optionalConstraints: optionalConstraints))
        let audioTrack = peerConnectionFactory.audioTrack(with: audioSource, trackId: MultipleDemoViewController.AUDIO_TRACK_ID)
        audioTrack.isEnabled = true
        return audioTrack
    }
    
    private func createVideoTrack(peerConnectionFactory: RTCPeerConnectionFactory) -> RTCVideoTrack? {
        let videoSource = peerConnectionFactory.videoSource()
        let videoTrack = peerConnectionFactory.videoTrack(with: videoSource, trackId: MultipleDemoViewController.VIDEO_TRACK_ID)
        videoTrack.isEnabled = true
        return videoTrack
    }
    
    private func createVideoCapturer(videoSource: RTCVideoSource) -> (captureDevice: AVCaptureDevice?, videoCapture: RTCVideoCapturer?) {
        let videoCapturer = RTCCameraVideoCapturer(delegate: videoSource)
        let captureDevices = RTCCameraVideoCapturer.captureDevices()
        if (captureDevices.count == 0) {
            return (nil, nil)
        }
        var captureDevice: AVCaptureDevice?
        for c in captureDevices {
            // 前摄像头
             if (c.position == .front) {
                captureDevice = c
                break
            }
        }
        if (captureDevice == nil) {
            return (nil, nil)
        }
        return (captureDevice, videoCapturer)
    }
    
    private func createPeerConnection(peerConnectionFactory: RTCPeerConnectionFactory, fromUserId: String) -> RTCPeerConnection {
        let configuration = RTCConfiguration()
        //        configuration.sdpSemantics = .unifiedPlan
        //        configuration.continualGatheringPolicy = .gatherContinually
        //        configuration.iceServers = [RTCIceServer(urlStrings: ["stun:stun.l.google.com:19302"])]
        let mandatoryConstraints : [String : String] = [:]
        //      let mandatoryConstraints = [kRTCMediaConstraintsOfferToReceiveAudio: kRTCMediaConstraintsValueTrue,
        //                                  kRTCMediaConstraintsOfferToReceiveVideo: kRTCMediaConstraintsValueTrue]
        let optionalConstraints : [String : String] = [:]
        //        let optionalConstraints = ["DtlsSrtpKeyAgreement" : kRTCMediaConstraintsValueTrue]
        let mediaConstraints = RTCMediaConstraints(mandatoryConstraints: mandatoryConstraints, optionalConstraints: optionalConstraints)
        return peerConnectionFactory.peerConnection(with: configuration, constraints: mediaConstraints, delegate: self)
    }
    
    @objc private func connect() {
        webSocketHelper.connect(url: tfServerUrl!.text!.trimmingCharacters(in: .whitespacesAndNewlines))
    }
    @objc private func disconnect() {
        webSocketHelper.disconnect()
    }
    
    @objc private func join() {
        var jsonObject = [String : String]()
        jsonObject["msgType"] = "join"
        jsonObject["userId"] = userId
        do {
            let data = try JSONSerialization.data(withJSONObject: jsonObject)
            webSocketHelper.send(message: String(data: data, encoding: .utf8)!)
        } catch {
            ShowLogUtil.verbose("error--->\(error)")
        }
    }
    
    @objc private func quit() {
        var jsonObject = [String : String]()
        jsonObject["msgType"] = "quit"
        jsonObject["userId"] = userId
        do {
            let data = try JSONSerialization.data(withJSONObject: jsonObject)
            webSocketHelper.send(message: String(data: data, encoding: .utf8)!)
        } catch {
            ShowLogUtil.verbose("error--->\(error)")
        }
        for peerConnection in peerConnectionDict.values {
            peerConnection.close()
        }
        peerConnectionDict.removeAll(keepingCapacity: false)
        for (key, value) in remoteViewDict {
            remoteViews.removeSubview(view: value)
        }
        remoteViewDict.removeAll(keepingCapacity: false)
    }

    
    private func sendOffer(offer: RTCSessionDescription, toUserId: String) {
        var jsonObject = [String : String]()
        jsonObject["msgType"] = "sdp"
        jsonObject["fromUserId"] = userId
        jsonObject["toUserId"] = toUserId
        jsonObject["type"] = "offer"
        jsonObject["sdp"] = offer.sdp
        do {
            let data = try JSONSerialization.data(withJSONObject: jsonObject)
            webSocketHelper.send(message: String(data: data, encoding: .utf8)!)
        } catch {
            ShowLogUtil.verbose("error--->\(error)")
        }
    }
    
    private func receivedOffer(jsonObject: [String : Any]) {
        let fromUserId = jsonObject["fromUserId"] as? String ?? ""
        var peerConnection = peerConnectionDict[fromUserId]
        if (peerConnection == nil) {
            // 创建 PeerConnection
            peerConnection = createPeerConnection(peerConnectionFactory: peerConnectionFactory, fromUserId: fromUserId)
            // 为 PeerConnection 添加音轨、视轨
            peerConnection!.add(audioTrack!, streamIds: MultipleDemoViewController.STREAM_IDS)
            peerConnection!.add(videoTrack!, streamIds: MultipleDemoViewController.STREAM_IDS)
            peerConnectionDict[fromUserId] = peerConnection
        }
        var remoteView = remoteViewDict[fromUserId]
        if (remoteView == nil) {
            let x = 0
            var y = 0
            if (remoteViews.subviews.count == 0) {
                y = 0
            } else {
                for i in 0..<remoteViews.subviews.count {
                    y += Int(remoteViews.subviews[i].frame.height)
                }
            }
            let width = 90
            let height = width / 9 * 16
            remoteView = RTCEAGLVideoView(frame: CGRect(x: x, y: y, width: width, height: height))
            remoteViews.appendSubView(view: remoteView!)
            remoteViewDict[fromUserId] = remoteView
        }
        // 将 offer sdp 作为参数 setRemoteDescription
        let type = jsonObject["type"] as? String
        let sdp = jsonObject["sdp"] as? String
        let sessionDescription = RTCSessionDescription(type: .offer, sdp: sdp!)
        peerConnection?.setRemoteDescription(sessionDescription, completionHandler: { _ in
            ShowLogUtil.verbose("\(fromUserId) set remote sdp success.")
            // 通过 PeerConnection 创建 answer,获取 sdp
            let mandatoryConstraints : [String : String] = [:]
            let optionalConstraints : [String : String] = [:]
            let mediaConstraints = RTCMediaConstraints(mandatoryConstraints: mandatoryConstraints, optionalConstraints: optionalConstraints)
            peerConnection?.answer(for: mediaConstraints, completionHandler: { sessionDescription, error in
                ShowLogUtil.verbose("\(fromUserId) create answer success.")
                // 将 answer sdp 作为参数 setLocalDescription
                peerConnection?.setLocalDescription(sessionDescription!, completionHandler: { _ in
                    ShowLogUtil.verbose("\(fromUserId) set local sdp success.")
                    // 发送 answer sdp
                    self.sendAnswer(answer: sessionDescription!, toUserId: fromUserId)
                })
            })
        })
    }
    
    private func sendAnswer(answer: RTCSessionDescription, toUserId: String) {
        var jsonObject = [String : String]()
        jsonObject["msgType"] = "sdp"
        jsonObject["fromUserId"] = userId
        jsonObject["toUserId"] = toUserId
        jsonObject["type"] = "answer"
        jsonObject["sdp"] = answer.sdp
        do {
            let data = try JSONSerialization.data(withJSONObject: jsonObject)
            webSocketHelper.send(message: String(data: data, encoding: .utf8)!)
        } catch {
            ShowLogUtil.verbose("error--->\(error)")
        }
    }
    
    private func receivedAnswer(jsonObject: [String : Any]) {
        let fromUserId = jsonObject["fromUserId"] as? String ?? ""
        var peerConnection = peerConnectionDict[fromUserId]
        if (peerConnection == nil) {
            peerConnection = createPeerConnection(peerConnectionFactory: peerConnectionFactory, fromUserId: fromUserId)
            peerConnection!.add(audioTrack!, streamIds: MultipleDemoViewController.STREAM_IDS)
            peerConnection!.add(videoTrack!, streamIds: MultipleDemoViewController.STREAM_IDS)
            peerConnectionDict[fromUserId] = peerConnection
        }
        DispatchQueue.main.async {
            var remoteView = self.remoteViewDict[fromUserId]
            if (remoteView == nil) {
                let x = 0
                var y = 0
                if (self.remoteViews.subviews.count == 0) {
                    y = 0
                } else {
                    for i in 0..<self.remoteViews.subviews.count {
                        y += Int(self.remoteViews.subviews[i].frame.height)
                    }
                }
                let width = 90
                let height = width / 9 * 16
                remoteView = RTCEAGLVideoView(frame: CGRect(x: x, y: y, width: width, height: height))
                self.remoteViews.appendSubView(view: remoteView!)
                self.remoteViewDict[fromUserId] = remoteView
            }
        }
        // 收到 answer sdp,将 answer sdp 作为参数 setRemoteDescription
        let type = jsonObject["type"] as? String
        let sdp = jsonObject["sdp"] as? String
        let sessionDescription = RTCSessionDescription(type: .answer, sdp: sdp!)
        peerConnection!.setRemoteDescription(sessionDescription, completionHandler: { _ in
            ShowLogUtil.verbose(fromUserId + " set remote sdp success.");
        })
    }
    
    private func sendIceCandidate(iceCandidate: RTCIceCandidate, toUserId: String)  {
        var jsonObject = [String : Any]()
        jsonObject["msgType"] = "iceCandidate"
        jsonObject["fromUserId"] = userId
        jsonObject["toUserId"] = toUserId
        jsonObject["id"] = iceCandidate.sdpMid
        jsonObject["label"] = iceCandidate.sdpMLineIndex
        jsonObject["candidate"] = iceCandidate.sdp
        do {
            let data = try JSONSerialization.data(withJSONObject: jsonObject)
            webSocketHelper.send(message: String(data: data, encoding: .utf8)!)
        } catch {
            ShowLogUtil.verbose("error--->\(error)")
        }
    }
    
    private func receivedCandidate(jsonObject: [String : Any]) {
        let fromUserId = jsonObject["fromUserId"] as? String ?? ""
        let peerConnection = peerConnectionDict[fromUserId]
        if (peerConnection == nil) {
            return
        }
        let id = jsonObject["id"] as? String
        let label = jsonObject["label"] as? Int32
        let candidate = jsonObject["candidate"] as? String
        let iceCandidate = RTCIceCandidate(sdp: candidate!, sdpMLineIndex: label!, sdpMid: id)
        peerConnection!.add(iceCandidate)
    }
    
    private func receiveOtherJoin(jsonObject: [String : Any]) {
        let userId = jsonObject["userId"] as? String ?? ""
        var peerConnection = peerConnectionDict[userId]
        if (peerConnection == nil) {
            // 创建 PeerConnection
            peerConnection = createPeerConnection(peerConnectionFactory: peerConnectionFactory, fromUserId: userId)
            // 为 PeerConnection 添加音轨、视轨
            peerConnection!.add(audioTrack!, streamIds: MultipleDemoViewController.STREAM_IDS)
            peerConnection!.add(videoTrack!, streamIds: MultipleDemoViewController.STREAM_IDS)
            peerConnectionDict[userId] = peerConnection
        }
        DispatchQueue.main.async {
            var remoteView = self.remoteViewDict[userId]
            if (remoteView == nil) {
                let x = 0
                var y = 0
                if (self.remoteViews.subviews.count == 0) {
                    y = 0
                } else {
                    for i in 0..<self.remoteViews.subviews.count {
                        y += Int(self.remoteViews.subviews[i].frame.height)
                    }
                }
                let width = 90
                let height = width / 9 * 16
                remoteView = RTCEAGLVideoView(frame: CGRect(x: x, y: y, width: width, height: height))
                self.remoteViews.appendSubView(view: remoteView!)
                self.remoteViewDict[userId] = remoteView
            }
        }
        // 通过 PeerConnection 创建 offer,获取 sdp
        let mandatoryConstraints : [String : String] = [:]
        let optionalConstraints : [String : String] = [:]
        let mediaConstraints = RTCMediaConstraints(mandatoryConstraints: mandatoryConstraints, optionalConstraints: optionalConstraints)
        peerConnection?.offer(for: mediaConstraints, completionHandler: { sessionDescription, error in
            ShowLogUtil.verbose("\(userId) create offer success.")
            if (error != nil) {
                return
            }
            // 将 offer sdp 作为参数 setLocalDescription
            peerConnection?.setLocalDescription(sessionDescription!, completionHandler: { _ in
                ShowLogUtil.verbose("\(userId) set local sdp success.")
                // 发送 offer sdp
                self.sendOffer(offer: sessionDescription!, toUserId: userId)
            })
        })
    }
    
    private func receiveOtherQuit(jsonObject: [String : Any]) {
        let userId = jsonObject["userId"] as? String ?? ""
        Thread(block: {
            let peerConnection = self.peerConnectionDict[userId]
            if (peerConnection != nil) {
                peerConnection?.close()
                self.peerConnectionDict.removeValue(forKey: userId)
            }
        }).start()
        let remoteView = remoteViewDict[userId]
        if (remoteView != nil) {
            remoteViews.removeSubview(view: remoteView!)
            remoteViewDict.removeValue(forKey: userId)
        }
        remoteStreamDict.removeValue(forKey: userId)
    }
}

// MARK: - RTCVideoViewDelegate
extension MultipleDemoViewController: RTCVideoViewDelegate {
    func videoView(_ videoView: RTCVideoRenderer, didChangeVideoSize size: CGSize) {
    }
}

// MARK: - RTCPeerConnectionDelegate
extension MultipleDemoViewController: RTCPeerConnectionDelegate {
    func peerConnection(_ peerConnection: RTCPeerConnection, didChange stateChanged: RTCSignalingState) {
    }
    
    func peerConnection(_ peerConnection: RTCPeerConnection, didAdd stream: RTCMediaStream) {
        ShowLogUtil.verbose("peerConnection didAdd stream--->\(stream)")
        var userId: String?
        for (key, value) in peerConnectionDict {
            if (value == peerConnection) {
                userId = key
            }
        }
        if (userId == nil) {
            return
        }
        remoteStreamDict[userId!] = stream
        let remoteView = remoteViewDict[userId!]
        if (remoteView == nil) {
            return
        }
        if let videoTrack = stream.videoTracks.first {
            ShowLogUtil.verbose("video track found.")
            videoTrack.add(remoteView!)
        }
        if let audioTrack = stream.audioTracks.first{
            ShowLogUtil.verbose("audio track found.")
            audioTrack.source.volume = 8
        }
    }
    
    func peerConnection(_ peerConnection: RTCPeerConnection, didRemove stream: RTCMediaStream) {
    }
    
    func peerConnectionShouldNegotiate(_ peerConnection: RTCPeerConnection) {
    }
    
    func peerConnection(_ peerConnection: RTCPeerConnection, didChange newState: RTCIceConnectionState) {
        if (newState == .disconnected) {
            DispatchQueue.main.async {
                var userId: String?
                for (key, value) in self.peerConnectionDict {
                    if (value == peerConnection) {
                        userId = key
                    }
                }
                if (userId == nil) {
                    return
                }
                Thread(block: {
                    let peerConnection = self.peerConnectionDict[userId!]
                    if (peerConnection != nil) {
                        peerConnection?.close()
                        self.peerConnectionDict.removeValue(forKey: userId!)
                    }
                }).start()
                let remoteView = self.remoteViewDict[userId!]
                if (remoteView != nil) {
                    self.remoteViews.removeSubview(view: remoteView!)
                    self.remoteViewDict.removeValue(forKey: userId!)
                }
                self.remoteStreamDict.removeValue(forKey: userId!)
            }
        }
    }
    
    func peerConnection(_ peerConnection: RTCPeerConnection, didChange newState: RTCIceGatheringState) {
    }
    
    func peerConnection(_ peerConnection: RTCPeerConnection, didGenerate candidate: RTCIceCandidate) {
//        ShowLogUtil.verbose("didGenerate candidate--->\(candidate)")
        var userId: String?
        for (key, value) in self.peerConnectionDict {
            if (value == peerConnection) {
                userId = key
            }
        }
        if (userId == nil) {
            return
        }
        self.sendIceCandidate(iceCandidate: candidate, toUserId: userId!)
    }
    
    func peerConnection(_ peerConnection: RTCPeerConnection, didRemove candidates: [RTCIceCandidate]) {
    }
    
    func peerConnection(_ peerConnection: RTCPeerConnection, didOpen dataChannel: RTCDataChannel) {
    }
}

// MARK: - UITextFieldDelegate
extension MultipleDemoViewController: UITextFieldDelegate {
    func textFieldShouldReturn(_ textField: UITextField) -> Bool {
        textField.resignFirstResponder()
        return true
    }
}

// MARK: - WebSocketDelegate
extension MultipleDemoViewController: WebSocketDelegate {
    func onOpen() {
        lbWebSocketState?.text = "WebSocket 已连接"
    }
    
    func onClose() {
        lbWebSocketState?.text = "WebSocket 已断开"
    }
    
    func onMessage(message: String) {
        do {
            let data = message.data(using: .utf8)
            let jsonObject: [String : Any] = try JSONSerialization.jsonObject(with: data!) as! [String : Any]
            let msgType = jsonObject["msgType"] as? String
            if ("sdp" == msgType) {
                let type = jsonObject["type"] as? String;
                if ("offer" == type) {
                    receivedOffer(jsonObject: jsonObject);
                } else if ("answer" == type) {
                    receivedAnswer(jsonObject: jsonObject);
                }
            } else if ("iceCandidate" == msgType) {
                receivedCandidate(jsonObject: jsonObject);
            } else if ("otherJoin" == msgType) {
                receiveOtherJoin(jsonObject: jsonObject)
            } else if ("otherQuit" == msgType) {
                receiveOtherQuit(jsonObject: jsonObject)
            }
        } catch {
        }
    }
}

其中 UIScrollView 的 appendSubView 和 removeSubView 是我为 UIScrollView 添加的两个扩展方法,方便纵向添加和删除控件:

import UIKit

extension UIScrollView {
    func appendSubView(view: UIView) {
        let oldShowsHorizontalScrollIndicator = showsHorizontalScrollIndicator
        let oldShowsVerticalScrollIndicator = showsVerticalScrollIndicator
        showsHorizontalScrollIndicator = false
        showsVerticalScrollIndicator = false
        var y = 0.0
        if (subviews.count == 0) {
            y = 0
        } else {
            for i in 0..<subviews.count {
                if ("_UIScrollViewScrollIndicator" == String(reflecting: type(of: subviews[i]))){
                    continue
                }
                y += subviews[i].frame.height
            }
        }
        view.frame.origin.y = y
        addSubview(view)
        let contentSizeWidth = contentSize.width
        // 重新计算 UIScrollView 内容高度
        var contentSizeHeight = 0.0
        for i in 0..<subviews.count {
            if ("_UIScrollViewScrollIndicator" == String(reflecting: type(of: subviews[i]))){
                continue
            }
            contentSizeHeight += subviews[i].frame.height
        }
        contentSize = CGSize(width: contentSizeWidth, height: contentSizeHeight)
        showsHorizontalScrollIndicator = oldShowsHorizontalScrollIndicator
        showsVerticalScrollIndicator = oldShowsVerticalScrollIndicator
    }
    
    func removeSubview(view: UIView) {
        let oldShowsHorizontalScrollIndicator = showsHorizontalScrollIndicator
        let oldShowsVerticalScrollIndicator = showsVerticalScrollIndicator
        showsHorizontalScrollIndicator = false
        showsVerticalScrollIndicator = false
        var index = -1
        for i in 0..<subviews.count {
            if (subviews[i] == view) {
                index = i
                break
            }
        }
        if (index == -1) {
            return
        }
        for i in index+1..<subviews.count {
            subviews[i].frame.origin.y = subviews[i].frame.origin.y-view.frame.height
        }
        view.removeFromSuperview()
        let contentSizeWidth = contentSize.width
        // 重新计算 UIScrollView 内容高度
        var contentSizeHeight = 0.0
        for i in 0..<subviews.count {
            if ("_UIScrollViewScrollIndicator" == String(reflecting: type(of: subviews[i]))){
                continue
            }
            contentSizeHeight += subviews[i].frame.height
        }
        contentSize = CGSize(width: contentSizeWidth, height: contentSizeHeight)
        showsHorizontalScrollIndicator = oldShowsHorizontalScrollIndicator
        showsVerticalScrollIndicator = oldShowsVerticalScrollIndicator
    }
}

好了,现在三端都实现了,我们可以来看看效果了。

七、效果展示

运行 MultipleWebSocketServerHelper 的 main() 方法,我们可以看到服务端已经开启,然后我们依次将 H5、Android、iOS 连接 WebSocket,再依次加入房间:

其中 iOS 在录屏的时候可能是系统限制,画面静止了,但其实跟另外两端是一样的,从另外两端的远端画面可以看到 iOS 是有在采集摄像头画面的。

八、总结

实现完成后可以感觉到多人呼叫其实也没有多难,跟点对点 Demo 的流程大致一样,只是我们需要重新定义创建 PeerConnection 的时机,但是流程仍然是不变的。以及信令有些许不同,信令这就是业务层面的,自己按需来设计,上面我定义的消息格式只是一个最简单的实现。

至此,WebRTC 单人和多人通话的 Demo 全部完成,这就说明它能满足我们基本的视频通话、视频会议等需求,至于丢包处理、美颜滤镜就是网络优化、图像处理相关的了,后续还会记录网络穿透如何去做,以及使用 WebRTC 时的一些小功能,比如屏幕录制、图片投屏、白板等这些视频会议常用的功能。

九、Demo

Demo 传送门

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/1082375.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

Linux开启SSH

Linux开启SSH 1.虚拟机确定连通性 如果是虚拟机的话则需要进行确定和宿主主机之间能正常联通(不能联通还远程个啥) 获取到虚拟机的IP 参考文章:Linux获取本机IP地址使用宿主机ping一下虚拟机的IP查看是否联通 2.安装SSH服务端 安装工具来使得能够通过SSH进行连接 命令 sudo a…

【推荐系统】推荐系统(RS)与大模型(LLM)的结合

【推荐系统】推荐系统&#xff08;RS&#xff09;与大模型&#xff08;LLM&#xff09;的结合 文章目录 【推荐系统】推荐系统&#xff08;RS&#xff09;与大模型&#xff08;LLM&#xff09;的结合1. 主流的推荐方法2. 大模型&#xff08;LLM&#xff09;可能作用的地方 1. 主…

Spring源码解析——ApplicationContext容器refresh过程

正文 在之前的博文中我们一直以BeanFactory接口以及它的默认实现类XmlBeanFactory为例进行分析&#xff0c;但是Spring中还提供了另一个接口ApplicationContext&#xff0c;用于扩展BeanFactory中现有的功能。 ApplicationContext和BeanFactory两者都是用于加载Bean的&#x…

graphviz 绘制单链表

dot 代码 digraph LinkedList {rankdirLR; // 设置布局方向为从左到右&#xff08;左侧到右侧&#xff09;node [fontname"Arial", shaperecord, stylefilled, color"#ffffff", fillcolor"#0077be", fontsize12, width1.5, height0.5];edge [fo…

汉诺塔问题:递归

经典汉诺塔问题 汉诺塔问题是经典的可以用递归解决的问题。 汉诺塔(Hanoi)游戏规则如下&#xff1a;在一块铜板装置上&#xff0c;有三根杆(编号A、B、C)&#xff0c;在A杆自下而上、由大到小按顺序放置64个金盘(如下图)。游戏的目标&#xff1a;把A杆上的金盘全部移到C杆上&a…

系统架构师备考倒计时25天(每日知识点)

面向对象设计原则 单一职责原则&#xff1a;设计目的单一的类开放-封闭原则&#xff1a;对扩展开放&#xff0c;对修改封闭李氏(Liskov)替换原则&#xff1a;子类可以替换父类依赖倒置原则&#xff1a;要依赖于抽象&#xff0c;而不是具体实现&#xff1b;针对接口编程&#x…

【Redis】Redis持久化深度解析

原创不易&#xff0c;注重版权。转载请注明原作者和原文链接 文章目录 Redis持久化介绍RDB原理Fork函数与写时复制关于写时复制的思考 RDB相关配置 AOF原理AOF持久化配置AOF文件解读AOF文件修复AOF重写AOF缓冲区与AOF重写缓存区AOF缓冲区可以替代AOF重写缓冲区吗AOF相关配置写后…

机器学习之自训练协同训练

前言 监督学习往往需要大量的标注数据&#xff0c; 而标注数据的成本比较高 &#xff0e; 因此 &#xff0c; 利用大量的无标注数据来提高监督学习的效果有着十分重要的意义&#xff0e; 这种利用少量标注数据和大量无标注数据进行学习的方式称为 半监督学习 &#xff08; Semi…

Java 序列化和反序列化为什么要实现 Serializable 接口?

序列化和反序列化 序列化&#xff1a;把对象转换为字节序列的过程称为对象的序列化. 反序列化&#xff1a;把字节序列恢复为对象的过程称为对象的反序列化. 什么时候需要用到序列化和反序列化呢或者对象序列化的两种用途… &#xff1a; (1) 对象持久化&#xff1a;把对象的…

NSSCTF做题(8)

[SWPUCTF 2022 新生赛]js_sign 看到了js代码 有一个base64编码&#xff0c;解密 最后发现这是一个加密方式 去掉空格之后得到了flag NSSCTF{youfindflagbytapcode} [MoeCTF 2022]baby_file 提示说有一个秘密看看你能不能找到 输入?filesecret 出现报错 输入php伪协议读取i…

Simulink仿真之离散系统

最近&#xff0c;为了完成课程作业&#xff0c;需要用到Simulink验证数字控制器的合理性。题目如下所示。 其实这道题在胡寿松老师的《自动控制原理&#xff08;第七版&#xff09;》的364页有答案。 这里给出数字控制器的脉冲传递函数为 ​​​​​​​ ​​​​​​​…

【22】c++设计模式——>外观模式

外观模式定义 为复杂系统提供一个简化接口&#xff0c;它通过创建一个高层接口(外观)&#xff0c;将多个子系统的复杂操作封装起来&#xff0c;以便客户端更容易使用。 简单实现 #include<iostream>// 子系统类 class SubsystemA { public:void operationA() {std::co…

经典循环命题:百钱百鸡

翁五钱一只&#xff0c;母三钱&#xff0c;小鸡三只一钱&#xff1b;百钱百鸡百鸡花百钱。 (本笔记适合能熟练应用for循环、会使if条件分支语句、能格式化字符输出的 coder 翻阅) 【学习的细节是欢悦的历程】 Python 官网&#xff1a;https://www.python.org/ Free&#xff1a…

自定义表单工具好用的优点是什么?

如果想提升办公效率&#xff0c;那么就离不开低代码技术平台了。它的轻量级、易掌握、易操作、简洁简便等优势特点深得很多领域客户朋友的喜爱。目前&#xff0c;IBPS开发平台在通信业、制造业、医疗、高校等很多行业中得到了客户的肯定和喜爱&#xff0c;推广价值高&#xff0…

懒人福利:不用动脑就能制作电子画册

对于很多企业来说&#xff0c;想要快速把自己的活动大面积的宣传出去&#xff0c;就要快人一步提前制作电子版画册&#xff0c;通过网络推送出去&#xff0c;让大众及时了解。如何制作电子版画册呢&#xff1f; 我发现了一个懒人福利&#xff01;就是FLBOOK &#xff0c;它简单…

学习函数式编程、可变参数及 defer - GO语言从入门到实战

函数是⼀等公⺠、学习函数式编程、可变参数及 defer - GO语言从入门到实战 函数是⼀等公⺠ 在Go语言中&#xff0c;函数可以分配给一个变量&#xff0c;可以作为函数的参数&#xff0c;也可以作为函数的返回值。这样的行为就可以理解为函数属于一等公民。 与其他主要编程语⾔…

Nodejs内置模块process

文章目录 内置模块process写在前面1. arch()2. cwd()3. argv4. memoryUsage()5. exit()6. kill()7. env【最常用】 内置模块process 写在前面 process是Nodejs操作当前进程和控制当前进程的API&#xff0c;并且是挂载到globalThis下面的全局API。 下面是process的一些常用AP…

linux开发板中的数据存储和读取操作

问题&#xff1a;MQTT远程下发的参数存储在本地linux开发板&#xff0c;开发板依据该参数执行相应功能&#xff0c;当开发板重新上电时依然能继续执行该功能 解决方式&#xff1a; 在linux板中写一个标识并存储为文件&#xff0c;依据读取的文件标识执行相应的功能 1)打开文…

【2023】M1/M2 Mac 导入Flac音频到Pr的终极解决方案

介绍 原作者链接&#xff1a;https://github.com/fnordware/AdobeOgg 很早之前就发现了这个插件&#xff0c;超级好用&#xff0c;在windows上完全没有问题&#xff0c;可惜移植到mac就不行了&#xff08;然后我给作者发了一个Issue&#xff0c;后来就有大佬把m1的编译出来了&…

Ai绘画描述词 关键词大全 真人美女 二次元卡通美女 国漫动漫效果

超好看的二次元动漫美少年 都是用Ai工具直接绘画生成出来的。 还有真人绘画&#xff0c;效果也很逼真 还有更多的场景效果 寂静天空素材 做抖音 短视频 精美的图片素材都是通过Ai绘画的 但是绘画需要关键词去描述场景&#xff0c;Ai才能自动根据场景描述词生成出来图片。 如何…