/* * ORXONOX - the hottest 3D action shooter ever to exist * > www.orxonox.net < * * * License notice: * * This program is free software; you can redistribute it and/or * modify it under the terms of the GNU General Public License * as published by the Free Software Foundation; either version 2 * of the License, or (at your option) any later version. * * This program is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the * GNU General Public License for more details. * * You should have received a copy of the GNU General Public License * along with this program; if not, write to the Free Software * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA. * * Author: * Damian 'Mozork' Frick * Co-authors: * ... * */ /** @file PickupCarrier.h @brief Definition of the PickupCarrier class. */ #ifndef _PickupCarrier_H__ #define _PickupCarrier_H__ #include "OrxonoxPrereqs.h" #include #include #include "Pickupable.h" #include "core/Identifier.h" #include "core/WeakPtr.h" #include "core/OrxonoxClass.h" namespace orxonox { //! Forward-declarations. class PickupManager; class Pickup; class HealthPickup; class InvisiblePickup; class MetaPickup; class SpeedPickup; /** @brief The PickupCarrier interface provides the means, for any class implementing it, to possess Pickupables. @author Damian 'Mozork' Frick */ class _OrxonoxExport PickupCarrier : virtual public OrxonoxClass { //! So that the different Pickupables have full access to their PickupCarrier. friend class Pickupable; friend class PickupManager; //! Friends. friend class Pickup; friend class HealthPickup; friend class InvisiblePickup; friend class MetaPickup; friend class SpeedPickup; public: PickupCarrier(); //!< Constructor. virtual ~PickupCarrier(); //!< Destructor. /** @brief Can be called to pick up a Pickupable. @param pickup A pointer to the Pickupable. @return Returns true if the Pickupable was picked up, false if not. */ bool pickup(Pickupable* pickup) { bool pickedUp = this->pickups_.insert(pickup).second; if(pickedUp) { COUT(4) << "Picked up Pickupable " << pickup->getIdentifier()->getName() << "(&" << pickup << ")." << std::endl; pickup->pickedUp(this); } return pickedUp; } /** @brief Can be called to drop a Pickupable. @param pickup A pointer to the Pickupable. @param drop If the Pickupable should just be removed from the PickupCarrier without further action, this can be set to false. true is default. @return Returns true if the Pickupable has been dropped, false if not. */ bool drop(Pickupable* pickup, bool drop = true) { bool dropped = this->pickups_.erase(pickup) == 1; if(dropped && drop) { COUT(4) << "Dropping Pickupable " << pickup->getIdentifier()->getName() << "(&" << pickup << ")." << std::endl; pickup->dropped(); } return dropped; } /** @brief Can be used to check whether the PickupCarrier or a child of his is a target ot the input Pickupable. @param pickup A pointer to the Pickupable. @return Returns true if the PickupCarrier or one of its children is a target, false if not. */ bool isTarget(const Pickupable* pickup) { if(pickup->isTarget(this)) //!< If the PickupCarrier itself is a target. return true; //! Go recursively through all children to check whether they are a target. std::vector* children = this->getCarrierChildren(); for(std::vector::const_iterator it = children->begin(); it != children->end(); it++) { if((*it)->isTarget(pickup)) return true; } children->clear(); delete children; return false; } /** @brief Get the carrier that is both a child of the PickupCarrier (or the PickupCarrier itself) and a target of the input Pickupable. @param pickup A pounter to the Pickupable. @return Returns a pointer to the PickupCarrier that is the target of the input Pickupable. */ PickupCarrier* getTarget(const Pickupable* pickup) { if(!this->isTarget(pickup)) return NULL; if(pickup->isTarget(this)) //!< If the PickupCarrier itself is a target. return this; //! Go recursively through all children to check whether they are the target. std::vector* children = this->getCarrierChildren(); for(std::vector::iterator it = children->begin(); it != children->end(); it++) { if(pickup->isTarget(*it)) return *it; } children->clear(); delete children; return NULL; } /** @brief Get the (absolute) position of the PickupCarrier. This method needs to be implemented by any direct derivative class of PickupCarrier. @return Returns the position as a Vector3. */ virtual const Vector3& getCarrierPosition(void) = 0; /** @brief Get the name of this PickupCarrier. @return Returns the name as a string. */ const std::string& getCarrierName(void) { return this->carrierName_; } // tolua_export protected: /** @brief Get all direct children of this PickupSpawner. This method needs to be implemented by any direct derivative class of PickupCarrier. The returned list will be deleted by the methods calling this function. @return Returns a pointer to a list of all direct children. */ virtual std::vector* getCarrierChildren(void) = 0; /** @brief Get the parent of this PickupSpawner This method needs to be implemented by any direct derivative class of PickupCarrier. @return Returns a pointer to the parent. */ virtual PickupCarrier* getCarrierParent(void) = 0; /** @brief Get all Pickupables this PickupCarrier has. @return Returns the set of all Pickupables this PickupCarrier has. */ std::set& getPickups(void) { return this->pickups_; } /** @brief Set the name of this PickupCarrier. The name needs to be set in the constructor of every class inheriting from PickupCarrier, by calling setCarrierName(). @param name The name to be set. */ void setCarrierName(const std::string& name) { this->carrierName_ = name; } private: std::set pickups_; //!< The list of Pickupables carried by this PickupCarrier. std::string carrierName_; //!< The name of the PickupCarrier, as displayed in the PickupInventory. /** @brief Get the number of carrier children this PickupCarrier has. @return Returns the number of carrier children. */ unsigned int getNumCarrierChildren(void) { std::vector* list = this->getCarrierChildren(); unsigned int size = list->size(); delete list; return size; } /** @brief Get the index-th child of this PickupCarrier. @param index The index of the child to return. @return Returns the index-th child. */ PickupCarrier* getCarrierChild(unsigned int index) { std::vector* list = this->getCarrierChildren(); if(list->size() < index) return NULL; PickupCarrier* carrier = (*list)[index]; delete list; return carrier; } /** @brief Get the number of Pickupables this PickupCarrier carries. @return returns the number of pickups. */ unsigned int getNumPickups(void) { return this->pickups_.size(); } /** @brief Get the index-th Pickupable of this PickupCarrier. @param index The index of the Pickupable to return. @return Returns the index-th pickup. */ Pickupable* getPickup(unsigned int index) { std::set::iterator it; for(it = this->pickups_.begin(); index != 0 && it != this->pickups_.end(); it++) index--; if(it == this->pickups_.end()) return NULL; return *it; } }; } #endif /* _PickupCarrier_H__ */