websocket入门(3)——网络对战版五子棋交互逻辑

2018-10-09  本文已影响0人  淡淡紫色

环境

  1. 服务器端使用Nodejs6.9.4 + Express4.x + Socket.IO 2.02
  2. 客户端使用socket.io.js

名词和变量

游戏大厅

1. 建立连接后就认为在游戏大厅中;
2. 断开连接(比如关闭浏览器或者其他),认为离开游戏大厅;
3. 即使在【游戏房间】中,也依然认定其在游戏大厅;
4. 通过服务器端onlineUsers(Map结构)来存储在线用户(作为key);

游戏房间

1. 创建游戏后,创建的内容就是游戏房间;
2. 游戏房间变量存储以下内容:
2.1 玩家player存储黑方black,白方white;
2.2 观战者watcher(Set结构);
2.3 房间创建时间ctime(Number类型);
2.4 下棋数据steps(Array);
2.5 棋盘大小size(Number);
2.6 游戏结束标志roomover(Boolean);
2.7 游戏房间的编号roomID(Number);

在线用户onlineUsers(服务器端)

1. 以userSocket作为key,存储在线用户,以及该用户当前的游戏信息(房间号 + 角色);
2. 用户建立连接时,将其加入;
3. 用户退出房间时,从中删除(并找到其对应的游戏房间,从房间删除),通报该房间所有人;
4. 退出游戏大厅时,读取到该用户所在游戏信息。如果有则调用3
5. Map结构;
6. key是userSocket;
7. value是对象;
7.1 属性id是游戏ID;
7.2 属性role是游戏角色

开启的房间roomlist(服务器端)

1. Map结构,以roomID作为key,存储该房间的信息;
2. 游戏房间的具体数据查看关键词【游戏房间】;
3. 可以通过.size获取到当前已开启的游戏房间的房间数

1、服务器端

1.1、建立长连接connection

  1. 用户建立创长连接,回调函数出现socket变量;
  2. 将socket作为key,存储在onlineUsers中,值为null;

1.2、退出长连接disconnect

  1. 用户退出长连接,触发”disconnect”;
  2. 调用【退出游戏房间】相关逻辑;

1.3、创建游戏房间createRoom

  1. 检查该用户是否在房间中,如果在则禁止创建;
  2. 随机创建一个不重复的房间ID;
  3. 初始化该房间信息;
  4. 将该房间信息和房间ID,以kv形式添加到roomlist中;
  5. 返回该房间ID;
  6. 调用加入房间的API;(见下面)

1.4、加入房间userEnterRoom

  1. 需要参数roomID,userSocket(用户的socket),role(黑方、白方或者观战,无参数时默认为观战);
  2. 首先通过userSocket去查onlineUsers,看是否有值;
  3. 如果有值,则调用【退出游戏房间】相关逻辑;
  4. 从roomID查看roomlist,是否有该房间;
  5. 如果查不到则通知用户该房间不存在,返回;
  6. 如果找到该房间,首先检查是否有role,如果没有,则将其设置为watcher(观战);
  7. 检查该房间指定role是否有人,如果有人,则将用户添加到watcher位置,并通报用户该位置有人;
  8. 如果没有人,则将用户添加到该位置;
  9. 更新该角色的onlineUser中的值({roomID, role});
  10. 通报该房间内所有用户,有玩家进入,其角色为role;
  11. 通过roomID从list里拉取信息,然后对该房间内所有用户,调用【向用户更新房间信息】逻辑;

1.5、退出游戏房间leaveRoom

  1. 需要参数userSocket;
  2. 以socket作为key,尝试从onlineUsers拉取值;
  3. 如果值不为空,可以从其数据中拉取到房间ID和游戏角色;
  4. 从roomlist拉取到该游戏,删除该角色;
  5. 对该游戏房间中所有玩家,通报该玩家已经退出游戏;
  6. 从onlineUsers删除该用户;
  7. 如果notLeave不为true,对该房间内所有用户,调用【向用户更新房间信息】逻辑;

1.6、将某人踢出游戏房间kickRoom(不做)

  1. 先从发起方,拉取他的socket;
  2. 从socket拉取到roomID;
  3. 从roomID拉取到房间;
  4. 从发起方拉取到踢人的位置;
  5. 删除该角色;
  6. 对该游戏房间中所有玩家,通报该玩家被踢出游戏;
  7. 对该角色通报他被踢出游戏;
  8. 更新该角色的onlineUser中的值(null);

