Planet
navi homePPSaboutscreenshotsdownloaddevelopmentforum

source: code/trunk/src/libraries/network/Client.cc @ 10328

Last change on this file since 10328 was 9667, checked in by landauf, 11 years ago

merged core6 back to trunk

  • Property svn:eol-style set to native
File size: 6.3 KB
Line 
1/*
2 *   ORXONOX - the hottest 3D action shooter ever to exist
3 *                    > www.orxonox.net <
4 *
5 *
6 *   License notice:
7 *
8 *   This program is free software; you can redistribute it and/or
9 *   modify it under the terms of the GNU General Public License
10 *   as published by the Free Software Foundation; either version 2
11 *   of the License, or (at your option) any later version.
12 *
13 *   This program is distributed in the hope that it will be useful,
14 *   but WITHOUT ANY WARRANTY; without even the implied warranty of
15 *   MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
16 *   GNU General Public License for more details.
17 *
18 *   You should have received a copy of the GNU General Public License
19 *   along with this program; if not, write to the Free Software
20 *   Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA  02110-1301, USA.
21 *
22 *   Author:
23 *      Oliver Scheuss
24 *   Co-authors:
25 *      ...
26 *
27 */
28
29//
30// C++ Implementation: Client
31//
32// Description:
33//
34//
35// Author:  Oliver Scheuss, (C) 2007
36//
37// Copyright: See COPYING file that comes with this distribution
38//
39//
40
41#include "Client.h"
42
43#include <cassert>
44
45#include "util/Clock.h"
46#include "util/Output.h"
47#include "util/ScopedSingletonManager.h"
48#include "synchronisable/Synchronisable.h"
49#include "packet/Chat.h"
50#include "packet/Gamestate.h"
51#include "FunctionCallManager.h"
52#include "core/CoreIncludes.h"
53#include "core/Game.h"
54#include "core/config/CommandLineParser.h"
55
56namespace orxonox
57{
58
59  ManageScopedSingleton( Client, ScopeID::Root, true );
60
61  /**
62  * Constructor for the Client class
63  * initializes the address and the port to default localhost:NETWORK_PORT
64  */
65  Client::Client():
66      isSynched_(false),
67      gameStateFailure_(false),
68      timeSinceLastUpdate_(0)
69  {
70    this->setDestination( CommandLineParser::getValue("dest").get<std::string>(), CommandLineParser::getValue("port") );
71  }
72
73  Client::~Client()
74  {
75    if ( ClientConnection::isConnected() )
76      closeConnection();
77  }
78
79  /**
80  * Establish the Connection to the Server
81  * @return true/false
82  */
83  bool Client::establishConnection()
84  {
85    Synchronisable::setClient(true);
86    if( ClientConnection::establishConnection() )
87    {
88      Host::setActive(true);
89      GamestateManager::addPeer(NETWORK_PEER_ID_SERVER);
90      return true;
91    }
92    else
93      return false;
94  }
95
96  /**
97  * closes the Connection to the Server
98  * @return true/false
99  */
100  bool Client::closeConnection()
101  {
102    Host::setActive(false);
103    GamestateManager::removePeer(NETWORK_PEER_ID_SERVER);
104    return ClientConnection::closeConnection();
105  }
106
107  void Client::setDestination(const std::string& serverAddress, unsigned int port)
108  {
109    ClientConnection::setServerAddress(serverAddress);
110    ClientConnection::setPort(port);
111  }
112
113  void Client::queuePacket(ENetPacket *packet, int clientID, uint8_t channelID)
114  {
115    ClientConnection::addPacket(packet, channelID);
116  }
117
118  void Client::printRTT()
119  {
120    orxout(message) << "Round trip time to server is " << ClientConnection::getRTT() << " ms" << endl;
121  }
122
123  /**
124   * @brief Sends a chat message to the server.
125   * @param message message to be sent
126   * @param sourceID the ID of the sender
127   * @param targetID the ID of the receiver
128   */
129  void Client::doSendChat(const std::string& message, unsigned int sourceID, unsigned int targetID)
130  {
131    // send the message to the server
132    packet::Chat* packet = new packet::Chat(message, sourceID, targetID);
133    packet->send(static_cast<Host*>(this));
134  }
135
136  /**
137   * @brief Gets called if a packet::Chat packet is received. Calls the parent function which passes the message to the listeners.
138   */
139  void Client::doReceiveChat(const std::string& message, unsigned int sourceID, unsigned int targetID)
140  {
141    // call the parent function which passes the message to the listeners
142    Host::doReceiveChat(message, sourceID, targetID);
143  }
144
145  /**
146   * Processes incoming packets, sends a gamestate to the server and does the cleanup
147   * @param time
148   */
149  void Client::update(const Clock& time)
150  {
151    //this steers our network frequency
152    timeSinceLastUpdate_+=time.getDeltaTime();
153    if(timeSinceLastUpdate_>=NETWORK_PERIOD)
154    {
155      timeSinceLastUpdate_ -= static_cast<unsigned int>( timeSinceLastUpdate_ / NETWORK_PERIOD ) * NETWORK_PERIOD;
156      if ( isConnected() && isSynched_ )
157      {
158        orxout(verbose, context::network) << "popping partial gamestate: " << endl;
159//         packet::Gamestate *gs = GamestateClient::getGamestate();
160        if( GamestateManager::update() )
161        {
162          std::vector<packet::Gamestate*> gamestates = GamestateManager::getGamestates();
163          std::vector<packet::Gamestate*>::iterator it;
164          for( it = gamestates.begin(); it != gamestates.end(); ++it )
165          {
166            (*it)->send( static_cast<Host*>(this) );
167          }
168        }
169        //assert(gs); <--- there might be the case that no data has to be sent, so its commented out now
170//         if(gs){
171//           orxout(verbose, context::network) << "client tick: sending gs " << gs << endl;
172//           if( !gs->send() )
173//             orxout(internal_warning, context::network) << "Problem adding partial gamestate to queue" << endl;
174//         // gs gets automatically deleted by enet callback
175//         }
176        FunctionCallManager::sendCalls(static_cast<Host*>(this));
177      }
178    }
179//     sendPackets(); // flush the enet queue
180
181    Connection::processQueue();
182    if(GamestateManager::processGamestates())
183    {
184      FunctionCallManager::processBufferedFunctionCalls();
185      if(!isSynched_)
186        isSynched_=true;
187    }
188//     GamestateManager::cleanup();;
189//     Connection::sendPackets();
190
191    return;
192  }
193
194  void Client::connectionClosed()
195  {
196    ObjectList<Synchronisable>::iterator it;
197    for(it = ObjectList<Synchronisable>::begin(); it; )
198    {
199      if( it->getSyncMode() != 0x0 )
200        (it++)->destroy();
201      else
202      {
203        ++it;
204      }
205    }
206    Game::getInstance().popState();
207    Game::getInstance().popState();
208  }
209
210  void Client::processPacket(packet::Packet* packet)
211  {
212    if( packet->isReliable() )
213    {
214      if( this->getLastReceivedGamestateID(packet->getPeerID()) >= packet->getRequiredGamestateID() )
215        packet->process(static_cast<Host*>(this));
216      else
217        this->packetQueue_.push_back(packet);
218    }
219    else
220      packet->process(static_cast<Host*>(this));
221  }
222}
Note: See TracBrowser for help on using the repository browser.