Planet
navi homePPSaboutscreenshotsdownloaddevelopmentforum

source: code/branches/cpp11_v2/src/modules/objects/triggers/MultiTrigger.cc @ 10985

Last change on this file since 10985 was 10918, checked in by muemart, 10 years ago

Use emplace_back instead of push_back if beneficial

  • Property svn:eol-style set to native
File size: 22.8 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 *      Damian 'Mozork' Frick
24 *   Co-authors:
25 *      ...
26 *
27*/
28
29/**
30    @file MultiTrigger.cc
31    @brief Implementation of the MultiTrigger class.
32    @ingroup MultiTrigger
33*/
34
35#include "MultiTrigger.h"
36
37#include "core/CoreIncludes.h"
38#include "core/XMLPort.h"
39
40#include "MultiTriggerContainer.h"
41
42namespace orxonox
43{
44
45    RegisterClass(MultiTrigger);
46
47    /**
48    @brief
49        Constructor. Registers the objects and initializes default values.
50    @param creator
51        The creator.
52    */
53    MultiTrigger::MultiTrigger(Context* context) : TriggerBase(context)
54    {
55        RegisterObject(MultiTrigger);
56
57        this->maxNumSimultaneousTriggerers_ = INF_s;
58
59        this->bBroadcast_ = false;
60
61        this->targetMask_.exclude(Class(BaseObject));
62
63        this->bMultiTrigger_ = true;
64
65        this->setSyncMode(ObjectDirection::None);
66    }
67
68    /**
69    @brief
70        Destructor. Cleans up the state queue.
71    */
72    MultiTrigger::~MultiTrigger()
73    {
74        orxout(verbose, context::triggers) << "Destroying MultiTrigger &" << this << ". " << this->stateQueue_.size() << " states still in queue. Deleting." << endl;
75        while(this->stateQueue_.size() > 0)
76        {
77            MultiTriggerState* state = this->stateQueue_.front().second;
78            this->stateQueue_.pop_front();
79            delete state;
80        }
81    }
82
83    /**
84    @brief
85        Method for creating a MultiTrigger object through XML.
86        For a detailed description of the parameters please see the class description in the header file.
87    */
88    void MultiTrigger::XMLPort(Element& xmlelement, XMLPort::Mode mode)
89    {
90        SUPER(MultiTrigger, XMLPort, xmlelement, mode);
91
92        XMLPortParam(MultiTrigger, "simultaneousTriggerers", setSimultaneousTriggerers, getSimultaneousTriggerers, xmlelement, mode);
93        XMLPortParam(MultiTrigger, "broadcast", setBroadcast, getBroadcast, xmlelement, mode);
94        XMLPortParamLoadOnly(MultiTrigger, "target", addTarget, xmlelement, mode).defaultValues("Pawn"); //TODO: Remove load only
95
96        orxout(verbose, context::triggers) << "MultiTrigger '" << this->getName() << "' (&" << this << ") created." << endl;
97    }
98
99
100    /**
101    @brief
102        A method that is executed each tick.
103        Most of the magic of MultiTriggers happens here.
104    @param dt
105        The duration of the last tick.
106    */
107    //TODO: Segment into some helper methods?
108    void MultiTrigger::tick(float dt)
109    {
110        // If this is the first tick.
111        if(this->bFirstTick_)
112        {
113            this->bFirstTick_ = false;
114            // Fire for all objects that are targets.
115            this->broadcast(false);
116        }
117
118        // Check if the object is active (this is NOT MultiTrigger::isActive()!), it is whether the MultiTrigger actually does anything, ever.
119        if (!this->BaseObject::isActive())
120            return;
121
122        SUPER(MultiTrigger, tick, dt);
123
124        // Let the MultiTrigger return the states that trigger and process the new states if there are any.
125        std::queue<MultiTriggerState*>* queue  = this->letTrigger();
126        if(queue != nullptr)
127        {
128            while(queue->size() > 0)
129            {
130                MultiTriggerState* state = queue->front();
131                // If the state is nullptr. (This really shouldn't happen)
132                if(state == nullptr)
133                {
134                    orxout(internal_error, context::triggers) << "In MultiTrigger '" << this->getName() << "' (&" << this << "), Error: State of new states queue was nullptr. State ignored." << endl;
135                    queue->pop();
136                    continue;
137                }
138
139                // The new triggered state dependent on the requested state, the mode and the invert-mode.
140                bool bTriggered = (state->bTriggered & this->isModeTriggered(state->originator)) ^ this->getInvert();
141
142                // If the 'triggered' state has changed or the MultiTrigger has delay and thus we don't know whether this state will actually change the 'triggered' state, a new state is added to the state queue.
143                if(this->getDelay() > 0.0f || bTriggered ^ this->isTriggered(state->originator))
144                {
145                    state->bTriggered = bTriggered;
146                    this->addState(state);
147                }
148                // Else the change is irrelevant.
149                else
150                    delete state;
151
152                queue->pop();
153            }
154            delete queue;
155        }
156
157        // Go through the state queue and activate all pending states whose remaining time has expired.
158        if(this->stateQueue_.size() > 0)
159        {
160            MultiTriggerState* state;
161            float timeRemaining;
162
163            // Go through all pending states.
164            for(int size = this->stateQueue_.size(); size >= 1; size--)
165            {
166                timeRemaining = this->stateQueue_.front().first;
167                state = this->stateQueue_.front().second;
168
169                // If the remaining time has expired, the state has to be set as the state of the MultiTrigger.
170                if(timeRemaining <= dt)
171                {
172                    // If the maximum number of objects simultaneously triggering this MultiTrigger is not exceeded.
173                    if(this->getSimultaneousTriggerers() == TriggerBase::INF_s || this->triggered_.size() < (unsigned int)this->getSimultaneousTriggerers())
174                    {
175                        bool bStateChanged = false;
176                        // If the 'triggered' state is different from what it is now, change it.
177                        if(state->bTriggered ^ this->isTriggered(state->originator))
178                        {
179                            // Add the originator to the objects triggering this MultiTrigger.
180                            if(state->bTriggered == true)
181                                this->triggered_.insert(state->originator);
182                            // Remove the originator from the objects triggering this MultiTrigger.
183                            else
184                                this->triggered_.erase(state->originator);
185
186                            bStateChanged = true;
187                        }
188
189                        // Get the activity of the new state.
190                        bool bActive;
191                        // If the MultiTrigger is in switch mode the 'active'-state only changes of the state changed to triggered.
192                        if(this->getSwitch() && !state->bTriggered)
193                            bActive = this->isActive(state->originator);
194                        else
195                            bActive = !this->isActive(state->originator);
196
197                        // If the activity is different from what it is now, change it and fire an Event.
198                        if(bActive ^ this->isActive(state->originator))
199                        {
200                            bool bFire = true;
201
202                            // Add the originator to the objects activating this MultiTrigger.
203                            if(bActive == true)
204                            {
205                                // If the MultiTrigger has not exceeded its remaining activations.
206                                if(this->hasRemainingActivations())
207                                {
208                                    this->active_.insert(state->originator);
209                                    if(this->remainingActivations_ != INF_s)
210                                        this->remainingActivations_--; // Decrement the remaining activations.
211                                }
212                                else
213                                    bFire = false;
214                            }
215                            // Remove the originator from the objects activating this MultiTrigger.
216                            else
217                            {
218                                // If the MultiTrigger doesn't stay active or hasn't' exceeded its remaining activations.
219                                if(!this->getStayActive() || this->hasRemainingActivations())
220                                    this->active_.erase(state->originator);
221                                else
222                                    bFire = false;
223                            }
224
225                            // Fire the Event if the activity has changed.
226                            if(bFire)
227                            {
228                                // If the MultiTrigger is set to broadcast and has no originator a boradcast is fired.
229                                if(this->getBroadcast() && state->originator == nullptr)
230                                    this->broadcast(bActive);
231                                // Else a normal event is fired.
232                                else
233                                    this->fire(bActive, state->originator);
234
235                                bStateChanged = true;
236                            }
237                        }
238
239                        if(bStateChanged)
240                        {
241                            // Print some debug output if the state has changed.
242                            if(state->originator != nullptr)
243                                orxout(verbose, context::triggers) << "MultiTrigger '" << this->getName() << "' (&" << this << ") changed state. originator: " << state->originator->getIdentifier()->getName() << " (&" << state->originator << "), active: " << bActive << ", triggered: " << state->bTriggered << "." << endl;
244                            else
245                                orxout(verbose, context::triggers) << "MultiTrigger '" << this->getName() << "' (&" << this << ") changed state. originator: nullptr, active: " << bActive << ", triggered: " << state->bTriggered << "." << endl;
246
247                            // If the MultiTrigger has a parent trigger, that is itself a MultiTrigger, it needs to call a method to notify him, that its activity has changed.
248                            if(this->parent_ != nullptr && this->parent_->isMultiTrigger())
249                                static_cast<MultiTrigger*>(this->parent_)->childActivityChanged(state->originator);
250                        }
251
252                        // If the MultiTrigger has exceeded its amount of activations and it doesn't stay active, it has to be deactivated.
253                        if(this->remainingActivations_ == 0 && !bActive)
254                        {
255                            this->BaseObject::setActive(false);
256                            orxout(verbose, context::triggers) << "MultiTrigger '" << this->getName() << "' (&" << this << ") ran out of activations. Setting it to inactive." << endl;
257                        }
258                    }
259
260                    // Remove the state from the state queue.
261                    this->stateQueue_.pop_front();
262                    delete state;
263                }
264                // If the remaining time has not yet expired. Decrement the remainig time and put the state at the end of the queue.
265                else
266                {
267                    this->stateQueue_.emplace_back(timeRemaining-dt, state);
268                    this->stateQueue_.pop_front();
269                }
270            }
271        }
272    }
273
274    /**
275    @brief
276        Check whether the MultiTrigger is active for a given object.
277    @param triggerer
278        A pointer to the object.
279    @return
280        Returns true if the MultiTrigger is active, false if not.
281    */
282    bool MultiTrigger::isActive(BaseObject* triggerer) const
283    {
284        std::set<BaseObject*>::const_iterator it = this->active_.find(triggerer);
285        if(it == this->active_.end())
286            return false;
287        return true;
288    }
289
290    /**
291    @brief
292        Add some target to the MultiTrigger.
293    @param targetStr
294        The target class name as a string.
295    */
296    void MultiTrigger::addTarget(const std::string& targetStr)
297    {
298        Identifier* target = ClassByString(targetStr);
299
300        // If the target is not a valid class name display an error.
301        if (target == nullptr)
302        {
303            orxout(internal_error, context::triggers) << "'" << targetStr << "' is not a valid class name to include in ClassTreeMask (in " << this->getName() << ", class " << this->getIdentifier()->getName() << ")" << endl;
304            return;
305        }
306
307        this->targetMask_.include(target);
308
309        // A MultiTrigger shouldn't react to itself or other triggers.
310        this->targetMask_.exclude(Class(TriggerBase), true);
311
312        // We only want WorldEntities
313        ClassTreeMask WEMask;
314        WEMask.include(Class(WorldEntity));
315        this->targetMask_ *= WEMask;
316    }
317
318    /**
319    @brief
320        Remove some target from the MultiTrigger.
321    @param targetStr
322        The target to be removed as a string.
323    */
324    void MultiTrigger::removeTarget(const std::string& targetStr)
325    {
326        Identifier* target = ClassByString(targetStr);
327
328        // If the target is not a valid class name display an error.
329        if (target == nullptr)
330        {
331            orxout(internal_error, context::triggers) << "'" << targetStr << "' is not a valid class name to include in ClassTreeMask (in " << this->getName() << ", class " << this->getIdentifier()->getName() << ")" << endl;
332            return;
333        }
334
335        this->targetMask_.exclude(target);
336    }
337
338    /**
339    @brief
340        This method is called by the MultiTrigger to get information about new trigger events that need to be looked at.
341        This method is the device for the behavior (the conditions under which the MultiTrigger triggers) of any derived class of MultiTrigger.
342    @return
343        Returns a pointer to a queue of MultiTriggerState pointers, containing all the necessary information to decide whether these states should indeed become new states of the MultiTrigger.
344        Please be aware that both the queue and the states in the queue need to be deleted once they have been used. This is already done in the tick() method of this class but would have to be done by any method calling this method.
345    */
346    std::queue<MultiTriggerState*>* MultiTrigger::letTrigger(void)
347    {
348        return nullptr;
349    }
350
351    /**
352    @brief
353        This method can be called by any class inheriting from MultiTrigger to change it's triggered status for a specified originator.
354
355        Compared to the letTrigger mode, which just polls and lets the pollee send it's state changed back, the changeTriggered method lets the trigger advertise its state changes just as they happen so it's much like the interrupt version of letTrigger.
356    @param originator
357        The originator which triggered the state change.
358    */
359    void MultiTrigger::changeTriggered(BaseObject* originator)
360    {
361        MultiTriggerState* state = new MultiTriggerState;
362        state->bTriggered = (!this->isTriggered(originator) & this->isModeTriggered(originator)) ^ this->getInvert();
363        state->originator = originator;
364        this->addState(state);
365    }
366
367    /**
368    @brief
369        This method is called by any child to advertise changes in its state to its parent.
370    @param originator
371        The object that caused the change in activity.
372    */
373    void MultiTrigger::childActivityChanged(BaseObject* originator)
374    {
375        MultiTriggerState* state = new MultiTriggerState;
376        state->bTriggered = (this->isTriggered(originator) & this->isModeTriggered(originator)) ^ this->getInvert();
377        state->originator = originator;
378        this->addState(state);
379    }
380
381    /**
382    @brief
383        Checks whether the children are in such a way that, according to the mode of the MultiTrigger, the MultiTrigger is triggered (only considering the children, not the state of MultiTrigger itself), for a given object.
384        To make an example: When the mode is <em>and</em>, then this would be true or a given object if all the children were triggered for the given object.
385    @param triggerer
386        The object.
387    @return
388        Returns true if the MultiTrigger is triggered concerning it's children.
389    */
390    bool MultiTrigger::isModeTriggered(BaseObject* triggerer)
391    {
392        if(this->children_.size() != 0)
393        {
394            bool triggered = false;
395
396            switch(this->mode_)
397            {
398                case TriggerMode::EventTriggerAND:
399                    triggered = checkAnd(triggerer);
400                    break;
401                case TriggerMode::EventTriggerOR:
402                    triggered = checkOr(triggerer);
403                    break;
404                case TriggerMode::EventTriggerXOR:
405                    triggered = checkXor(triggerer);
406                    break;
407                default: // This will never happen.
408                    triggered = false;
409                    break;
410            }
411
412            return triggered;
413        }
414
415        return true;
416    }
417
418    /**
419    @brief
420        Get whether the MultiTrigger is triggered for a given object.
421    @param triggerer
422        The object.
423    @return
424        Returns true if the MultiTrigger is triggered for the given object.
425    */
426    bool MultiTrigger::isTriggered(BaseObject* triggerer)
427    {
428        std::set<BaseObject*>::iterator it = this->triggered_.find(triggerer);
429        if(it == this->triggered_.end())
430            return false;
431        return true;
432    }
433
434    /**
435    @brief
436        Helper method. Creates an Event for the given status and originator and fires it.
437        Or more precisely creates a MultiTriggerContainer to encompass all neccesary information, creates an Event from that and sends it.
438    @param status
439        The status of the Event to be fired. This is equivalent to the activity of the MultiTrigger.
440    @param originator
441        The object that triggered the MultiTrigger to fire this Event.
442    */
443    void MultiTrigger::fire(bool status, BaseObject* originator)
444    {
445        // If the originator is nullptr, a normal event without MultiTriggerContainer is sent.
446        if(originator == nullptr)
447        {
448            this->fireEvent(status);
449            orxout(verbose, context::triggers) << "MultiTrigger '" <<  this->getName() << "' (&" << this << "): Fired event. status: " << status << "." << endl;
450            return;
451        }
452
453        MultiTriggerContainer* container = new MultiTriggerContainer(this->getContext(), this, originator);
454        this->fireEvent(status, container);
455        orxout(verbose, context::triggers) << "MultiTrigger '" <<  this->getName() << "' (&" << this << "): Fired event. originator: " << originator->getIdentifier()->getName() << " (&" << originator << "), status: " << status << "." << endl;
456        delete container;
457    }
458
459    /**
460    @brief
461        Helper method. Broadcasts an Event for every object that is a target.
462    @param status
463        The status of the Events to be fired. This is equivalent to the activity of the MultiTrigger.
464    */
465    void MultiTrigger::broadcast(bool status)
466    {
467        for(ClassTreeMaskObjectIterator it = this->getTargetMask().begin(); it != this->getTargetMask().end(); ++it)
468            this->fire(status, static_cast<BaseObject*>(*it));
469    }
470
471    /**
472    @brief
473        Helper method. Adds a state to the state queue, where the state will wait to become active.
474    @param state
475        The state to be added.
476    @return
477        Returns true if the state has been added, false if not. If the state has not been added this method destroys it.
478    */
479    bool MultiTrigger::addState(MultiTriggerState* state)
480    {
481        assert(state); // The state really shouldn't be nullptr.
482
483        // If the originator is no target of this MultiTrigger.
484        if(!this->isTarget(state->originator))
485        {
486            delete state;
487            return false;
488        }
489
490        // Add it ot the state queue with the delay specified for the MultiTrigger.
491        this->stateQueue_.emplace_back(this->getDelay(), state);
492
493        return true;
494    }
495
496    /**
497    @brief
498        Checks whether the children amount to true for the <em>and</em> mode for a given object.
499    @param triggerer
500        The object.
501    @return
502        Returns true if all the sub-triggers are active.
503    */
504    bool MultiTrigger::checkAnd(BaseObject* triggerer)
505    {
506        for(TriggerBase* trigger : this->children_)
507        {
508            if(trigger->isMultiTrigger())
509            {
510                if(!static_cast<MultiTrigger*>(trigger)->isActive(triggerer))
511                    return false;
512            }
513            else
514            {
515                if(!trigger->isActive())
516                    return false;
517            }
518        }
519        return true;
520    }
521
522    /**
523    @brief
524        Checks whether the children amount to true for the <em>or</em> mode for a given object.
525    @param triggerer
526        The object.
527    @return
528        Returns true if at least one sub-trigger is active.
529    */
530    bool MultiTrigger::checkOr(BaseObject* triggerer)
531    {
532        for(TriggerBase* trigger : this->children_)
533        {
534            if(trigger->isMultiTrigger())
535            {
536                if(static_cast<MultiTrigger*>(trigger)->isActive(triggerer))
537                    return true;
538            }
539            else
540            {
541                if(trigger->isActive())
542                    return true;
543            }
544        }
545        return false;
546    }
547
548    /**
549    @brief
550        Checks whether the children amount to true for the <em>xor</em> mode for a given object.
551    @param triggerer
552        The object.
553    @return
554        Returns true if exactly one sub-trigger is active.
555    */
556    bool MultiTrigger::checkXor(BaseObject* triggerer)
557    {
558        bool triggered = false;
559        for(TriggerBase* trigger : this->children_)
560        {
561            if(triggered)
562            {
563                if(trigger->isMultiTrigger())
564                {
565                    if(static_cast<MultiTrigger*>(trigger)->isActive(triggerer))
566                        return false;
567                }
568                else
569                {
570                    if(trigger->isActive())
571                        return false;
572                }
573            }
574
575            if(trigger->isMultiTrigger())
576            {
577                if(static_cast<MultiTrigger*>(trigger)->isActive(triggerer))
578                    triggered = true;
579            }
580            else
581            {
582                if(trigger->isActive())
583                    triggered = true;
584            }
585        }
586        return triggered;
587    }
588
589}
Note: See TracBrowser for help on using the repository browser.