This next example starts with the code from the peer-to-peer solution and converts it to a client/server-based network. The main difference with this example is that the clients send their data directly to the server rather than to all of the other clients. The server then distributes the entire collection of data to the clients.
To start, an extra class is needed in Network.cs to store the alien position, view, and identification:
public class Alien{
public Vector3 position;
public Vector3 view;
public int alienID;
public Alien() { }
public Alien(int alienNum){
alienID = alienNum;
} }
The server collects all of the remote client and local data and stores it in a list, so a list declaration is needed in theXNANetworkclass:
public List<Alien> alienData = new List<Alien>();
A new version ofGamerJoinEvent()stores the instance of each new gamer lo- cally as each new player joins the game. The structure,e.Gamer.Tag, stores each new gamer’s identity. e.Gamer.Tag will be referenced later during reads and writes to identify the gamer data. Each new gamer is added to theXNANetworklist.
To add this code, replaceGamerJoinEvent()with this new version:
void GamerJoinEvent(object sender, GamerJoinedEventArgs e){
int gamerIndex = session.AllGamers.IndexOf(e.Gamer);
C H A P T E R 2 9
Networking
e.Gamer.Tag = new Alien(gamerIndex);
Alien tempAlien = new Alien();
tempAlien.alienID = gamerIndex;
alienData.Add(tempAlien);
}
ClientWrite()belongs in theXNANetworkclass to write local data from the client to the network:
void ClientWrite(LocalNetworkGamer gamer,
Vector3 localPosition, Vector3 localView){
Alien localAlien = gamer.Tag as Alien;
// find local players in list and write their data to the network for (int i = 0; i < alienData.Count; i++){
if (alienData[i].alienID == localAlien.alienID && gamer.IsLocal){
Alien tempAlien = new Alien();
tempAlien.alienID = localAlien.alienID;
tempAlien.position = localPosition;
tempAlien.view = localView;
alienData[i] = tempAlien;
// Write our latest input state into a network packet.
packetWriter.Write(alienData[i].alienID);
packetWriter.Write(localPosition);
packetWriter.Write(localView);
} }
// Send our input data to the server.
gamer.SendData(packetWriter,
SendDataOptions.InOrder, session.Host);
}
The routine that writes data packets from the server,ServerWrite(), is differ- ent thanClientWrite().ServerWrite()sends local data to the network and also distributes all data generated on other clients as one collection.
ServerWrite()must be placed inside theXNANetworkclass to perform this data transfer:
void ServerWrite(Vector3 localPosition, Vector3 localView){
// iterate through all local and remote players
M I C R O S O F T X N A G A M E S T U D I O C R E A T O R ’ S G U I D E
522
523
foreach (NetworkGamer gamer in session.AllGamers){
Alien alien = gamer.Tag as Alien;
for (int i = 0; i < alienData.Count; i++){
// update local data only
if (gamer.IsLocal && alienData[i].alienID == alien.alienID){
Alien tempAlien = new Alien();
tempAlien.alienID = alien.alienID;
tempAlien.position = localPosition;
tempAlien.view = localView;
alienData[i] = tempAlien;
}
// write data for all players
packetWriter.Write(alienData[i].alienID);
packetWriter.Write(alienData[i].position);
packetWriter.Write(alienData[i].view);
} }
// send all data to everyone on session
LocalNetworkGamer server = (LocalNetworkGamer)session.Host;
server.SendData(packetWriter, SendDataOptions.InOrder);
}
Next, addServerRead()to theXNANetworkclass to read all remote data and store it locally in the list:
void ServerRead(LocalNetworkGamer gamer){
// read all incoming packets while (gamer.IsDataAvailable){
NetworkGamer sender;
// read single packet
gamer.ReceiveData(packetReader, out sender);
// store remote data only if (!sender.IsLocal){
int tag = packetReader.ReadInt32();
remotePosition = packetReader.ReadVector3();
remoteView = packetReader.ReadVector3();
for (int i = 0; i < alienData.Count; i++){
C H A P T E R 2 9
Networking
M I C R O S O F T X N A G A M E S T U D I O C R E A T O R ’ S G U I D E
524
if (alienData[i].alienID == tag){
Alien tempAlien = new Alien();
tempAlien.alienID = tag;
tempAlien.position = remotePosition;
tempAlien.view = remoteView;
alienData[i] = tempAlien;
} } } } }
Then, addClientRead()to read all data from the network and to store the re- mote data in the alien data list:
void ClientRead(LocalNetworkGamer gamer){
while (gamer.IsDataAvailable){
NetworkGamer sender;
// read single packet
gamer.ReceiveData(packetReader, out sender);
int gamerId = packetReader.ReadInt32();
Vector3 pos = packetReader.ReadVector3();
Vector3 view = packetReader.ReadVector3();
// get current gamer id
NetworkGamer remoteGamer = session.FindGamerById(gamer.Id);
// don't update if gamer left game if (remoteGamer == null)
return;
Alien alien = remoteGamer.Tag as Alien;
// search all aliens and find match with remote ones for (int i = 0; i < alienData.Count; i++){
if (alienData[i].alienID == gamerId) { Alien tempAlien = new Alien();
tempAlien.alienID = gamerId;
tempAlien.position = pos;
525
tempAlien.view = view;
alienData[i] = tempAlien;
remotePosition = alienData[i].position;
remoteView = alienData[i].view;
} } } }
A revisedUpdateNetwork()method must replace the existing one to handle the client/server processing. If the session is in progress, this method triggers read and write routines on the client and the server:
public void UpdateNetwork(Vector3 localPosition, Vector3 localView){
// ensure session has not ended if (session == null)
return;
// read incoming network packets.
foreach (LocalNetworkGamer gamer in session.LocalGamers) if (gamer.IsHost)
ServerRead(gamer);
else
ClientRead(gamer);
// write from clients if (!session.IsHost)
foreach (LocalNetworkGamer gamer in session.LocalGamers) ClientWrite(gamer, localPosition, localView);
// write from server else
ServerWrite(localPosition, localView);
// update session object session.Update();
}
When you run your code now, your client/server network will allow you to con- trol two different aliens, each on its own machine. With either the peer-to-peer framework or the client/server framework, you have a performance-friendly way to exchange data between machines in your game as long as you design the game for ef- ficient data transfer.
C H A P T E R 2 9
Networking