/* * 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: * Reto Grieder * Co-authors: * ... * */ #include "CompoundCollisionShape.h" #include "BulletCollision/CollisionShapes/btCompoundShape.h" #include "util/Exception.h" #include "core/CoreIncludes.h" #include "core/XMLPort.h" #include "tools/BulletConversions.h" namespace orxonox { CreateFactory(CompoundCollisionShape); CompoundCollisionShape::CompoundCollisionShape(BaseObject* creator) : CollisionShape(creator) { RegisterObject(CompoundCollisionShape); this->compoundShape_ = new btCompoundShape(); } CompoundCollisionShape::~CompoundCollisionShape() { if (this->isInitialized()) { // Delete all children for (std::map::iterator it = this->attachedShapes_.begin(); it != this->attachedShapes_.end(); ++it) { // make sure that the child doesn't want to detach itself --> speedup because of the missing update it->first->notifyDetached(); delete it->first; } delete this->compoundShape_; } } void CompoundCollisionShape::XMLPort(Element& xmlelement, XMLPort::Mode mode) { SUPER(CompoundCollisionShape, XMLPort, xmlelement, mode); // Attached collision shapes XMLPortObject(CompoundCollisionShape, CollisionShape, "", attach, detach, xmlelement, mode); } void CompoundCollisionShape::attach(CollisionShape* shape) { if (!shape || static_cast(this) == shape) return; if (this->attachedShapes_.find(shape) != this->attachedShapes_.end()) { CCOUT(2) << "Warning: Attaching a CollisionShape twice is not yet supported." << std::endl; return; } if (!shape->notifyBeingAttached(this)) return; this->attachedShapes_[shape] = shape->getCollisionShape(); if (shape->getCollisionShape()) { // Only actually attach if we didn't pick a CompoundCollisionShape with no content btTransform transf(omni_cast(shape->getOrientation()), omni_cast(shape->getPosition())); this->compoundShape_->addChildShape(transf, shape->getCollisionShape()); this->updatePublicShape(); } } void CompoundCollisionShape::detach(CollisionShape* shape) { if (this->attachedShapes_.find(shape) != this->attachedShapes_.end()) { this->attachedShapes_.erase(shape); if (shape->getCollisionShape()) this->compoundShape_->removeChildShape(shape->getCollisionShape()); shape->notifyDetached(); this->updatePublicShape(); } else CCOUT(2) << "Warning: Cannot detach non child collision shape" << std::endl; } void CompoundCollisionShape::detachAll() { while (this->attachedShapes_.size() > 0) this->detach(this->attachedShapes_.begin()->first); } void CompoundCollisionShape::updateAttachedShape(CollisionShape* shape) { if (!shape) return; std::map::iterator it = this->attachedShapes_.find(shape); if (it == this->attachedShapes_.end()) { CCOUT(2) << "Warning: Cannot update child shape: Instance not a child." << std::endl; return; } // Remove old btCollisionShape, stored in the children map if (it->second) this->compoundShape_->removeChildShape(it->second); if (shape->getCollisionShape()) { // Only actually attach if we didn't pick a CompoundCollisionShape with no content btTransform transf(omni_cast(shape->getOrientation()), omni_cast(shape->getPosition())); this->compoundShape_->addChildShape(transf, shape->getCollisionShape()); it->second = shape->getCollisionShape(); } this->updatePublicShape(); } void CompoundCollisionShape::updatePublicShape() { btCollisionShape* primitive = 0; bool bPrimitive = true; bool bEmpty = true; for (std::map::const_iterator it = this->attachedShapes_.begin(); it != this->attachedShapes_.end(); ++it) { if (it->second) { bEmpty = false; if (!it->first->hasTransform() && !bPrimitive) primitive = it->second; else bPrimitive = false; } } if (bEmpty) { if (this->collisionShape_ == 0) { this->collisionShape_ = 0; return; } this->collisionShape_ = 0; } else if (bPrimitive) { // --> Only one shape to be added, no transform; return it directly this->collisionShape_ = primitive; } else { // Make sure we use the compound shape when returning a btCollisionShape this->collisionShape_ = this->compoundShape_; } this->updateParent(); } CollisionShape* CompoundCollisionShape::getAttachedShape(unsigned int index) const { unsigned int i = 0; for (std::map::const_iterator it = this->attachedShapes_.begin(); it != this->attachedShapes_.end(); ++it) { if (i == index) return it->first; ++i; } return 0; } }