Planet
navi homePPSaboutscreenshotsdownloaddevelopmentforum

source: code/branches/pCuts/src/modules/tetris/Tetris.cc @ 9352

Last change on this file since 9352 was 9092, checked in by jo, 13 years ago

Tetris on the way to get completed. I fixed some bugs concerning rotation, false brick placement on brick-brick collisions, clearing multiple rows and added a preview brick feature and a basic hud.

  • Property svn:eol-style set to native
File size: 15.5 KB
RevLine 
[8249]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 *      ...
24 *   Co-authors:
[9087]25 *      Johannes Ritz
[8249]26 *
[9089]27 *
[9087]28 *BUG c) destroying the old stones causes segfault -> WeakPointer as solution ?
[9089]29 *BUG d) wrong collision detection: sometimes stones "bounce off"
[9087]30 *
[9090]31 *
[9087]32 *TASK c) end the game in a nicer way
33 *TASK d) save the highscore
34 *TASK e) eye candy
[8249]35 */
36
37/**
38    @file Tetris.cc
39    @brief Implementation of the Tetris class.
40*/
41
42#include "Tetris.h"
43
44#include "core/CoreIncludes.h"
45#include "core/EventIncludes.h"
46#include "core/command/Executor.h"
47
48#include "gamestates/GSLevel.h"
49
50#include "TetrisCenterpoint.h"
51#include "TetrisStone.h"
[9082]52#include "TetrisBrick.h"
[8249]53#include "infos/PlayerInfo.h"
54
55namespace orxonox
56{
57
58    CreateUnloadableFactory(Tetris);
59
60    /**
61    @brief
62        Constructor. Registers and initializes the object.
[9092]63    @ingroup Tetris
[8249]64    */
65    Tetris::Tetris(BaseObject* creator) : Deathmatch(creator)
66    {
67        RegisterObject(Tetris);
68
[9082]69        this->activeBrick_ = NULL;
[8249]70
71        // Pre-set the timer, but don't start it yet.
[9082]72        this->starttimer_.setTimer(1.0, false, createExecutor(createFunctor(&Tetris::startBrick, this)));
[8249]73        this->starttimer_.stopTimer();
[8564]74
75        this->player_ = NULL;
[9087]76        this->endGameCriteria_ = 0.0f;
[9092]77        this->setHUDTemplate("TetrisHUD");
78        this->futureBrick_ = NULL;
[8249]79    }
80
81    /**
82    @brief
83        Destructor. Cleans up, if initialized.
84    */
85    Tetris::~Tetris()
86    {
87        if (this->isInitialized())
88            this->cleanup();
89    }
90
91    /**
92    @brief
93        Cleans up the Gametype.
94    */
95    void Tetris::cleanup()
96    {
[9082]97        /*for(int i = 0;i < this->stones_.size(); i++) //TODO: Why isn't there any code like this
98        {                                              // compensating the 'new' statement?
99            delete this->stones_[i];
100        }//*/
[8564]101
[8249]102    }
[8537]103
[8249]104    void Tetris::tick(float dt)
105    {
106        SUPER(Tetris, tick, dt);
[8537]107
[9087]108        if((this->activeBrick_ != NULL)&&(!this->hasEnded()))
[8537]109        {
[9087]110                this->endGameCriteria_ += dt;
[9082]111            if(!this->isValidBrickPosition(this->activeBrick_, this->activeBrick_->getPosition()))
[8563]112            {
[9082]113                this->activeBrick_->setVelocity(Vector3::ZERO);
[9085]114                this->activeBrick_->releaseStones(this->center_);
115                //delete this->activeBrick_; //releasing the memory
[9087]116                this->findFullRows();
117                if(this->endGameCriteria_ < 0.1f) //end game if two bricks are created within a 0.1s interval.
118                    this->end();
[9082]119                this->createBrick();
120                this->startBrick();
[9087]121                this->endGameCriteria_ = 0.0f;
[8563]122            }
[8537]123        }
124    }
125
[8566]126    bool Tetris::isValidMove(TetrisStone* stone, const Vector3& position)
[8537]127    {
128        assert(stone);
[8563]129
[8537]130        if(position.x < this->center_->getStoneSize()/2.0)  //!< If the stone touches the left edge of the level
[8566]131            return false;
[8537]132        else if(position.x > (this->center_->getWidth()-0.5)*this->center_->getStoneSize()) //!< If the stone touches the right edge of the level
[8566]133            return false;
[8488]134
[8563]135        for(std::vector<TetrisStone*>::const_iterator it = this->stones_.begin(); it != this->stones_.end(); ++it)
136        {
137            if(stone == *it)
138                continue;
139
140            const Vector3& currentStonePosition = (*it)->getPosition(); //!< Saves the position of the currentStone
141
[8566]142            if((position.x == currentStonePosition.x) && abs(position.y-currentStonePosition.y) < this->center_->getStoneSize())
143                return false;
[8565]144        }
145
[8566]146        return true;
[8565]147    }
148
[9082]149    /**
150    @brief
[9084]151        Check for each stone in a brick if it is moved the right way.
[9082]152    */
[9084]153    bool Tetris::isValidMove(TetrisBrick* brick, const Vector3& position, bool isRotation = false)
[9082]154    {
155        assert(brick);
156
157        for (unsigned int i = 0; i < brick->getNumberOfStones(); i++ )
158        {
159            TetrisStone* stone = brick->getStone(i);
[9092]160            Vector3 stonePosition; //the current stone's offset to position
[9084]161            if(isRotation)
162                stonePosition = rotateVector(stone->getPosition(), brick->getRotationCount()+1);
163            else
164                stonePosition = rotateVector(stone->getPosition(), brick->getRotationCount());
165
[9092]166            if(! this->isValidMove(stone, position + stonePosition ))
[9084]167            {
[9082]168                return false;
[9084]169            }
[9092]170
171            //catch illegal rotation (such that collisions with ground are not permitted)
172            if(isRotation)
173            {
174                if((position + stonePosition).y < this->center_->getStoneSize()/2.0f) //!< If the stone has reached the bottom of the level
175                {
176                    return false;
177                }
178            }
[9082]179        }
180        return true;
181
182    }
183
184
185
[8567]186    bool Tetris::isValidStonePosition(TetrisStone* stone, const Vector3& position)
[8565]187    {
188        assert(stone);
189
[8661]190        // we use a reverse iterator because we have to check for collisions with the topmost stones first
191        for(std::vector<TetrisStone*>::const_reverse_iterator it = this->stones_.rbegin(); it != this->stones_.rend(); ++it)
[8565]192        {
[9089]193            if(this->activeBrick_->contains(*it))//skip the moving brick' stones
[8565]194                continue;
[9085]195            //Vector3 currentStonePosition = rotateVector((*it)->getPosition(), this->activeBrick_->getRotationCount());
196            const Vector3& currentStonePosition = (*it)->getPosition(); //!< Saves the position of the currentStone
[9084]197            //!< Saves the position of the currentStone
[8565]198
[9089]199            //filter out cases where the falling stone is already below a steady stone
200            if(position.y < currentStonePosition.y - this->center_->getStoneSize()/2.0f)
201                continue;
[8565]202            if((position.x == currentStonePosition.x) && (position.y < currentStonePosition.y + this->center_->getStoneSize()))
[9086]203            {
[9092]204                int y_offset = static_cast<int>((this->activeBrick_->getPosition().y-currentStonePosition.y+10)/10)*10 + currentStonePosition.y;
205                if(y_offset < 0) //filter out extreme cases (very rare bug)
206                        y_offset = 0;
207                this->activeBrick_->setPosition(Vector3(this->activeBrick_->getPosition().x, y_offset, this->activeBrick_->getPosition().z));
[8567]208                return false;
[8563]209            }// This case applies if the stones overlap partially vertically
210        }
211
[8661]212        // after we checked for collision with all stones, we also check for collision with the bottom
213        if(position.y < this->center_->getStoneSize()/2.0f) //!< If the stone has reached the bottom of the level
[9084]214        {
[9092]215                int yOffset = stone->getPosition().y + this->center_->getStoneSize()/2.0f;//calculate offset
216                if(yOffset < 0) //catch brake-throughs
217                    yOffset = 0;
218                this->activeBrick_->setPosition(Vector3(this->activeBrick_->getPosition().x, yOffset, this->activeBrick_->getPosition().z));
[8661]219            return false;
220        }
221
[8567]222        return true;
[8249]223    }
[9092]224    /**
225     * @brief This function determines wether a brick touches another brick or the ground.
226     *
227     */
[9082]228    bool Tetris::isValidBrickPosition(TetrisBrick* brick, const Vector3& position)
229    {
230        assert(brick);
231
232        for (unsigned int i = 0; i < brick->getNumberOfStones(); i++ )
233        {
234            TetrisStone* stone = brick->getStone(i);
[9084]235            Vector3 stonePosition = rotateVector(stone->getPosition(), brick->getRotationCount());
236            if(! this->isValidStonePosition(stone, position + stonePosition) )
[9082]237                return false;
238        }
239        return true;
[9084]240    }
[9082]241
[9084]242    /**
243    @brief
[9085]244        A Vector3 is rolled 90 * degrees * amount (anticlockwise rotation)
[9084]245    */
246    Vector3 Tetris::rotateVector(Vector3 position, unsigned int amount)
247    {
248        int temp = 0;
249        for(unsigned int i = 0; i < amount; i++)
250        {
251            temp = position.x;
252            position.x = -position.y;
253            position.y = temp;
254        }
255        return position;
[9082]256    }
257
[8249]258    /**
259    @brief
260        Starts the Tetris minigame.
261    */
262    void Tetris::start()
263    {
264        if (this->center_ != NULL) // There needs to be a TetrisCenterpoint, i.e. the area the game takes place.
265        {
[9092]266            // Create the futureBrick_
267            this->createBrick();
[9085]268            // Create the first brick.
[9082]269            this->createBrick();
[8249]270        }
271        else // If no centerpoint was specified, an error is thrown and the level is exited.
272        {
[8858]273            orxout(internal_error) << "Tetris: No Centerpoint specified." << endl;
[8249]274            GSLevel::startMainMenu();
275            return;
276        }
277
[8537]278        // Start the timer. After it has expired the stone is started.
[8249]279        this->starttimer_.startTimer();
280
281        // Set variable to temporarily force the player to spawn.
282        bool temp = this->bForceSpawn_;
283        this->bForceSpawn_ = true;
284
285        // Call start for the parent class.
286        Deathmatch::start();
287
288        // Reset the variable.
289        this->bForceSpawn_ = temp;
290    }
291
292    /**
293    @brief
294        Ends the Tetris minigame.
295    */
296    void Tetris::end()
297    {
[9087]298        this->activeBrick_->setVelocity(Vector3::ZERO);
[9089]299        if(this->activeBrick_ != NULL)
300        {
301            this->player_->stopControl();
302        }
303
[8249]304        this->cleanup();
305
306        // Call end for the parent class.
307        Deathmatch::end();
308    }
309
310    /**
311    @brief
312        Spawns player.
313    */
314    void Tetris::spawnPlayersIfRequested()
315    {
316        // Spawn a human player.
317        for (std::map<PlayerInfo*, Player>::iterator it = this->players_.begin(); it != this->players_.end(); ++it)
318            if (it->first->isHumanPlayer() && (it->first->isReadyToSpawn() || this->bForceSpawn_))
319                this->spawnPlayer(it->first);
320    }
321
322    /**
323    @brief
324        Spawns the input player.
325    @param player
326        The player to be spawned.
327    */
328    void Tetris::spawnPlayer(PlayerInfo* player)
329    {
330        assert(player);
331
[8564]332        if(this->player_ == NULL)
[8249]333        {
334            this->player_ = player;
335            this->players_[player].state_ = PlayerState::Alive;
336        }
337    }
338
[9082]339
340
341    void Tetris::startBrick(void)
[8249]342    {
343        if(this->player_ == NULL)
344            return;
[8680]345
346        unsigned int cameraIndex = 0;
[9082]347        if(this->activeBrick_ != NULL)
[8680]348        {
349            // Get camera settings
[9082]350            cameraIndex = this->activeBrick_->getCurrentCameraIndex();
[8249]351            this->player_->stopControl();
[8680]352        }
[9082]353
354        // Make the last brick to be created the active brick.
355        this->activeBrick_ = this->bricks_.back();
356
357        this->player_->startControl(this->activeBrick_);
358        this->activeBrick_->setVelocity(0.0f, -this->center_->getStoneSpeed(), 0.0f);
359        this->activeBrick_->setCameraPosition(cameraIndex);
[8249]360    }
361
[9082]362    void Tetris::createBrick(void)             //TODO: random rotation offset between 0 and 3 (times 90°)
[8249]363    {
[9092]364        // Use the futureBrick_ as new currentBrick by setting its position and storing it in this->bricks
365        if(this->futureBrick_ != NULL)
[9082]366        {
[9092]367            for (unsigned int i = 0; i < this->futureBrick_->getNumberOfStones(); i++)
368            {
369                this->stones_.push_back(this->futureBrick_->getStone(i));
370            }
371            float xPos = (this->center_->getWidth()/2 + ((this->center_->getWidth() % 2)*2-1)/2.0f)*this->center_->getStoneSize();
372            float yPos = (this->center_->getHeight()-0.5f)*this->center_->getStoneSize();
373            this->futureBrick_->setPosition(xPos, yPos, 0.0f);
374            this->bricks_.push_back(this->futureBrick_);
[9082]375        }
[9092]376        // create new futureBrick_
377        this->futureBrick_ = new TetrisBrick(this->center_);
[9082]378
[9092]379
[8249]380        // Apply the stone template to the stone.
[9092]381        this->futureBrick_->addTemplate(this->center_->getBrickTemplate());
[9082]382
[9092]383        // Attach the brick to the Centerpoint and set the position of the brick to be at the left side.
384        this->center_->attach(this->futureBrick_);
385        float xPos = (this->center_->getWidth()*1.6 + ((this->center_->getWidth() % 2)*2-1)/2.0f)*this->center_->getStoneSize();
386        float yPos = (this->center_->getHeight()-5.1f)*this->center_->getStoneSize();
387        this->futureBrick_->setPosition(xPos, yPos, 0.0f);
388        this->futureBrick_->setGame(this);
[8249]389    }
390
[9082]391
[8249]392    /**
393    @brief
394        Get the player.
395    @return
396        Returns a pointer to the player. If there is no player, NULL is returned.
397    */
398    PlayerInfo* Tetris::getPlayer(void) const
399    {
400        return this->player_;
401    }
402
[9082]403    /*TetrisCenterpoint* Tetris::getCenterpoint(void) const
404    {
405        return this->center_;
406    }*/
407
[8537]408    /**
409    @brief Set the TetrisCenterpoint (the playing field).
410    @param center A pointer to the TetrisCenterpoint to be set.
411    */
412    void Tetris::setCenterpoint(TetrisCenterpoint* center)
413    {
414        this->center_ = center;
415    }
416
[9086]417    /**
[9092]418    @brief Check each row if it is full. Removes all full rows. Update
419    @brief Manages score.
[9086]420    */
[9087]421    void Tetris::findFullRows()
[9086]422    {
[9087]423        unsigned int correctPosition = 0;
[9086]424        unsigned int stonesPerRow = 0;
[9087]425        for (unsigned int row = 0; row < this->center_->getHeight(); row++)
[9086]426        {
427            stonesPerRow = 0;
[9092]428            for(std::vector<TetrisStone*>::iterator it = this->stones_.begin(); it != this->stones_.end(); ++it)
[9086]429            {
[9087]430                correctPosition = static_cast<unsigned int>(((*it)->getPosition().y - 5)/this->center_->getStoneSize());
431                if(correctPosition == row)
[9089]432                {
433                    stonesPerRow++;
434                    if(stonesPerRow == this->center_->getWidth())
435                    {
436                        clearRow(row);
[9092]437                        row--; //the row counter has to be decreased in order to detect multiple rows!
[9090]438                        this->playerScored(this->player_);// add points
439                        //increase the stone's speed
440                        this->center_->setStoneSpeed(this->center_->getStoneSpeed()+1.0f);
[9089]441                    }
442                }
[9086]443            }
444        }
445    }
446
[9087]447    void Tetris::clearRow(unsigned int row)
[9089]448    {// clear the full row
[9087]449        for(std::vector<TetrisStone*>::iterator it = this->stones_.begin(); it != this->stones_.end(); ++it)
450        {
451            if(static_cast<unsigned int>(((*it)->getPosition().y - 5)/this->center_->getStoneSize()) == row)
[9089]452                (*it)->setPosition(Vector3(-50,-50,100));
[9087]453                //{(*it)->destroy(); this->stones_.erase(it); orxout()<< "destroy row "<<endl;}//experimental
454        }
[9089]455      // adjust height of stones above the deleted row //TODO: check if this could be a source of a bug.
456        for(std::vector<TetrisStone*>::iterator it2 = this->stones_.begin(); it2 != this->stones_.end(); ++it2)
[9087]457        {
[9089]458            if(static_cast<unsigned int>(((*it2)->getPosition().y - 5)/this->center_->getStoneSize()) > row)
459                (*it2)->setPosition((*it2)->getPosition()-Vector3(0,10,0));
[9087]460        }
461
462    }
463
464
[8249]465}
Note: See TracBrowser for help on using the repository browser.