/* * 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: * Benjamin Knecht * Co-authors: * ... * */ #include "Trigger.h" #include "core/CoreIncludes.h" #include "core/ConsoleCommand.h" #include "core/GameMode.h" #include "core/XMLPort.h" #include "Scene.h" namespace orxonox { SetConsoleCommand(Trigger, debugFlares, false).defaultValues(false); CreateFactory(Trigger); Trigger::Trigger(BaseObject* creator) : StaticEntity(creator) { RegisterObject(Trigger); this->mode_ = TriggerMode::EventTriggerAND; this->bFirstTick_ = true; this->bActive_ = false; this->bTriggered_ = false; this->latestState_ = 0x0; this->bInvertMode_ = false; this->bSwitch_ = false; this->bStayActive_ = false; this->delay_ = 0.0f; this->remainingTime_ = 0.0f; this->timeSinceLastEvent_ = 0.0f; this->remainingActivations_ = -1; // this->bUpdating_ = false; if (this->getScene() && GameMode::showsGraphics()) { this->debugBillboard_.setBillboardSet(this->getScene()->getSceneManager(), "Examples/Flare", ColourValue(1.0, 0.0, 0.0), 1); this->debugBillboard_.setVisible(false); if (this->debugBillboard_.getBillboardSet()) this->attachOgreObject(this->debugBillboard_.getBillboardSet()); } this->setSyncMode(0x0); } Trigger::~Trigger() { } void Trigger::XMLPort(Element& xmlelement, XMLPort::Mode mode) { SUPER(Trigger, XMLPort, xmlelement, mode); XMLPortParam(Trigger, "delay", setDelay, getDelay, xmlelement, mode).defaultValues(0.0f); XMLPortParam(Trigger, "switch", setSwitch, getSwitch, xmlelement, mode).defaultValues(false); XMLPortParam(Trigger, "stayactive", setStayActive, getStayActive, xmlelement, mode).defaultValues(false); XMLPortParam(Trigger, "activations", setActivations, getActivations, xmlelement, mode).defaultValues(-1); XMLPortParam(Trigger, "invert", setInvert, getInvert, xmlelement, mode).defaultValues(false); XMLPortParamTemplate(Trigger, "mode", setMode, getModeString, xmlelement, mode, const std::string&).defaultValues("or"); XMLPortObject(Trigger, Trigger, "", addTrigger, getTrigger, xmlelement, mode); } void Trigger::tick(float dt) { if (this->bFirstTick_) { this->bFirstTick_ = false; this->triggered(false); } // Check if the object is active (this is NOT Trigger::isActive()!) if (!this->BaseObject::isActive()) return; SUPER(Trigger, tick, dt); bool newTriggered = this->isTriggered() ^ this->bInvertMode_; // check if new triggering event is really new if ((this->latestState_ & 0x1) != newTriggered) { // create new state if (newTriggered) { this->latestState_ |= 1; // set trigger bit to 1 this->switchState(); } else { this->latestState_ &= 0xFE; // set trigger bit to 0 if (!this->bSwitch_) this->switchState(); } } if (this->remainingTime_ > 0.0) { this->remainingTime_ -= dt; // only increase when acctually waiting for a state in the queue if (this->timeSinceLastEvent_ >= 0.0) this->timeSinceLastEvent_ += dt; } while (this->remainingTime_ <= 0.0 && this->stateChanges_.size() > 0) { // time ran out, change state to new one char newState = this->stateChanges_.front().second; this->bTriggered_ = (newState & 0x1); this->bActive_ = newState & 2; this->triggered(this->bActive_); this->stateChanges_.pop(); if (this->stateChanges_.size() != 0) this->remainingTime_ = this->stateChanges_.front().first; else this->timeSinceLastEvent_ = this->delay_; } if (this->bTriggered_ && this->bActive_) this->setBillboardColour(ColourValue(0.5, 1.0, 0.0)); else if (!this->bTriggered_ && this->bActive_) this->setBillboardColour(ColourValue(0.0, 1.0, 0.0)); else if (this->bTriggered_ && !this->bActive_) this->setBillboardColour(ColourValue(1.0, 0.5, 0.0)); else this->setBillboardColour(ColourValue(1.0, 0.0, 0.0)); } void Trigger::triggered(bool bIsTriggered) { this->fireEvent(bIsTriggered); } bool Trigger::isTriggered(TriggerMode::Value mode) { // if (this->bUpdating_) // return this->bTriggered_; // this->bUpdating_ = true; if (this->children_.size() != 0) { bool returnval = false; switch (mode) { case TriggerMode::EventTriggerAND: returnval = checkAnd(); break; case TriggerMode::EventTriggerOR: returnval = checkOr(); break; case TriggerMode::EventTriggerXOR: returnval = checkXor(); break; default: returnval = false; break; } // this->bUpdating_ = false; return returnval; } return true; } bool Trigger::checkAnd() { std::set::iterator it; for(it = this->children_.begin(); it != this->children_.end(); ++it) { if (!(*it)->isActive()) return false; } return true; } bool Trigger::checkOr() { std::set::iterator it; for(it = this->children_.begin(); it != this->children_.end(); ++it) { if ((*it)->isActive()) return true; } return false; } bool Trigger::checkXor() { std::set::iterator it; bool test = false; for(it = this->children_.begin(); it != this->children_.end(); ++it) { if (test && (*it)->isActive()) return false; if ((*it)->isActive()) test = true; } return test; } bool Trigger::switchState() { if (( (this->latestState_ & 2) && this->bStayActive_ && (this->remainingActivations_ <= 0)) || (!(this->latestState_ & 2) && (this->remainingActivations_ == 0))) return false; else { this->latestState_ ^= 2; // toggle state bit // increase activation count if (this->latestState_ & 2 && this->remainingActivations_ > 0) this->remainingActivations_--; this->storeState(); return true; } } void Trigger::storeState() { // put state change into queue this->stateChanges_.push(std::pair(this->timeSinceLastEvent_, this->latestState_)); // reset time since last event this->timeSinceLastEvent_ = 0.0; if (this->stateChanges_.size() == 1) this->remainingTime_ = this->stateChanges_.front().first; } void Trigger::setDelay(float delay) { this->delay_ = delay; this->timeSinceLastEvent_ = delay; } void Trigger::setMode(const std::string& modeName) { if (modeName == "and") this->setMode(TriggerMode::EventTriggerAND); else if (modeName == "or") this->setMode(TriggerMode::EventTriggerOR); else if (modeName == "xor") this->setMode(TriggerMode::EventTriggerXOR); } std::string Trigger::getModeString() const { if (this->mode_ == TriggerMode::EventTriggerAND) return "and"; else if (this->mode_ == TriggerMode::EventTriggerOR) return "or"; else if (this->mode_ == TriggerMode::EventTriggerXOR) return "xor"; else return "and"; } void Trigger::addTrigger(Trigger* trigger) { if (this != trigger) this->children_.insert(trigger); } const Trigger* Trigger::getTrigger(unsigned int index) const { if (this->children_.size() <= index) return NULL; std::set::const_iterator it; it = this->children_.begin(); for (unsigned int i = 0; i != index; ++i) ++it; return (*it); } void Trigger::debugFlares(bool bVisible) { for (ObjectList::iterator it = ObjectList::begin(); it != ObjectList::end(); ++it) it->setVisible(bVisible); } void Trigger::setBillboardColour(const ColourValue& colour) { this->debugBillboard_.setColour(colour); } void Trigger::changedVisibility() { SUPER(Trigger, changedVisibility); this->debugBillboard_.setVisible(this->isVisible()); } }