1.7、切换游戏角色changeRoomRole

  1. 需要参数userSocket和role;
  2. 从userSocket获游戏ID和游戏角色currentRole;
  3. 去roomlist找该房间,如果房间不存在则提示,并return(理论上不可能吧?但为了代码健壮性写上);
  4. 找确定该房间该位置是否有人,如果有人则提示,并return;
  5. 将当前用户添加到该位置;
  6. 从之前获取的角色位置,在游戏房间中找到,并删除该userSocket;
  7. 更新该角色的onlineUser中的值(当前有roomID和role);
  8. 向房间内所有用户,通报该用户角色切换;

1.8、向用户更新房间信息updateRoomRoom

  1. 需要参数roomID;
  2. 从roomlist拉取当前游戏房间用户的信息;
  3. 创建对象,里面添加进去黑方白方和观战者的信息;
  4. 遍历游戏房间的黑方、白方和观战方的userSocket,并向他们emit第三步的对象,emit的key是updateRoomRoom;

2、客户端

2.1、基本功能

  1. SETTING:设置棋子大小
  2. ROOMINFO:房间信息,从服务器同步到本地;
  3. USERINFO:目前只存储了用户名;
  4. init函数:初始化函数,当页面加载完成后执行(内含仿jQuery的简化选择器,连接websocket的后端服务器,DOM事件监听,socket.io事件监听);
  5. selector函数:使用document.querySelector和querySelectorAll来简化选择器操作;
  6. listenDomclickEvent函数:监听各种dom的点击事件,比如创建房间啦,加入房间啦,切换角色啦,开始新游戏啦之类;
  7. listenSocketEvent函数:监听服务器发送过来的socket.io事件;
  8. getName函数:生成随机姓名;
  9. getNowDate函数:生成事件戳;
  10. initBoard函数:通过房间信息,更新;
  11. createBox函数:生成棋盘的格子;
  12. createPiece函数:生成棋子;
  13. toMakePieceWhenHasStart函数:如果已经在下棋中,那么需要通过这个方法来生成棋盘上所有的棋子;
  14. resetRoomInfo函数:重置棋盘、房间信息;
  15. checkerboardClick函数:监听下棋下在哪里了;
  16. checkCanPutPiece函数:前端判定是否可以下这一步棋(后端会二次校验);
  17. GameOver函数:游戏结束后的处理;
  18. changeColor函数:切换下棋角色时的提示;

2.2、三种不同的提示

【函数】通过roomID向游戏内所有用户发送通知信息alertToRoom

  1. 当然需要参数roomID,需要参数信息内容;
  2. 通过roomID去从roomlist里找游戏;
  3. 遍历游戏里的每个人,调用emit发送信息,key是alertToRoom;

【函数】通过room去遍历每个用户forEachUserSocketInRoom

  1. 参数是从roomlist里找到的room,以及回调函数;
  2. 遍历每个角色,调用回调函数。回调函数参数一是该位置的userSocket,参数二是role

【消息key】通报给当前用户alertToUser

  1. 参数是对象,{msg:”“}

3、通信关键字

3.1、服务器端socket响应

connection

当用户和服务器建立长连接时触发

disconnect

用户断开链接(主动被动都会触发)

createRoom

用户创建游戏时触发

updateName

用户更新名字

leaveRoom

用户离开房间

userEnterRoom

用户进入房间

speakwords

用户发言(大厅或者room里)

restartRoom

当要求开始新的一局的请求发送时,判断游戏是否结束,黑方和白方是否都在(并且要是其中一方发起的),如果是,则返回信息初始化棋盘

3.2、客户端socket响应

connection-success

建立链接后,服务器端会发送这个过来

disconnect

断线时会触发本事件(无论是主动还是被动),提示用户断线了。

getCurrentName

获取当前姓名,当用户通过updateName推送姓名到服务器后,服务器端响应后会发送这个给客户端

tipsToUser

黑色 字体的系统消息

alertToUser

红色 字体的系统消息

alertToRoom

红色 字体系统消息,对房间内所有人推送时使用。

leaveRoom

蓝色 字体系统消息,用户离开房间时推送离开消息时使用。

chat-words

收到聊天信息时,推送到聊天栏。房间和大厅的聊天都会通过这个来获取

broadcast

广播信息,每个人都会收到。

userINFO

收到当前用户信息,暂时没用

转自: https://blog.csdn.net/qq20004604/article/details/73835546

上一篇下一篇

猜你喜欢

热点阅读