我需要在我的网络项目上实现聊天。如何在一个页面上实现它 - 有很多关于它的文章。但我需要有能力: 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对。当我删除此客户端代码时 - 未调用该对。如何正确实现,通知其他用户,有人在线,但每次都没有重新连接?
答案 0 :(得分:1)
由于您在razor(_layout.cshtml)页面中进行了connect()调用,因此您将有效地断开/连接每个回发(页面加载)。
您尝试使用的方案最适用于SPA(或AJAX)方案,其中导航由客户端JavaScript异步处理。您当前的设置是刷新屏幕,重新加载JavaScript,在每个服务器渲染导航时重新运行document ready()函数。
另一种方法是使用客户端的实际用户ID,并将其传递给server.connect(id)方法。然后使用此用户ID来跟踪用户活动,而不是集线器ConnectionId。