Planet
navi homePPSaboutscreenshotsdownloaddevelopmentforum

source: code/branches/physics/src/orxonox/objects/worldentities/ControllableEntity.cc @ 2274

Last change on this file since 2274 was 2201, checked in by martisty, 16 years ago

loading objects and creating shapes…

  • Property svn:eol-style set to native
File size: 16.5 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 *      Fabian 'x3n' Landau
24 *   Co-authors:
25 *      ...
26 *
27 */
28
29#include "OrxonoxStableHeaders.h"
30#include "ControllableEntity.h"
31
32#include "core/CoreIncludes.h"
33#include "core/Core.h"
34#include "core/XMLPort.h"
35#include "core/Template.h"
36
37#include "objects/infos/PlayerInfo.h"
38#include "objects/worldentities/Camera.h"
39#include "objects/worldentities/CameraPosition.h"
40#include "overlays/OverlayGroup.h"
41
42namespace orxonox
43{
44    CreateFactory(ControllableEntity);
45
46    ControllableEntity::ControllableEntity(BaseObject* creator) : MovableEntity(creator)
47    {
48        RegisterObject(ControllableEntity);
49
50        this->bControlled_ = false;
51        this->server_overwrite_ = 0;
52        this->client_overwrite_ = 0;
53        this->player_ = 0;
54        this->playerID_ = network::OBJECTID_UNKNOWN;
55        this->hud_ = 0;
56        this->camera_ = 0;
57        this->bDestroyWhenPlayerLeft_ = false;
58
59        this->velocity_ = Vector3::ZERO;
60        this->acceleration_ = Vector3::ZERO;
61
62        this->server_position_ = Vector3::ZERO;
63        this->client_position_ = Vector3::ZERO;
64        this->server_velocity_ = Vector3::ZERO;
65        this->client_velocity_ = Vector3::ZERO;
66        this->server_orientation_ = Quaternion::IDENTITY;
67        this->client_orientation_ = Quaternion::IDENTITY;
68
69        this->registerVariables();
70    }
71
72    ControllableEntity::~ControllableEntity()
73    {
74        if (this->isInitialized())
75        {
76            if (this->bControlled_)
77                this->stopLocalControl();
78
79            if (this->hud_)
80                delete this->hud_;
81
82            if (this->camera_)
83                delete this->camera_;
84
85            if (this->getPlayer() && this->getPlayer()->getControllableEntity() == this)
86                this->getPlayer()->stopControl(this, false);
87        }
88    }
89
90    void ControllableEntity::XMLPort(Element& xmlelement, XMLPort::Mode mode)
91    {
92        SUPER(ControllableEntity, XMLPort, xmlelement, mode);
93
94        XMLPortParam(ControllableEntity, "hudtemplate", setHudTemplate, getHudTemplate, xmlelement, mode);
95        XMLPortParam(ControllableEntity, "camerapositiontemplate", setCameraPositionTemplate, getCameraPositionTemkplate, xmlelement, mode);
96
97        XMLPortObject(ControllableEntity, CameraPosition, "camerapositions", addCameraPosition, getCameraPosition, xmlelement, mode);
98    }
99
100    void ControllableEntity::addCameraPosition(CameraPosition* position)
101    {
102        this->attach(position);
103        this->cameraPositions_.push_back(position);
104    }
105
106    CameraPosition* ControllableEntity::getCameraPosition(unsigned int index) const
107    {
108        unsigned int i = 0;
109        for (std::list<CameraPosition*>::const_iterator it = this->cameraPositions_.begin(); it != this->cameraPositions_.end(); ++it)
110        {
111            if (i == index)
112                return (*it);
113            ++i;
114        }
115        return 0;
116    }
117
118    void ControllableEntity::switchCamera()
119    {
120        if (this->camera_)
121        {
122            if (this->camera_->getParent() == this && this->cameraPositions_.size() > 0)
123            {
124                this->cameraPositions_.front()->attachCamera(this->camera_);
125            }
126            else if (this->cameraPositions_.size() > 0)
127            {
128                for (std::list<CameraPosition*>::const_iterator it = this->cameraPositions_.begin(); it != this->cameraPositions_.end(); ++it)
129                {
130                    if ((*it) == this->camera_->getParent())
131                    {
132                        ++it;
133                        if (it != this->cameraPositions_.end())
134                            (*it)->attachCamera(this->camera_);
135                        else
136                            (*this->cameraPositions_.begin())->attachCamera(this->camera_);
137                        break;
138                    }
139                }
140            }
141            else
142            {
143                this->attach(this->camera_);
144            }
145        }
146    }
147
148    void ControllableEntity::setPlayer(PlayerInfo* player)
149    {
150        if (!player)
151        {
152            this->removePlayer();
153            return;
154        }
155
156        this->player_ = player;
157        this->playerID_ = player->getObjectID();
158        this->bControlled_ = (player->isLocalPlayer() && player->isHumanPlayer());
159        if (this->bControlled_)
160        {
161            this->startLocalControl();
162
163            if (!Core::isMaster())
164            {
165                this->client_overwrite_ = this->server_overwrite_;
166COUT(0) << "CE: bidirectional synchronization" << std::endl;
167                this->setObjectMode(network::direction::bidirectional);
168            }
169        }
170    }
171
172    void ControllableEntity::removePlayer()
173    {
174        if (this->bControlled_)
175            this->stopLocalControl();
176
177        this->player_ = 0;
178        this->playerID_ = network::OBJECTID_UNKNOWN;
179        this->bControlled_ = false;
180        this->setObjectMode(network::direction::toclient);
181
182        if (this->bDestroyWhenPlayerLeft_)
183            delete this;
184    }
185
186    void ControllableEntity::networkcallback_changedplayerID()
187    {
188        // just do this in case the entity wasn't yet synchronized when the corresponding PlayerInfo got our objectID
189        if (this->playerID_ != network::OBJECTID_UNKNOWN)
190        {
191            this->player_ = dynamic_cast<PlayerInfo*>(network::Synchronisable::getSynchronisable(this->playerID_));
192            if (this->player_ && (this->player_->getControllableEntity() != this))
193                this->player_->startControl(this);
194        }
195    }
196
197    void ControllableEntity::startLocalControl()
198    {
199//        std::cout << this->getObjectID() << " ###### start local control" << std::endl;
200        this->camera_ = new Camera(this);
201        this->camera_->requestFocus();
202        if (this->cameraPositionTemplate_ != "")
203            this->addTemplate(this->cameraPositionTemplate_);
204        if (this->cameraPositions_.size() > 0)
205            this->cameraPositions_.front()->attachCamera(this->camera_);
206        else
207            this->attach(this->camera_);
208
209        if (this->hudtemplate_ != "")
210        {
211            this->hud_ = new OverlayGroup(this);
212            this->hud_->addTemplate(this->hudtemplate_);
213        }
214    }
215
216    void ControllableEntity::stopLocalControl()
217    {
218//        std::cout << "###### stop local control" << std::endl;
219        this->camera_->detachFromParent();
220        delete this->camera_;
221        this->camera_ = 0;
222
223        delete this->hud_;
224        this->hud_ = 0;
225    }
226
227    void ControllableEntity::tick(float dt)
228    {
229        if (this->isActive())
230        {
231            this->velocity_ += (dt * this->acceleration_);
232            this->node_->translate(dt * this->velocity_, Ogre::Node::TS_LOCAL);
233
234            if (Core::isMaster())
235            {
236                this->server_velocity_ = this->velocity_;
237                this->server_position_ = this->node_->getPosition();
238            }
239            else if (this->bControlled_)
240            {
241//                COUT(2) << "setting client position" << endl;
242                this->client_velocity_ = this->velocity_;
243                this->client_position_ = this->node_->getPosition();
244            }
245        }
246    }
247
248    void ControllableEntity::registerVariables()
249    {
250        REGISTERSTRING(this->cameraPositionTemplate_, network::direction::toclient);
251
252        REGISTERDATA(this->server_position_,    network::direction::toclient, new network::NetworkCallback<ControllableEntity>(this, &ControllableEntity::processServerPosition));
253        REGISTERDATA(this->server_velocity_,    network::direction::toclient, new network::NetworkCallback<ControllableEntity>(this, &ControllableEntity::processServerVelocity));
254        REGISTERDATA(this->server_orientation_, network::direction::toclient, new network::NetworkCallback<ControllableEntity>(this, &ControllableEntity::processServerOrientation));
255
256        REGISTERDATA(this->server_overwrite_,   network::direction::toclient, new network::NetworkCallback<ControllableEntity>(this, &ControllableEntity::processOverwrite));
257        REGISTERDATA(this->client_overwrite_,   network::direction::toserver);
258
259        REGISTERDATA(this->client_position_,    network::direction::toserver, new network::NetworkCallback<ControllableEntity>(this, &ControllableEntity::processClientPosition));
260        REGISTERDATA(this->client_velocity_,    network::direction::toserver, new network::NetworkCallback<ControllableEntity>(this, &ControllableEntity::processClientVelocity));
261        REGISTERDATA(this->client_orientation_, network::direction::toserver, new network::NetworkCallback<ControllableEntity>(this, &ControllableEntity::processClientOrientation));
262
263
264        REGISTERDATA(this->playerID_, network::direction::toclient, new network::NetworkCallback<ControllableEntity>(this, &ControllableEntity::networkcallback_changedplayerID));
265    }
266
267    void ControllableEntity::processServerPosition()
268    {
269        if (!this->bControlled_)
270            this->node_->setPosition(this->server_position_);
271    }
272
273    void ControllableEntity::processServerVelocity()
274    {
275        if (!this->bControlled_)
276            this->velocity_ = this->server_velocity_;
277    }
278
279    void ControllableEntity::processServerOrientation()
280    {
281        if (!this->bControlled_)
282            this->node_->setOrientation(this->server_orientation_);
283    }
284
285    void ControllableEntity::processOverwrite()
286    {
287        if (this->bControlled_)
288        {
289            this->setPosition(this->server_position_);
290            this->setVelocity(this->server_velocity_);
291            this->setOrientation(this->server_orientation_);
292
293            this->client_overwrite_ = this->server_overwrite_;
294        }
295    }
296
297    void ControllableEntity::processClientPosition()
298    {
299        if (this->server_overwrite_ == this->client_overwrite_)
300        {
301//            COUT(2) << "callback: setting client position" << endl;
302            this->node_->setPosition(this->client_position_);
303            this->server_position_ = this->client_position_;
304        }
305//        else
306//          COUT(2) << "callback: not setting client position" << endl;
307    }
308
309    void ControllableEntity::processClientVelocity()
310    {
311        if (this->server_overwrite_ == this->client_overwrite_)
312        {
313            this->velocity_ = this->client_velocity_;
314            this->server_velocity_ = this->client_velocity_;
315        }
316    }
317
318    void ControllableEntity::processClientOrientation()
319    {
320        if (this->server_overwrite_ == this->client_overwrite_)
321        {
322            this->node_->setOrientation(this->client_orientation_);
323            this->server_orientation_ = this->client_orientation_;
324        }
325    }
326
327
328    // virtual void PositionChanged() { }
329    void ControllableEntity::positionChanged(const Vector3& position)
330    {
331        if (Core::isMaster())
332        {
333            this->server_position_ = this->getPosition();
334            ++this->server_overwrite_;
335        }
336        else if (this->bControlled_)
337        {
338            this->client_position_ = position;
339        }
340    }
341
342    void ControllableEntity::setVelocity(const Vector3& velocity)
343    {
344        if (Core::isMaster())
345        {
346            this->velocity_ = velocity;
347            this->server_velocity_ = velocity;
348            ++this->server_overwrite_;
349        }
350        else if (this->bControlled_)
351        {
352            this->velocity_ = velocity;
353            this->client_velocity_ = velocity;
354        }
355    }
356    // virtual void translateChanged() { }
357    void ControllableEntity::translateChanged(const Vector3& distance, Ogre::Node::TransformSpace relativeTo)
358    {
359        if (Core::isMaster())
360        {
361            this->node_->translate(distance, relativeTo);
362            this->server_position_ = this->node_->getPosition();
363            ++this->server_overwrite_;
364        }
365        else if (this->bControlled_)
366        {
367            this->node_->translate(distance, relativeTo);
368            this->client_position_ = this->node_->getPosition();
369        }
370    }
371    // virtual void orientationChanged() { }
372    void ControllableEntity::orientationChanged(const Quaternion& orientation)
373    {
374        if (Core::isMaster())
375        {
376            this->node_->setOrientation(orientation);
377            this->server_orientation_ = orientation;
378            ++this->server_overwrite_;
379        }
380        else if (this->bControlled_)
381        {
382            this->node_->setOrientation(orientation);
383            this->client_orientation_ = orientation;
384        }
385    }
386    // virtual void rotateChanged() { }
387    void ControllableEntity::rotateChanged(const Quaternion& rotation, Ogre::Node::TransformSpace relativeTo)
388    {
389        if (Core::isMaster())
390        {
391            this->node_->rotate(rotation, relativeTo);
392            this->server_orientation_ = this->node_->getOrientation();
393            ++this->server_overwrite_;
394        }
395        else if (this->bControlled_)
396        {
397            this->node_->rotate(rotation, relativeTo);
398            this->client_orientation_ = this->node_->getOrientation();
399        }
400    }
401    // virtual void yawChanged() { }
402    void ControllableEntity::yawChanged(const Degree& angle, Ogre::Node::TransformSpace relativeTo)
403    {
404        if (Core::isMaster())
405        {
406            this->node_->yaw(angle, relativeTo);
407            this->server_orientation_ = this->node_->getOrientation();
408            ++this->server_overwrite_;
409        }
410        else if (this->bControlled_)
411        {
412            this->node_->yaw(angle, relativeTo);
413            this->client_orientation_ = this->node_->getOrientation();
414        }
415    }
416    // virtual void pitchChanged() { }
417    void ControllableEntity::pitchChanged(const Degree& angle, Ogre::Node::TransformSpace relativeTo)
418    {
419        if (Core::isMaster())
420        {
421            this->node_->pitch(angle, relativeTo);
422            this->server_orientation_ = this->node_->getOrientation();
423            ++this->server_overwrite_;
424        }
425        else if (this->bControlled_)
426        {
427            this->node_->pitch(angle, relativeTo);
428            this->client_orientation_ = this->node_->getOrientation();
429        }
430    }
431    // virtual void rollChanged() { }
432    void ControllableEntity::rollChanged(const Degree& angle, Ogre::Node::TransformSpace relativeTo)
433    {
434        if (Core::isMaster())
435        {
436            this->node_->roll(angle, relativeTo);
437            this->server_orientation_ = this->node_->getOrientation();
438            ++this->server_overwrite_;
439        }
440        else if (this->bControlled_)
441        {
442            this->node_->roll(angle, relativeTo);
443            this->client_orientation_ = this->node_->getOrientation();
444        }
445    }
446    //virtual void lookAtChanged() { }
447    void ControllableEntity::lookAtChanged(const Vector3& target, Ogre::Node::TransformSpace relativeTo, const Vector3& localDirectionVector)
448    {
449        if (Core::isMaster())
450        {
451            this->node_->lookAt(target, relativeTo, localDirectionVector);
452            this->server_orientation_ = this->node_->getOrientation();
453            ++this->server_overwrite_;
454        }
455        else if (this->bControlled_)
456        {
457            this->node_->lookAt(target, relativeTo, localDirectionVector);
458            this->client_orientation_ = this->node_->getOrientation();
459        }
460    }
461    // virtual void directionChanged( ) { }
462    void ControllableEntity::directionChanged(const Vector3& direction, Ogre::Node::TransformSpace relativeTo, const Vector3& localDirectionVector)
463    {
464        if (Core::isMaster())
465        {
466            this->node_->setDirection(direction, relativeTo, localDirectionVector);
467            this->server_orientation_ = this->node_->getOrientation();
468            ++this->server_overwrite_;
469        }
470        else if (this->bControlled_)
471        {
472            this->node_->setDirection(direction, relativeTo, localDirectionVector);
473            this->client_orientation_ = this->node_->getOrientation();
474        }
475    }
476}
Note: See TracBrowser for help on using the repository browser.