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

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 服务端代码:

java 复制代码
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 消息:

java 复制代码
// 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 复制代码
<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 复制代码
<?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

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

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 添加的两个扩展方法,方便纵向添加和删除控件:

Swift 复制代码
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 传送门

相关推荐
加油吧x青年9 小时前
Web端开启直播技术方案分享
前端·webrtc·直播
Rookie也要加油10 小时前
WebRtc一对一视频通话_New_peer信令处理
笔记·学习·音视频·webrtc
superconvert1 天前
主流流媒体的综合性能大 PK ( smart_rtmpd, srs, zlm, nginx rtmp )
websocket·ffmpeg·webrtc·hevc·rtmp·h264·hls·dash·rtsp·srt·flv
staritstarit5 天前
通过LiveGBS实现安防监控摄像头GB28181转成WebRTC流实现web浏览器网页无插件低延迟直播...
webrtc
JeasonTly5 天前
WebRTC服务器搭建
运维·服务器·webrtc
DogDaoDao5 天前
音视频开发常见的开源项目汇总
ffmpeg·开源·音视频·webrtc·x264·live555·obs
Likeadust10 天前
EasyCVR视频汇聚平台:巧妙解决WebRTC无法播放H.265视频的难题
音视频·webrtc·h.265
LCRxxoo10 天前
Chrome 本地调试webrtc 获取IP是xxx.local
前端·chrome·webrtc
Wu Youlu12 天前
WebRTC 代码实现详述
webrtc