SignalR - 正确实施聊天

时间:2017-07-23 13:46:28

标签: signalr signalr-hub signalr.client signalr-2

我需要在我的网络项目上实现聊天。如何在一个页面上实现它 - 有很多关于它的文章。但我需要有能力: 1.通知其他用户,有人登录到网站(在任何页面上,不仅在聊天页面上) 2.通知其他用户,有人注销

所以,我有以下代码:

    public void Connect()
    {
        try
        {
            var id = Context.ConnectionId;
            string username = Context.User.Identity.Name;

            var currentUser = connectedUsers.Where(p => p.Username == username).FirstOrDefault();
            if (currentUser == null)
            {
                AddNewUserToCollection();
            }
            else
            {
                // update ConnectionId for sure (connection id is changed sometimes (probably if user is logged out and login again))
                if (currentUser.ConnectionId != id)
                {
                    var companyId = _chatRepository.GetCompanyIdOfUser(username); // throws exception if companyId is null
                    Groups.Remove(currentUser.ConnectionId, companyId.ToString());
                    Groups.Add(id, companyId.ToString());
                    currentUser.ConnectionId = id;
                    //Clients.Group(companyId.ToString()).onNewUserConnected(username);
                }
            }
        }
        catch(InvalidCompanyException c_ex)
        {
            Clients.Client(Context.ConnectionId).onErrorMessage($"User '{c_ex.Username}' does not exist");
        }
    }


    public void Disconnect()
    {
        string username = Context.User.Identity.Name;
        var item = connectedUsers.Where(p => p.Username == username).FirstOrDefault();
        if (item != null)
        {
            connectedUsers.Remove(item);
            Groups.Remove(item.ConnectionId, item.CompanyID.ToString());
            Clients.Group(item.CompanyID.ToString()).onUserDisconnected(item.Username);
        }
    }

    public override Task OnDisconnected(bool stopCalled)
    {
        var item = connectedUsers.Where(p => p.ConnectionId == Context.ConnectionId).FirstOrDefault();
        if (item != null)
        {
            connectedUsers.Remove(item);
            Groups.Remove(item.ConnectionId, item.CompanyID.ToString());
            Clients.Group(item.CompanyID.ToString()).onUserDisconnected(item.Username);
        }
        return base.OnDisconnected(stopCalled);
    }

我将以下代码添加到_layout.cshtml:

    <script>
        $(document).ready(function () {
            var chat = $.connection.chatHub;

            $.connection.hub.start().done(function () {
                chat.server.connect();
            });
        });
    </script>

通知其他用户,记录当前用户。但是调试器说,当用户使用不同的connectionId重新加载页面(通过页面)时,每次调用OnDisconnected / Connect对。当我删除此客户端代码时 - 未调用该对。如何正确实现,通知其他用户,有人在线,但每次都没有重新连接?

1 个答案:

答案 0 :(得分:1)

由于您在razor(_layout.cshtml)页面中进行了connect()调用,因此您将有效地断开/连接每个回发(页面加载)。

您尝试使用的方案最适用于SPA(或AJAX)方案,其中导航由客户端JavaScript异步处理。您当前的设置是刷新屏幕,重新加载JavaScript,在每个服务器渲染导航时重新运行document ready()函数。

另一种方法是使用客户端的实际用户ID,并将其传递给server.connect(id)方法。然后使用此用户ID来跟踪用户活动,而不是集线器ConnectionId。