Planet
navi homePPSaboutscreenshotsdownloaddevelopmentforum

source: code/trunk/src/modules/pickup/PickupManager.cc @ 11702

Last change on this file since 11702 was 11701, checked in by landauf, 7 years ago

[HUD_HS16] removed debug output

  • Property svn:eol-style set to native
File size: 22.8 KB
RevLine 
[6474]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:
[7494]23 *      Damian 'Mozork' Frick
[6474]24 *   Co-authors:
25 *      ...
26 *
27*/
28
29/**
[6540]30    @file PickupManager.cc
[6474]31    @brief Implementation of the PickupManager class.
32*/
33
34#include "PickupManager.h"
35
36#include "core/CoreIncludes.h"
[6711]37#include "core/LuaState.h"
38#include "core/GUIManager.h"
[9667]39#include "core/class/Identifier.h"
[10624]40#include "core/singleton/ScopedSingletonIncludes.h"
[7504]41#include "network/Host.h"
[10624]42#include "network/NetworkFunctionIncludes.h"
[11353]43#include "core/input/KeyBinderManager.h"    //for keybinding
44#include "core/input/KeyBinder.h"           //for keybinding
45#include "core/command/ConsoleCommandIncludes.h"
[7533]46
[7539]47#include "infos/PlayerInfo.h"
[6474]48#include "interfaces/PickupCarrier.h"
49#include "worldentities/pawns/Pawn.h"
[7533]50
[7163]51#include "CollectiblePickup.h"
[6474]52#include "PickupRepresentation.h"
[11353]53#include "overlays/hud/HUDPickupSystem.h"
[6474]54
55namespace orxonox
56{
[10624]57    ManageScopedSingleton(PickupManager, ScopeID::ROOT, false);
[7163]58
[7533]59    // Initialization of the name of the PickupInventory GUI.
[6711]60    /*static*/ const std::string PickupManager::guiName_s = "PickupInventory";
[7163]61
[7801]62    // Register static network functions that are used to communicate changes to pickups over the network, such that the PickupInventory can display the information about the pickups properly.
[7504]63    registerStaticNetworkFunction(PickupManager::pickupChangedUsedNetwork);
64    registerStaticNetworkFunction(PickupManager::pickupChangedPickedUpNetwork);
65    registerStaticNetworkFunction(PickupManager::dropPickupNetworked);
66    registerStaticNetworkFunction(PickupManager::usePickupNetworked);
67
[10624]68    RegisterAbstractClass(PickupManager).inheritsFrom<PickupListener>();
69
[11353]70    SetConsoleCommand("useUnusePickup", &PickupManager::useUnusePickup).addShortcut().setActive(true);
71
[6474]72    /**
73    @brief
74        Constructor. Registers the PickupManager and creates the default PickupRepresentation.
75    */
[11071]76    PickupManager::PickupManager() : guiLoaded_(false), pickupHighestIndex_(0), defaultRepresentation_(nullptr)
[6474]77    {
[7504]78        RegisterObject(PickupManager);
[7163]79
[6474]80        this->defaultRepresentation_ = new PickupRepresentation();
[7163]81
[8858]82        orxout(internal_info, context::pickups) << "PickupManager created." << endl;
[6474]83    }
[7163]84
[6474]85    /**
86    @brief
87        Destructor.
[7533]88        Destroys the default PickupRepresentation and does some cleanup.
[6474]89    */
90    PickupManager::~PickupManager()
91    {
[7533]92        // Destroying the default representation.
[11071]93        if(this->defaultRepresentation_ != nullptr)
[6474]94            this->defaultRepresentation_->destroy();
[7163]95
[6725]96        this->representations_.clear();
[7163]97
[7533]98        // Destroying all the PickupInventoryContainers that are still there.
[11071]99        for(const auto& mapEntry : this->pickupInventoryContainers_)
100            delete mapEntry.second;
[7533]101        this->pickupInventoryContainers_.clear();
[7504]102
[7533]103        // Destroying all the WeakPointers that are still there.
104        this->pickups_.clear();
105        this->indexes_.clear();
106
[11701]107        orxout(internal_info, context::pickups) << "PickupManager destroyed." << endl;
[6474]108    }
[7163]109
[6474]110    /**
111    @brief
[9348]112        Registers a PickupRepresentation.
113    @param name
114        The representation's name.
[6474]115    @param representation
116        A pointer to the PickupRepresentation.
117    @return
118        Returns true if successful and false if not.
119    */
[9348]120    bool PickupManager::registerRepresentation(const std::string& name, PickupRepresentation* representation)
[7163]121    {
[7494]122        assert(representation);
123
[7533]124        // If the list is not empty and Pickupable already has a Representation registered.
[9348]125        if(!this->representations_.empty() && this->representations_.find(name) != this->representations_.end())
[6474]126            return false;
[7163]127
[9348]128        this->representations_[name] = representation;
[7163]129
[8858]130        orxout(verbose, context::pickups) << "PickupRepresentation &" << representation << " registered with the PickupManager." << endl;
[6474]131        return true;
132    }
[7163]133
[6474]134    /**
135    @brief
[9348]136        Unegisters a PickupRepresentation.
137    @param name
138        The representation's name.
[6725]139    @return
140        Returns true if successful and false if not.
141    */
[9348]142    bool PickupManager::unregisterRepresentation(const std::string& name)
[7163]143    {
[9348]144        std::map<std::string, PickupRepresentation*>::iterator it = this->representations_.find(name);
[7533]145        if(it == this->representations_.end()) // If the Pickupable is not registered in the first place.
[6725]146            return false;
[7163]147
[6725]148        this->representations_.erase(it);
[7163]149
[9348]150        orxout(verbose, context::pickups) << "PickupRepresentation &" << name << " unregistered with the PickupManager." << endl;
[6725]151        return true;
152    }
[7163]153
[7533]154    /**
155    @brief
[9348]156        Get the PickupRepresentation with the given name.
157    @param name
158        The name of the PickupRepresentation.
[7533]159    @return
[6474]160        Returns a pointer to the PickupRepresentation.
161    */
[9348]162    PickupRepresentation* PickupManager::getRepresentation(const std::string& name)
[6474]163    {
[9348]164        std::map<std::string, PickupRepresentation*>::iterator it = this->representations_.find(name);
165        if(it == this->representations_.end()) // If there is no PickupRepresentation associated with the input name.
[6474]166        {
[8858]167            orxout(verbose, context::pickups) << "PickupManager::getRepresentation() returned default representation." << endl;
[6474]168            return this->defaultRepresentation_;
169        }
[7163]170
[6474]171        return it->second;
172    }
[7163]173
[7206]174    /**
175    @brief
[7533]176        Is called by the PickupListener to notify the PickupManager, that the input Pickupable has transited to the input used state.
[7206]177    @param pickup
[7533]178        The Pickupable whose used status changed.
179    @param used
180        The used status the Pickupable changed to.
[7206]181    */
[7504]182    void PickupManager::pickupChangedUsed(Pickupable* pickup, bool used)
183    {
184        assert(pickup);
185
186        if(!GameMode::isMaster()) // If this is neither standalone nor the server.
187            return;
188
189        CollectiblePickup* collectible = orxonox_cast<CollectiblePickup*>(pickup);
[7533]190        // If the Pickupable is part of a PickupCollection it isn't displayed in the PickupInventory, just the PickupCollection is.
[11071]191        if(collectible != nullptr && collectible->isInCollection())
[7504]192            return;
193
[7533]194        // Getting clientId of the host this change of the pickup's used status concerns.
[7504]195        PickupCarrier* carrier = pickup->getCarrier();
[11071]196        while(carrier->getCarrierParent() != nullptr)
[7504]197            carrier = carrier->getCarrierParent();
198        Pawn* pawn = orxonox_cast<Pawn*>(carrier);
[11071]199        if(pawn == nullptr)
[7504]200            return;
201        PlayerInfo* info = pawn->getPlayer();
[11071]202        if(info == nullptr)
[7504]203            return;
204        unsigned int clientId = info->getClientID();
205
[7533]206        // Get the number identifying the pickup.
207        std::map<Pickupable*, uint32_t>::iterator it = this->indexes_.find(pickup);
[7504]208        assert(it != this->indexes_.end());
[7533]209        uint32_t index = it->second;
[7504]210
[7533]211        // If we're either in standalone mode or this is the host whom the change of the pickup's status concerns.
[7504]212        if(GameMode::isStandalone() || Host::getPlayerID() == clientId)
213        {
214            PickupManager::pickupChangedUsedNetwork(index, used, pickup->isUsable(), pickup->isUnusable());
215        }
[7533]216        // If the concerned host is somewhere in the network, we call pickupChangedUsedNetwork() on its PickupManager.
[7504]217        else
218        {
[10624]219            callStaticNetworkFunction(&PickupManager::pickupChangedUsedNetwork, clientId, index, used, pickup->isUsable(), pickup->isUnusable());
[7504]220        }
221    }
222
[7533]223    /**
224    @brief
225        Helper method to react to the change in the used status of a Pickupable.
226        Static method that is used by the server to inform the client it concerns about the status change.
227        The parameters that are given are used to update the information (i.e. the PickupInventoryContainer) the concerning PickupManager has about the Pickupable that changed.
228    @param pickup
229        A number identifying the Pickupable that changed its used status.
230    @param inUse
231        The used status the Pickupable changed to. (i.e. whether the Pickupable is in use or not).
232    @param usable
233        Whether the Pickupable's used status can be changed used in the PickupInventory.
234    @param unusable
235        Whether the Pickupable's used status can be changed to unused in the PickupInventory.
236    */
237    /*static*/ void PickupManager::pickupChangedUsedNetwork(uint32_t pickup, bool inUse, bool usable, bool unusable)
[7504]238    {
[7533]239        PickupManager& manager = PickupManager::getInstance(); // Get the PickupManager singleton on this host.
240        // If the input Pickupable (i.e its identifier) is not present in the list the PickupManager has.
[7504]241        if(manager.pickupInventoryContainers_.find(pickup) == manager.pickupInventoryContainers_.end())
242        {
[8858]243            orxout(internal_error, context::pickups) << "Pickupable &(" << pickup << ") was not registered with PickupManager for the PickupInventory, when it changed used." << endl;
[7504]244            return;
245        }
246
[7533]247        // Update the Pickupable's container with the information transferred.
[7504]248        manager.pickupInventoryContainers_[pickup]->inUse = inUse;
249        manager.pickupInventoryContainers_[pickup]->usable = usable;
250        manager.pickupInventoryContainers_[pickup]->unusable = unusable;
251
[7533]252        manager.updateGUI(); // Tell the PickupInventory that something has changed.
[7504]253    }
254
[7533]255    /**
256    @brief
257        Is called by the PickupListener to notify the PickupManager, that the input Pickupable has transited to the input pickedUp state.
258    @param pickup
259        The Pickupable whose pickedUp status changed.
260    @param pickedUp
261        The pickedUp status the Pickupable changed to.
262    */
[7504]263    void PickupManager::pickupChangedPickedUp(Pickupable* pickup, bool pickedUp)
264    {
265        assert(pickup);
266
[11353]267        for (HUDPickupSystem* hud : ObjectList<HUDPickupSystem>())
268            pickupSystem = hud;
269       
[7533]270        if(!GameMode::isMaster()) // If this is neither standalone nor the server.
[7504]271            return;
272
273        CollectiblePickup* collectible = orxonox_cast<CollectiblePickup*>(pickup);
[7533]274        // If the Pickupable is part of a PickupCollection it isn't displayed in the PickupInventory, just the PickupCollection is.
[11071]275        if(collectible != nullptr && collectible->isInCollection())
[7504]276            return;
277
[7533]278        // Getting clientId of the host this change of the pickup's pickedUp status concerns.
[7504]279        PickupCarrier* carrier = pickup->getCarrier();
[11071]280        while(carrier->getCarrierParent() != nullptr)
[7504]281            carrier = carrier->getCarrierParent();
282        Pawn* pawn = orxonox_cast<Pawn*>(carrier);
[11071]283        if(pawn == nullptr)
[7504]284            return;
[7533]285        PlayerInfo* info = pawn->getFormerPlayer();
[11071]286        if(info == nullptr)
[7504]287            return;
288        unsigned int clientId = info->getClientID();
289
[7533]290        uint32_t index = 0;
291        if(pickedUp) // If the Pickupable has changed to picked up, it is added to the required lists.
[7504]292        {
[11353]293            index = this->getPickupIndex(); // Get a new identifier (index) for the Pickupable.
[7533]294            // Add the Pickupable to the indexes_ and pickups_ lists.
[7504]295            this->indexes_[pickup] = index;
[10624]296            this->pickups_[index] = pickup;
[11353]297
298            this->picks.push_back(pickup);
299
300            if(pickupSystem)
[11700]301                pickupSystem->sync(picks, indexes_);
[11353]302           
[7504]303        }
[7533]304        else // If it was dropped, it is removed from the required lists.
[7504]305        {
[7533]306            // Get the indentifier (index) that identifies the input Pickupable.
307            std::map<Pickupable*, uint32_t>::iterator it = this->indexes_.find(pickup);
[7504]308            index = it->second;
[7533]309
[11353]310            this->indexes_.erase(pickup);
311            this->pickups_.erase(index); //set to null, so that can be identified as free slot by getPickupIndex()
312
313
314            this->picks.erase(std::remove(this->picks.begin(), this->picks.end(), pickup), this->picks.end()); //remove pickup from vector
315
316            if(pickupSystem)
[11700]317                pickupSystem->sync(picks, indexes_);
[7504]318        }
319
[7533]320        // If we're either in standalone mode or this is the host whom the change of the pickup's status concerns.
[7504]321        if(GameMode::isStandalone() || Host::getPlayerID() == clientId)
322        {
[7533]323            // If there is no PickupRepresentation registered the default representation is used.
[9348]324            if(this->representations_.find(pickup->getRepresentationName()) == this->representations_.end())
325                PickupManager::pickupChangedPickedUpNetwork(index, pickup->isUsable(), this->defaultRepresentation_->getObjectID(), pickup->getRepresentationName(), pickedUp);
[7531]326            else
[9348]327                PickupManager::pickupChangedPickedUpNetwork(index, pickup->isUsable(), this->representations_[pickup->getRepresentationName()]->getObjectID(), pickup->getRepresentationName(), pickedUp);
[7504]328        }
[7533]329        // If the concerned host is somewhere in the network, we call pickupChangedPickedUpNetwork() on its PickupManager.
[7504]330        else
331        {
[7533]332            // If there is no PickupRepresentation registered the default representation is used.
[9348]333            if(this->representations_.find(pickup->getRepresentationName()) == this->representations_.end())
[7531]334            {
[10624]335                callStaticNetworkFunction(&PickupManager::pickupChangedPickedUpNetwork, clientId, index, pickup->isUsable(), this->defaultRepresentation_->getObjectID(), pickedUp);
[7531]336            }
337            else
338            {
[10624]339                callStaticNetworkFunction(&PickupManager::pickupChangedPickedUpNetwork, clientId, index, pickup->isUsable(), this->representations_[pickup->getRepresentationName()]->getObjectID(), pickedUp);
[7531]340            }
[7504]341        }
342
343    }
344
[11353]345    //This function is called by the command line or by the key binding
346    //it uses or unuses the pickup, depending on its current state
347    //or drops it (depends what you comment/uncomment)
348    void PickupManager::useUnusePickup(uint32_t index) 
349    {
350        PickupManager& manager = PickupManager::getInstance();
351
352        if(!manager.pickups_.count(index)) return; //if pickup is no longer here, dont do anything
353
354        Pickupable* pickup=manager.pickups_.find(index)->second;
355        if(pickup==nullptr)
356        {
357            return;                       //pickup does not exist
358        }
359
360        //if the pickup should be dropped upon key press
361        manager.dropPickup(index);
362
363        //if the pickup should be used/unused upon key press
364
365        // if(pickup->isUsed())
366        //     manager.usePickup(index, false);
367        // else
368        //     manager.usePickup(index, true);
369    }
370
371
[7533]372    /**
373    @brief
374        Helper method to react to the change in the pickedUp status of a Pickupable.
375        Static method that is used by the server to inform the client it concerns about the status change.
376        The parameters that are given are used to update the information (i.e. the PickupInventoryContainer) the concerning PickupManager has about the Pickupable that changed.
377    @param pickup
378        A number identifying the Pickupable that changed its pickedUp status.
[7547]379    @param usable
380        Whether the Pickupable's used status can be changed to used in the PickupInventory.
[7533]381    @param representationObjectId
382        The objectId identifying (over the network) the PickupRepresentation that represents this Pickupable.
[11099]383    @param representationName
384        The name of the associated PickupRepresentation
[7533]385    @param pickedUp
386        The pickedUp status the Pickupable changed to.
387    */
[9348]388    /*static*/ void PickupManager::pickupChangedPickedUpNetwork(uint32_t pickup, bool usable, uint32_t representationObjectId, const std::string& representationName, bool pickedUp)
[7504]389    {
[7533]390        PickupManager& manager = PickupManager::getInstance(); // Get the PickupManager singleton on this host.
391        // If the Pickupable has been picked up, we create a new PickupInventoryContainer for it.
[7504]392        if(pickedUp)
393        {
[7533]394            // Create a new PickupInventoryContainer for the Pickupable and set all the necessary information.
[7504]395            PickupInventoryContainer* container = new PickupInventoryContainer;
396            container->pickup = pickup;
397            container->inUse = false;
398            container->pickedUp = pickedUp;
399            container->usable = usable;
400            container->unusable = false;
401            container->representationObjectId = representationObjectId;
[9348]402            container->representationName = representationName;
[7533]403            // Insert the container into the pickupInventoryContainers_ list.
404            manager.pickupInventoryContainers_.insert(std::pair<uint32_t, PickupInventoryContainer*>(pickup, container));
[7504]405
[7533]406            manager.updateGUI(); // Tell the PickupInventory that something has changed.
[7504]407        }
[7533]408        // If the Pickupable has been dropped, we remove it from the pickupInventoryContainers_ list.
[7504]409        else
410        {
[7533]411            std::map<uint32_t, PickupInventoryContainer*>::iterator it = manager.pickupInventoryContainers_.find(pickup);
[7504]412            if(it != manager.pickupInventoryContainers_.end())
413                delete it->second;
414            manager.pickupInventoryContainers_.erase(pickup);
415
[7533]416            manager.updateGUI(); // Tell the PickupInventory that something has changed.
[7504]417        }
418    }
419
[7533]420    /**
421    @brief
422        Get the number of pickups currently picked up by the player.
423        This method is used in lua to populate the PickupInventory. The intended usage is to call this method to reset the iterator of the list of PickupInventoryContainers and then use popPickup() to get the individual PickupInventoryContainers.
424    @return
425        Returns the number of the players picked up Pickupables.
426    */
427    int PickupManager::getNumPickups(void)
428    {
429        this->pickupsIterator_ = this->pickupInventoryContainers_.begin(); // Reset iterator.
430
431        return this->pickupInventoryContainers_.size();
432    }
433
434    /**
435    @brief
436        Drop the input Pickupable.
437        This method checks whether the input Pickupable still exists and drops it, if so.
438    @param pickup
439        The identifier of the Pickupable to be dropped.
440    */
441    void PickupManager::dropPickup(uint32_t pickup)
442    {
443        // If we're either server or standalone and the list of pickups is not empty, we find and drop the input pickup.
[7539]444        if(GameMode::isMaster())
[7533]445        {
[7539]446            if(this->pickups_.empty())
447                return;
[10624]448            Pickupable* pickupable = this->pickups_.find(pickup)->second;
[11071]449            if(pickupable != nullptr)
[11353]450            {
[7533]451                pickupable->drop();
[11353]452
453            }
[7533]454        }
455        // If we're neither server nor standalone we drop the pickup by calling dropPickupNetworked() of the PickupManager on the server.
456        else
457        {
[10624]458            callStaticNetworkFunction(&PickupManager::dropPickupNetworked, 0, pickup);
[7533]459        }
460    }
461
462    /**
463    @brief
464        Helper method to drop the input pickup on the server.
465        Static method that is used by clients to instruct the server to drop the input pickup.
466    @param pickup
467        The identifier of the Pickupable to be dropped.
468    */
469    /*static*/ void PickupManager::dropPickupNetworked(uint32_t pickup)
470    {
471        if(GameMode::isServer()) // Obviously we only want to do this on the server.
472        {
473            PickupManager& manager = PickupManager::getInstance();
[7539]474            manager.dropPickup(pickup);
[7533]475        }
476    }
477
478    /**
479    @brief
480        Use (or unuse) the input Pickupable.
481        This method checks whether the input Pickupable still exists and uses (or unuses) it, if so,
482    @param pickup
483        The identifier of the Pickupable to be used (or unused).
484    @param use
485        If true the input Pickupable is used, if false it is unused.
486    */
487    void PickupManager::usePickup(uint32_t pickup, bool use)
488    {
489        // If we're either server or standalone and the list of pickups is not empty, we find and change the used status of the input pickup.
[7539]490        if(GameMode::isMaster())
[7533]491        {
[7539]492            if(this->pickups_.empty())
493                return;
[10624]494            Pickupable* pickupable = this->pickups_.find(pickup)->second;
[11071]495            if(pickupable != nullptr)
[7533]496                pickupable->setUsed(use);
497        }
498        // If we're neither server nor standalone we change the used status of the pickup by calling usePickupNetworked() of the PickupManager on the server.
499        else
500        {
[10624]501            callStaticNetworkFunction(&PickupManager::usePickupNetworked, 0, pickup, use);
[7533]502        }
503    }
504
505    /**
506    @brief
507        Helper method to use (or unuse) the input Pickupable on the server.
508        Static method that is used by clients to instruct the server to use (or unuse) the input pickup.
509    @param pickup
510        The identifier of the Pickupable to be used (or unused).
511    @param use
512        If true the input Pickupable is used, if false it is unused.
513    */
514    /*static*/ void PickupManager::usePickupNetworked(uint32_t pickup, bool use)
515    {
516        if(GameMode::isServer())
517        {
518            PickupManager& manager = PickupManager::getInstance();
[7539]519            manager.usePickup(pickup, use);
[7533]520        }
521    }
522
523    /**
524    @brief
525        Updates the PickupInventory GUI.
526        Also loads the PickupInventory GUI if is hasn't been done already.
527    */
[7504]528    inline void PickupManager::updateGUI(void)
529    {
[7533]530        // We only need to update (and load) the GUI if this host shows graphics.
[7504]531        if(GameMode::showsGraphics())
532        {
[7533]533            if(!this->guiLoaded_) // If the GUI hasn't been loaded, yet, we load it.
[7504]534            {
535                GUIManager::getInstance().loadGUI(PickupManager::guiName_s);
536                this->guiLoaded_ = true;
537            }
[7533]538
539            // Update the GUI.
[7504]540            GUIManager::getInstance().getLuaState()->doString(PickupManager::guiName_s + ".update()");
541        }
542    }
543
[7533]544    /**
545    @brief
[11353]546        Get a new index between 0 and 9 for a Pickupable.
547        If all slots are occupied, the Pickupable in the first slot will be dropped.
[7533]548    @return
549        Returns the new index.
550    */
551    uint32_t PickupManager::getPickupIndex(void)
[7504]552    {
[11353]553        //check if there are free slots available
554
555        for(uint32_t i=0; i<10; i++)
556        {
557            if(!pickups_.count(i)) return i;
558        }
559        //all slots are full and we have to drop sth
560        orxout(internal_info, context::pickups) << "everything was full and we have now dropped the first element" << endl;
561        this->dropPickup(0);
562        return 0;
[7504]563    }
564
[6474]565}
Note: See TracBrowser for help on using the repository browser.