Asynchronous multiplayer games are awesome because players can enjoy social gaming experiences without having to commit to long sit-down sessions. This convenient style of play is made possible by online servers that deliver updates on games being played directly to a player’s device. This article will explain how games previously saved to a database are loaded by the server and made accessible within the game client’s user interface.
Hero Mages Asynchronous Multiplayer Preview
This article will explain:
- How to query a list of game records stored in a MySQL database and send the results set to the game client
- How to interpret the query results on the client side and design a meaningful games list user interface the player can use to resume playing
- How to recreate a live and synchronous multiplayer experience from an async game
- How to replay animations to represent the moves made by the player’s opponent while they were away
Requirements |
|
Generating the Player’s Games List
Popular async games like Hero Academy utilize a “games list” user interface that allow players to access and resume playing their asynchronous game sessions.
When building my games list for Hero Mages, I started by creating a new user interface screen class called “HM_GamesList”. I wanted to focus on the data and code components first, so the initial design is limited to a header and a scrollable list component that will be used to populate information retrieved from the server:
This interface is populated by a database query that gets a list of the active player’s game records. All of the MySQL query generation takes place on the server side, and the process for communicating with our online server looks like this:
- Game Client: Request data from server
- Server: Handle request, send client response
- Game Client: Receive server response
- Game Client: Perform desired task with data
Step 1: Request Game List
The game client requests the game list from the server:
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 | //CODE EXAMPLE 1: Request Game List from Server private function getGameList(lowerLimit:int){ //Create a new object to send command parameters var params = new Object(); //Pass the player's unique member id params.pId = pId; //Show user prompt while waiting for response showPrompt("ProcessingRequestPrompt"); //Send Smart Fox Server an extension message /* sendXtMessage(xtName:String, cmd:String, paramObj:*, type:String = "xml") xtName = Name of your server side extension cmd = Unique identifier name for this command paramObj = Object contain parameters for command type = Indicates whether we're sending as XML or raw string */ smartFox.sendXtMessage("HMServer", "Game List", params, "xml"); } |
Step 2: Handle Game List Request
The server side code handles the request and sends the client back a response. In Part 2: Saving the Game State to Online Database, I explain how games are saved to a MySQL database using two tables, hm_games and hm_gameresults. The function loadGameList will build a MySQL query that returns the relevant data we need to populate the game list.
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 | //CODE EXAMPLE 2: Handle Game List Request on Server function handleRequest(cmd, params, user, fromRoom, protocol){ if(protocol == "xml"){ //....CODE FOR OTHER EXTENSION COMMANDS OMITTED.... else if(cmd == "Game List"){ if(params.hmId != null){ //THE FOLLOWING MYSQL STATEMENT GATHERS A LIST OF GAMES PLAYER HAS PLAYED BY JOINING //THE GAME AND GAME RESULTS TABLES CREATED IN PART 2 var sql = "SELECT ID_GAME from hm_games JOIN hm_gameresults using (ID_GAME) WHERE ID_MEMBER =1"+params.hmId; //WE CREATE AN ARRAY TO STORE THE GAME LIST var gameList = []; //WE EXECUTE THE QUERY var queryRes = dbase.executeQuery(sql); //IF THE QUERY RETURNS RESULTS, POPULATE TO ARRAY if(queryRes != null && queryRes.size() > 0){ for(var i = 0; i < queryRes.size(); i++){ //GET THE ACTIVE ROW var dataRow = queryRes.get(i); //CREATE GAME RECORD OBJECT var gameRecord = {}; //STORE THE GAME ID IN THE RECORD gameRecord.ID_GAME = dataRow.getItem("ID_GAME"); //ADD RECORD TO ARRAY gameList.push(gameRecord); } } //STORE THE GAME LIST IN THE SERVER RESPONSE response.gameList = getGameList(params.hmId); } } //....CODE FOR OTHER EXTENSION COMMANDS OMITTED.... } } |
Step 3: Receive Game List Response
The game client receives the server response.
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 | //CODE EXAMPLE 3: Receive Game List from Server private function onExtensionResponse(evt:SFSEvent):void{ //EXTRACT RESPONSE TYPE AND RESPONSE DATA FROM SFSEvent var type:String = evt.params.type; var dataObj:Object = evt.params.dataObj; //....CODE OMITTED.... //EXTRA COMMAND FROM RETURNED DATA OBJECT cmd= dataObj.cmd; var error:String = dataObj.error; //....CODE OMITTED.... if(error != ""){ //IF RESPONSE RETURNS AN ERROR, SHOW USER A MESSAGE PROMPT showPrompt("HM_MessagePrompt", cmd + " Error", error); } else{ //....CODE OMITTED.... //ADD CONDITION FOR SERVER RESPONSE CREATE GAME RECORD else if(cmd == "Game List"){ //HIDE OUR PROCESSING REQUEST PROMPT hidePrompt(); //INSRUCT OUR GAME LIST CLASS TO RECEIVE THE LIST gameList.receiveGameList(dataObj); } //....CODE OMITTED.... } } |
Step 4: Populate Game List
The game client performs the desired task of populating the list:
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 | //CODE EXAMPLE 4: Populate Game List private function receiveGameList(gameList:Object):void{ //The game list is returned from server as array var gameList:Array = dataObj.gameList; //Create a new data provider to store the list var dp:DataProvider = new DataProvider(); //Iterate through the list to add new items to data provider for(var i:int = 0; i < gameList.length; i++){ var gameRecord:Object = gameList[i]; //Add a label property to object so it shows up in list cell gameRecord.label = gameRecord.ID_GAME; //Add item to data provider dp.addItem(gameRecord); } //Set our UI list's data provider list.dataProvider = dp; } |
And here’s our result:
Advanced Game List Query
While functional, the basic game list created above lacks key information needed for a solid user experience. Players need to know when the game was created, when the last turn was, whose turn it is, and most importantly who their opponents are for the game to be loaded.
Merging the Games and Games Results Tables
The ideal query needs to return all of the relevant information back to the client using the least resources and bandwidth as possible. This query, designed by Marco Rousonelos, programmer at Reflection Software, and laptop alias of the MySQL help forums uses ranking and derived tables to generate the desired results set:
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 | #CODE EXAMPLE 5: ADVANCED GAME LIST QUERY SELECT IF(whoseTurn = 2 and status != 2, 1, 0) as myTurn, ID_GAME, ID_GAMETYPE, version, timeLastTurn, timeCreated, timeRecorded, status, isAsync, whoseTurn, MAX(CASE WHEN PN = 1 THEN ID_MEMBER ELSE NULL END) AS 'P1ID', MAX(CASE WHEN PN = 1 THEN memberName ELSE NULL END) AS 'P1N', MAX(CASE WHEN PN = 1 THEN result ELSE NULL END) AS 'P1R', MAX(CASE WHEN PN = 2 THEN ID_MEMBER ELSE NULL END) AS 'P2ID', MAX(CASE WHEN PN = 2 THEN memberName ELSE NULL END) AS 'P2N', MAX(CASE WHEN PN = 2 THEN result ELSE NULL END) AS 'P2R' FROM (SELECT g.ID_GAME, g.ID_GAMETYPE, g.version, timeLastTurn, timeCreated, timeRecorded, status, isAsync, whoseTurn, r.ID_MEMBER, r.result, ( CASE g.ID_GAME WHEN @curGame THEN @curRow := @curRow + 1 ELSE @curRow := 1 AND @curGame := g.ID_GAME END ) AS PN FROM hm_games g JOIN hm_gameresults r USING(ID_GAME) JOIN hm_gameresults pg ON g.ID_GAME = pg.ID_GAME AND pg.ID_MEMBER =2 ,(SELECT @curRow := 0, @curGame := -1) n ) data JOIN smf_members m USING(ID_MEMBER) GROUP BY ID_GAME |
With this query we can generate results sets that look like this:
ID_GAME | P1ID | P1N | P1R | P2ID | PD2 | P2R | Status | whoseTurn | … |
1010 | 1 | Ross | 0 | 2 | Kelly | 0 | 1 | 1 | … |
Adding Additional Players
For games that support more than two players, we add additional lines to the query as follows:
1 2 3 4 5 6 7 | #CODE EXAMPLE 6: Additional Player Support MAX(CASE WHEN PN = 3 THEN ID_MEMBER ELSE NULL END) AS 'P3ID', MAX(CASE WHEN PN = 3 THEN memberName ELSE NULL END) AS 'P3N', MAX(CASE WHEN PN = 3 THEN result ELSE NULL END) AS 'P3R', MAX(CASE WHEN PN = 4 THEN ID_MEMBER ELSE NULL END) AS 'P4ID', MAX(CASE WHEN PN = 4 THEN memberName ELSE NULL END) AS 'P4N', MAX(CASE WHEN PN = 4 THEN result ELSE NULL END) AS 'P4R' |
Ordering the Results
Ideally, we’d like to return the results set in the following order:
- Game Status (display games still in progress first)
- Turn (display games where it’s player’s turn first)
- Time Last Update (display games last updated first)
We can achieve this by adding some sort order by statements to the query:
1 2 | #CODE EXAMPLE 7: Order Statement Order by status asc, myTurn desc, timeRecorded desc |
Limiting the Results
It’s important to note that this query returns ALL game record results for the provided member id. As the game becomes popular and player starts to accrue games, this can result in a very large set of data. To ensure the server, network, and the user’s device are not overloaded, it is best to include a LIMIT statement so that only a targeted portion of the results are returned:
1 2 | #CODE EXAMPLE 8: Limit Statement Limit 0, 30 |
Customizing the Query
Queries can be tweaked and customized as needed for inpidual games and controlled by packaging additional parameters in the server request. For example, you could store a “lowerLimit” property and a “limitSpan” property to control the limits of the query.
With a solid query capable of returning the necessary results set, we’re ready to generate a more effective user experience to display the results.
Designing an Asynchronous Multiplayer UI
The player’s game list is the core of the async multiplayer experience. The list is used to navigate, check the status of, and join games in progress. Additionally, the game list is a great leaderboard/logging tool that can be used as a means to review past battles, opponents, and more!
Relevant Game Record Information
A good game list starts with a good game record cell. Each game record should contain the following information for the player:
- Time last turn was taken or that game was completed
- Time game was created
- Status (whether it’s players turn to attack, waiting for turn, defeat, or victor)
- Names of players participating
These properties are helpful to the player in choosing the desired game to load. It’s always possible to add more details, and some other possibilities might include:
- Unique game record id
- Whether or not match is ranked
- Name of the map
- Game objective
Game record cells should ideally be designed to adapt to the size of the list so they can display appropriately on any size mobile device. Hero Mages makes use of player avatars to display character portraits for a more interesting visual display:
Populating the List
Once the game record cell layout has been designed, instances of the game record class can be added to a list component that the player can use to access their game sessions. The process of adding records to the list is similar to adding items to our basic UI list above, except instead of adding simple cells, we’re adding our own custom designed cells.
Hero Mages leverages AURA multiscreen component UIs. AURA stands for Animations, Utilities, and Resources for ActionScript 3.0. It’s a library of classes and components I wrote to speed of tasks like managing listeners, resources, and designing UI. The list seen in the screenshot below is advanced component that adapts to the screen size and input controls of the user’s device. For example, if you’re playing on a touch-based input mobile device, the list is operated using swipes. The same list running on a desktop is navigated using a standard scroll bar. The list is also optimized for mobile GPU’s and can render cells at 60fps on devices like first generation iPads.
The implementation of the component list is beyond the scope of this article. I will be releasing an article that explains the mechanisms of the component that will include downloadable examples in the near future.
Loading a Game
The primary function of the games list is to allow the player to load saved game sessions by selecting an item in the list. Similar to the process of saving games, loading games will require both client-side and server-side code to make a load game request, retrieve game state from the database, and fire up the game engine to restore the desired game. Once again, we follow our 4 communication steps:
- Game Client: Request data from server
- Server: Handle request, send client response
- Game Client: Receive server response
- Game Client: Perform desired task with data
Note: While it would be possible to gather the game state data in the game list query, I recommend using a separate server request for the inpidual game record as outlined below in order to conserve bandwidth.
Step 1: Request Load Game
Each cell in our game list will use the following code to make the request to the server for the desired game record:
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 | //CODE EXAMPLE 9: Client Side Load Game Request //In the Game List constructor, add an event listener to our list for when a cell is clicked public function HM_GameList(){ //...CODE OMITTED list.addEventListener(ListEvent.ITEM_CLICK, gameSelected, false, 0, true); } //The game selected function handles our server request private function gameSelected(evt:Event):void{ //First ensure a valid cell is selected if(list.selectedIndex != -1){ /* It's possible that older games may not be compatible with newer versions of the engine. So, It's a good idea to store the required game version in the game record data. You can write an compatability check function to ensure the version is compatible. */ if(HM_App.isCompatibleVersion(list.selectedItem.v) == false){ HM_Main.HMMain.showPrompt("HM_MessagePrompt", "Version Mismatch", "Your game version '"+ HM_App.appVersion +"' is not compatible with this recorded game's version '" + list.selectedItem.v+"'."); return; } //Once again, create a new params object to store request parameters var params = new Object(); params.gId = list.selectedItem.ID_GAME; //Show our request prompt to the user showPrompt("ProcessingRequestPrompt"); //And send the message to server smartFox.sendXtMessage("HMServer", "Load Game", params, "xml"); } } |
Step 2: Handle Load Game Request
On the server side, we add a condition for “Load Game” request. Since Hero Mages games can also be played synchronously using Smart Fox, why not transform an async game to a live online match if the players are both online? The room loop iterates through the list of rooms on servers and checks to see if any room’s game id matches the record the player is attempting to load asynchronously. If a match is found, the server returns the room id so they can connect immediately with live player. For live matches, players load game data directly from the game room host. If a live match is not found, the server loads the game state from the database.
NOTE: In order for this loop to work, it’s necessary to store the game record id as a room variable when creating a live game room. You do not need to do this if you’re only interested in asynchronous gameplay.
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 | //CODE EXAMPLE 10: Server Side Handle Load Game Request function handleRequest(cmd, params, user, fromRoom, protocol){ if(protocol == "xml"){ //....CODE FOR OTHER EXTENSION COMMANDS OMITTED.... else if(cmd == "Load Game"){ //First, we package our response to include the game record id and a room id response.gId = params.gId; response.rId = -1; //HERE WE WANT TO GO THROUGH LIST OF ACTIVE GAMES AND SEE IF ANY MATCH TARGET GAME ID, IF SO, JOIN THAT ROOM, OTHERWISE, FIRE UP GAME var rooms = _server.getCurrentZone().getRooms(); for(var i = 0; i < rooms.length; i++){ var room = rooms[i]; if(room.getName().indexOf("#"+params.gId) != -1 || (room.getVariable("gId") != null && room.getVariable("gId").getValue() == params.gId)){ response.rId = room.getId(); break; } } //A live room matching the game id was not found, so we need to load the game if(response.rId == -1){ var gameRecord = loadGame(params.gId, response); response.cL = gameRecord.cL; var memberId = user.getVariable("hmId").getValue(); sql = "SELECT lastCmd from hm_gameResults WHERE ID_MEMBER = "+memberId+" and ID_GAME ="+params.gId; queryRes = dbase.executeQuery(sql); dataRow = queryRes.get(0); response.lC = dataRow.getItem("lastCmd"); } } //....CODE FOR OTHER EXTENSION COMMANDS OMITTED.... } } |
The load game function will handle the retrieval of the necessary information for loading the game.
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 | //CODE EXAMPLE 11: Server Side Load Game Function function loadGame(ID_GAME, response){ //Generate a MySQL statement to load the record for the provided game record id sql = "SELECT cmdLog, timeCreated, timeRecorded, timeLastTurn, status from hm_games WHERE ID_GAME = " + ID_GAME; queryRes = dbase.executeQuery(sql); //If the query was unsuccessful, add an error message to the prompt to inform user if(queryRes == null || queryRes.size() <= 0){ response.error = "Unable to load game"; return ""; } else{ //Hero Mages can be played synchronously and asynchronously //Whenever an unfinished game is loaded as an async game, we change isAsync property to reflect if(queryRes.get(0).getItem("status") != 2){ dbase.executeCommand("Update hm_games set isAsync = 1 WHERE ID_GAME = " + ID_GAME); } //Package our response with the cmdLog, which is where we store game state in PART 2 var gameRecord = {}; gameRecord.cL =queryRes.get(0).getItem("cmdLog") return gameRecord; } } |
Step 3: Receive Load Game Response
Back on the client side, our Smart Fox Server extension response listener needs a new condition to listen for “Load Game” response from the server. Depending on the response type, we’ll either join the existing live game or create a new game with the returned parameters:
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 | //CODE EXAMPLE 12: Client Side Load Game Response private function onExtensionResponse(evt:SFSEvent):void{ //....CODE OMITTED.... //ADD CONDITION FOR SERVER RESPONSE LOAD GAME else if(cmd == "Load Game"){ //Hide the waiting for response prompt hidePrompt(); //Check to see if server provided a room id if(dataObj.rId == -1){ //Room id not provided, so we load the game state directly from response object loadGame(dataObj); } else{ /* The server provided a room id. This means there is a live game for this session already created by another player so all we have to do is join the game */ joinRoom(dataObj.rId, "", false); } } //....CODE OMITTED.... } |
Step 4: Load Game State
The load game function will be different for each game engine, but here are some tasks the function will need to handle:
- Convert the returned game state data string back into an object (See Part 2)
- Set two distinct flags “isRunningCommandList” = true and “useAnimations” = false
- Run the list of commands through the engine to effectively “play the game” in the background up to the last command. Your engine code should check the isRunningCommandList flag to ensure any automatic responses to commands (such as counter attack actions) are not fired if they are already included in the command list.
Replaying Animations
Following the above steps will allow you to load any game from your games list and restore the game state so that it matches the last recorded move. Consider, however, that an asynchronous multiplayer opponent will make changes to the game state while the other player is away. Simply loading the current game state will be confusing to the player because they won’t know what commands were carried out by their opponent. For an effective asynchronous multiplayer experience, we need to make some modifications to the command recording process and the game loading code.
Recording the Last Move Witnessed
The last move witnessed must be recorded separately for each player, so we need to add an additional property to the table hm_gameresults first created in Part 2 called “lastCmd”. This property is an integer value designed to store the index of the last command this player witnessed for the game.
When sending new game commands, simply pass the index of the command log along with the command. Then, in our code block for handling updates to the game state (created in Part 2) we’re going to add the following code just below the error response:
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 | //CODE EXAMPLE 13: Storing lastCmd function handleRequest(cmd, params, user, fromRoom, protocol){ if(protocol == "str"){ //GENERATE LIST OF RECIPIENTS THE SERVER WILL SEND THIS COMMAND TO //....CODE OMITTED.... //params[2] stores game record id //If this game record id is included, we need to write this command to stored game log if(params[2] != undefined){ if(params[1].indexOf("cT=eT") != -1){//If this is an end turn command //Convert notated command into object var cmdObj = stringToObject(params[1]+"", "|", "="); //Get the id of player whose turn is next var nextTurnId = cmdObj.nId; //Write update to game record in database sql = "UPDATE hm_games set cmdLog = CONCAT(cmdLog, '<c>" + params[1] + "</c>'), timeRecorded = " + Math.floor(getTimer() / 1000) + ", timeLastTurn = " + Math.floor(getTimer() / 1000) +", whoseTurn = "+nextTurnId+" WHERE ID_GAME = " + params[2]; } else{ //Write update to game record in database sql = "UPDATE hm_games set cmdLog = CONCAT(cmdLog, '<c>" + params[1] + "</c>'), timeRecorded = " + Math.floor(getTimer() / 1000) +" WHERE ID_GAME = " + params[2]; } success = dbase.executeCommand(sql); if(success == false){ //THE DATABASE DID NOT RECORD THE MOVE CORRECTLY //CREATE A NEW RESPONSE TO NOTIFY GAME CLIENT OF THE ERROR } //***NEW CODE BEGIN*** //Get list of all users in this room and update the lastCmd property in game results for everyone who witnessed this move live var lastCmd = cmdObj.lC; //Store the lastCmd to record last witnessed move in live players' gameresult records var allUsers = room.getAllUsers(); for(i = 0; i < allUsers.length; i++){ var memberId = allUsers[i].getVariable("hmId").getValue(); sql = "UPDATE hm_gameResults set lastCmd ="+lastCmd+" WHERE ID_MEMBER = "+memberId+" and ID_GAME ="+gId; trace("GAME RECORD UPDATE: " + sql); dbase.executeCommand(sql); } //***NEW CODE END*** } _server.sendResponse([params[1]], -1, null, recipients, "str"); return; } } |
Show Animations for Unseen Moves
In our load game handler on the server, we’ll add an additional item to the response to store the player’s lastCmd witnessed as follows:
1 2 3 4 5 6 | //CODE EXAMPLE 14: Getting lastCmd var memberId = user.getVariable("hmId").getValue(); sql = "SELECT lastCmd from hm_gameResults WHERE ID_MEMBER = "+memberId+" and ID_GAME ="+params.gId; queryRes = dbase.executeQuery(sql); dataRow = queryRes.get(0); response.lC = dataRow.getItem("lastCmd"); |
Here’s what the complete load game response looks like (combining Step 2: Handle Game Load Request with our new code)
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 | //CODE EXAMPLE 15: Revised Game Load Handler function loadGame(ID_GAME, response){ //Generate a MySQL statement to load the record for the provided game record id sql = "SELECT cmdLog, timeCreated, timeRecorded, timeLastTurn, status from hm_games WHERE ID_GAME = " + ID_GAME; queryRes = dbase.executeQuery(sql); //If the query was unsuccessful, add an error message to the prompt to inform user if(queryRes == null || queryRes.size() <= 0){ response.error = "Unable to load game"; return ""; } else{ //Hero Mages can be played synchronously and asynchronously //Whenever an unfinished game is loaded as an async game, we change isAsync property to reflect if(queryRes.get(0).getItem("status") != 2){ dbase.executeCommand("Update hm_games set isAsync = 1 WHERE ID_GAME = " + ID_GAME); } //Package our response with the cmdLog, which is where we store game state in PART 2 var gameRecord = {}; gameRecord.cL =queryRes.get(0).getItem("cmdLog") //***NEW CODE BEGIN*** //CODE EXAMPLE 14: Getting lastCmd var memberId = user.getVariable("hmId").getValue(); sql = "SELECT lastCmd from hm_gameResults WHERE ID_MEMBER = "+memberId+" and ID_GAME ="+params.gId; queryRes = dbase.executeQuery(sql); dataRow = queryRes.get(0); response.lC = dataRow.getItem("lastCmd"); //***NEW CODE END*** return gameRecord; } } |
With the last command index stored, the trick to playing the appropriate animations is to reactivate the useAnimations flag once the lastCmd index is reached.
Other Considerations
Going beyond the steps I’ve explained, there are many ways to customize the asynchronous experience and add additional features and functionality. For instance, what might happen if a player is given unlimited time to respond to an asynchronous game? A losing player with bad sportsmanship might decide to suspend taking their turn indefinitely, preventing the winning player from claiming victory. One way to solve this problem is to enable a “maxWait” period that allows players to drop opponents if they haven’t taken their turn in so much time. For Hero Mages, I allow players to drop their opponents if they haven’t taken their turn in 3 days.
Other features that might be helpful would be search filters for displaying only active games, finding games against particular opponents, the ability to look up the stats of your opponents, etc.
Coming Next
This article explained the process of building an asynchronous multiplayer game list user interface, loading stored game sessions, and replaying the opponent’s last move animations. The next article will focus on the aspect of the concept of asynchronous multiplayer match-making so that players can start new games or join existing ones without requiring a live online connection.