Planet
navi homePPSaboutscreenshotsdownloaddevelopmentforum

source: code/branches/towerdefenseHS14/src/modules/towerdefense/TowerDefense.cc @ 10158

Last change on this file since 10158 was 10158, checked in by maxima, 10 years ago

Changes of Nico. Upgrading the towers included.

  • Property svn:eol-style set to native
File size: 14.0 KB
RevLine 
[9098]1/*
[9112]2 *   ORXONOX - the hottest 3D action shooter ever to exist
3 *                    > www.orxonox.net <
[9098]4 *
[9112]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:
25 *      ...
26 *
27 *NACHRICHT:
28 *
29 * Hier empfehle ich euch die gesamte Spielogik unter zu bringen. Viele Funktionen werden automatisch
30 * bei gewissen Ereignissen aufgerufen bzw. lösen Ereignisse aus
31 *
32 *Z.B:
33 * start() //wird aufgerufen, bevor das Spiel losgeht
34 * end() //wenn man diese Funktion aufruft wird
35 * pawnKilled() // wird aufgerufen, wenn ein Pawn stirbt (z.B: wenn )
36 * playerScored() // kann man aufrufen um dem Spieler Punkte zu vergeben.
37 *
38 *
39 *
40 *TIPP: Eclipse hilft euch schnell auf bereits vorhanden Funktionen zuzugreifen:
41 * einfach "this->" eingeben und kurz warten. Dann tauch eine Liste mit Vorschlägen auf. Wenn ihr jetzt weiter
42 * tippt, werden die Vorschläge entsprechend gefiltert.
43 *
44 *
45 *TIPP: schaut euch mal Tetris::createStone() an. Dort wird ein TetrisStone-Objekt (ControllableEntity) erzeugt,
46 * ihm ein Template zugewiesen (welches vorher im Level definiert wurde und dem CenterPoint übergeben wurde)
47 * Ähnlich könnt ihr vorgehen, um einen Turm zu erzeugen. (Zusätzlich braucht ein Turm noch einen Controller)
48 * z.B: WaypointPatrolController. Wenn kein Team zugewiesen wurde bekämpft ein WaypointPatrolController alles,
49 * was in seiner Reichweite liegt.
50 *
[9143]51 *
52 *HUD:
53 * Ein Gametype kann ein HUD (Head up Display haben.) Z.B: hat Pong eine Anzeige welcher Spieler wieviele Punkte hat.
54 * Generell kann man a) Grafiken oder b) Zeichen in einer HUD anzeigen.
55 * Fuer den ersten Schritt reicht reiner Text.
56 *
57 * a)
58 * PongScore.cc uebernehmen und eigene Klasse draus machen.
59 * Wenn ihr bloss anzeigen wollt wieviele Punkte der Spieler bereits erspielt hat (Punkte = Kapital fuer neue Tuerme) dann orientiert ihr euch an
60 * TetrisScore.cc (im pCuts branch): http://www.orxonox.net/browser/code/branches/pCuts/src/modules/tetris/TetrisScore.cc
61 * Ich habe TetrisScore lediglich dazu gebraucht, um eine Variable auf dem HUD auszugeben. Ein Objekt fuer statischen Text gibt es bereits.
62 *
63 * b)
64 * Im naesten Schritt erstellt man die Vorlage fuer das HUD-Objekt: siehe /data/overlays/pongHUD
65 * OverlayText ist eine Vorlage fuer statischen text zb: "Points Scored:". Aus mir nicht erklaerlichen Gruenden sollte man die OverlayText
66 * Objekte immer erst nach dem PongScore anlegen.
67 *
[9180]68 * c)  Im TowerDefense gamtype muss im Constructor noch das HUD-Template gesetzt werden.
[9143]69 *
70 * d) in CMakeLists.txt noch das Module includen das fuer die Overlays zustaendig ist. Siehe das gleiche File im Pong module.
71 *
72 *
73 *
[9098]74 */
75
[9112]76#include "TowerDefense.h"
[10142]77#include "TowerDefenseTower.h"
[10091]78#include "TowerTurret.h"
[9137]79#include "TowerDefenseCenterpoint.h"
[10105]80//#include "TDCoordinate.h"
[9122]81#include "worldentities/SpawnPoint.h"
82#include "worldentities/pawns/Pawn.h"
[9129]83#include "worldentities/pawns/SpaceShip.h"
[9142]84#include "controllers/WaypointController.h"
[9272]85
[9142]86#include "graphics/Model.h"
87#include "infos/PlayerInfo.h"
[9138]88#include "chat/ChatManager.h"
[9667]89#include "core/CoreIncludes.h"
[9138]90
[9136]91/* Part of a temporary hack to allow the player to add towers */
92#include "core/command/ConsoleCommand.h"
93
[9112]94namespace orxonox
95{
[9667]96    RegisterUnloadableClass(TowerDefense);
[9272]97
[9667]98    TowerDefense::TowerDefense(Context* context) : Deathmatch(context)
[9112]99    {
100        RegisterObject(TowerDefense);
[9272]101
[9180]102        this->setHUDTemplate("TowerDefenseHUD");
103
[9272]104        this->stats_ = new TowerDefensePlayerStats();
[9207]105
[9272]106        /* Temporary hack to allow the player to add towers */
107        this->dedicatedAddTower_ = createConsoleCommand( "addTower", createExecutor( createFunctor(&TowerDefense::addTower, this) ) );
[10158]108        this->dedicatedUpgradeTower_ = createConsoleCommand( "upgradeTower", createExecutor( createFunctor(&TowerDefense::upgradeTower, this) ) );
[9112]109    }
[9272]110
[9136]111    TowerDefense::~TowerDefense()
112    {
[9272]113        /* Part of a temporary hack to allow the player to add towers */
[9136]114        if (this->isInitialized())
115        {
116            if( this->dedicatedAddTower_ )
117                delete this->dedicatedAddTower_;
118        }
119    }
[9272]120
121    void TowerDefense::setCenterpoint(TowerDefenseCenterpoint *centerpoint)
122    {
123        orxout() << "Centerpoint now setting..." << endl;
124        this->center_ = centerpoint;
125        orxout() << "Centerpoint now set..." << endl;
126    }
127
[9112]128    void TowerDefense::start()
129    {
[10109]130        orxout() << "test0" << endl;
131
[9112]132        Deathmatch::start();
[10158]133        credits = 5000;
134        life = 20;
135        waves = 0;
136        time=0.0;
[10142]137
138// Waypoints: [1,3] [10,3] [10,11] [13,11]
139        for (int i=0; i < 16 ; i++){
140                for (int j = 0; j< 16 ; j++){
141                        towermatrix[i][j] = false;
142                }
143        }
144        for (int k=0; k<3; k++)
145                towermatrix[1][k]=true;
146        for (int l=1; l<11; l++)
147                towermatrix[l][3]=true;
148        for (int m=3; m<12; m++)
149                towermatrix[10][m]=true;
150        for (int n=10; n<14; n++)
151                towermatrix[n][11]=true;
152        for (int o=13; o<16; o++)
153                towermatrix[13][o]=true;
154
155
[10140]156        credits = 5000;
157        life = 20;
158        waves = 0;
[10132]159        time=0.0;
160
[10142]161        const int kInitialTowerCount = 3;
[9175]162
[9272]163        for (int i = 0; i < kInitialTowerCount; i++)
164        {
[10144]165            addTower(i+4,i+5);
[9272]166        }
167
168
[10142]169
[10109]170        //add some TowerDefenseEnemys
171
172
[10123]173
174
175
[10109]176        //ChatManager::message("Use the console command addTower x y to add towers");
177
[9272]178        //TODO: let the player control his controllable entity && TODO: create a new ControllableEntity for the player
179    }
180
[10125]181    // Generates a TowerDefenseEnemy. Uses Template "enemytowerdefense". Sets position at first waypoint of path.
182
[10141]183     void TowerDefense::addTowerDefenseEnemy(std::vector<TDCoordinate*> path, int templatenr){ 
[10105]184
[10109]185        TowerDefenseEnemy* en1 = new TowerDefenseEnemy(this->center_->getContext());
[10141]186       
187        switch(templatenr) 
188        { 
189        case 1 : 
190                en1->addTemplate("enemytowerdefense1"); 
191                en1->setScale(3); 
192                break; 
193        case 2 : 
194                en1->addTemplate("enemytowerdefense2");
195                en1->setScale(2); 
196                break; 
197        case 3 : 
198                en1->addTemplate("enemytowerdefense3"); 
199                en1->setScale(1); 
200                break; 
201        } 
202                 
[10126]203        en1->getController();
[10125]204
205        en1->setPosition(path.at(0)->get3dcoordinate());
[10123]206
207
208        TowerDefenseEnemyvector.push_back(en1);
209
[10109]210
[10123]211        for(unsigned int i = 0; i < path.size(); ++i)
[10109]212        {
[10123]213            en1->addWaypoint((path.at(i)));
214        }
[10109]215
[10123]216
[10132]217       /*
[10123]218        WaypointController *newController = new WaypointController(en1->getContext());//
[10109]219        newController->setAccuracy(3);
220
221        for(int i =0; i < path.size(); ++i)
222        {
223
[10123]224            Model *wayPoint = new Model(en1->getContext());
225            wayPoint->setMeshSource("cube.mesh");
226            wayPoint->setPosition(path.at(i)->get3dcoordinate());
[10109]227            wayPoint->setScale(0.2);
228            newController->addWaypoint(wayPoint);
229        }*/
230    }
231
232
[9272]233    void TowerDefense::end()
[10123]234
[9123]235    {
[9272]236        Deathmatch::end();
237
238        ChatManager::message("Match is over");
239    }
240
[10158]241    void TowerDefense::upgradeTower(int x,int y)
242    {
243        const TowerCost upgraeCost = TDDefaultUpgradeCost;
[10105]244
[10158]245        if (!this->hasEnoughCreditForTower(upgraeCost))
246        {
247            orxout() << "not enough credit: " << (this->stats_->getCredit()) << " available, " << TDDefaultTowerCost << " needed.";
248            return;
249        }
[10105]250
[10158]251        if (towermatrix [x][y] == NULL)
252        {
253            orxout() << "no tower on this position" << endl;
254            return;
255        }
256
257        else
258        {
259                (towermatrix [x][y])->upgradeTower();
260        }
261    }
262
263
[9272]264    void TowerDefense::addTower(int x, int y)
[10142]265    {
[10123]266        const TowerCost towerCost = TDDefaultTowerCost;
[9272]267
268        if (!this->hasEnoughCreditForTower(towerCost))
269        {
270            orxout() << "not enough credit: " << (this->stats_->getCredit()) << " available, " << TDDefaultTowerCost << " needed.";
271            return;
272        }
273
[10142]274        if (towermatrix [x][y]==true)
[9272]275        {
[10142]276            orxout() << "not possible to put tower here!!" << endl;
[9272]277            return;
278        }
279
[10109]280
[9272]281        unsigned int width = this->center_->getWidth();
282        unsigned int height = this->center_->getHeight();
283
[10109]284
[9272]285        int tileScale = (int) this->center_->getTileScale();
286
287        if (x > 15 || y > 15 || x < 0 || y < 0)
288        {
289            //Hard coded: TODO: let this depend on the centerpoint's height, width and fieldsize (fieldsize doesn't exist yet)
290            orxout() << "Can not add Tower: x and y should be between 0 and 15" << endl;
291            return;
292        }
293
294        orxout() << "Will add tower at (" << (x-8) * tileScale << "," << (y-8) * tileScale << ")" << endl;
295
[10142]296       
[10105]297
[10142]298       //Reduce credit
299        this->stats_->buyTower(towerCost);
[10105]300
[10142]301        orxout() << "Credit: " << this->stats_->getCredit() << endl;
[9272]302
[10142]303  //    Create tower
304        TowerDefenseTower* towernew = new TowerDefenseTower(this->center_->getContext());
[10144]305        towernew->addTemplate("towerturret");
[10142]306        towernew->setPosition(static_cast<float>((x-8) * tileScale), static_cast<float>((y-8) * tileScale), 75);
307        towernew->setGame(this);
[9272]308
[10142]309        towermatrix[x][y]=true;
[9272]310    }
311
312    bool TowerDefense::hasEnoughCreditForTower(TowerCost towerCost)
313    {
314        return ((this->stats_->getCredit()) >= towerCost);
315    }
316
[10158]317    bool TowerDefense::hasEnoughCreditForUpgrade()
318    {
319        return true;
320    }
321
[10142]322 
[9272]323    void TowerDefense::tick(float dt)
324    {
[10123]325        SUPER(TowerDefense, tick, dt);
[10132]326        time +=dt;
[10123]327
[10132]328        TDCoordinate* coord1 = new TDCoordinate(1,1);
329        std::vector<TDCoordinate*> path;
330        path.push_back(coord1);
[10123]331
[10151]332        if(time>1 && TowerDefenseEnemyvector.size() < 30)
333        {
334            addTowerDefenseEnemy(path, rand() %3 +1 );
335            time = time-1;
336        }
[10132]337
[10151]338        Vector3* endpoint = new Vector3(500, 700, 150);
[10132]339
[10151]340        for(int i =0; i < TowerDefenseEnemyvector.size(); ++i)
341        {
342
343                //orxout() << TowerDefenseEnemyvector.at(i) << endl;
344                //continue;
345
346                        //ArtificialController* controller = (ArtificialController*)this->getController();
347                        if(TowerDefenseEnemyvector.at(i) != NULL && TowerDefenseEnemyvector.at(i)->isAlive())
348                        {
349                                orxout() << "Variable i: " << i << endl;
350
351                                Vector3 ship = TowerDefenseEnemyvector.at(i)->getRVWorldPosition();
352
353                                float distance = ship.distance(*endpoint);
354
355                                //orxout() << "distance" << distance << endl;
356                                if(distance <50){
357                                  //    orxout() << "ENEMY KILLED!!!!" << endl;
358                                  TowerDefenseEnemyvector.at(i)->destroy();
359                                }
360                        }
361                        else
362                        {
363                                //TowerDefenseEnemyvector.erase(TowerDefenseEnemyvector.begin() +i);
364                        }
365                }
366
367        int count= 0;
368        for(int i =0; i < TowerDefenseEnemyvector.size(); ++i)
369        {
370                if(TowerDefenseEnemyvector.at(i)!= NULL)
371                {
372                        ++count;
373                }
374        }
375
376        if(count== 0)
377        {
378                time2 +=dt;
379                if(time2 > 10)
380                {
381                        TowerDefenseEnemyvector.clear();
382                        time=0;
383                        time2=0;
384                }
385        }
386
387
388
[9123]389    }
[9148]390
[9272]391    // Function to test if we can add waypoints using code only. Doesn't work yet
392
393    // THE PROBLEM: WaypointController's getControllableEntity() returns null, so it won't track. How do we get the controlableEntity to NOT BE NULL???
394    /*
395    void TowerDefense::addWaypointsAndFirstEnemy()
396    {
397        SpaceShip *newShip = new SpaceShip(this->center_);
398        newShip->addTemplate("spaceshipassff");
399
400        WaypointController *newController = new WaypointController(newShip);
401        newController->setAccuracy(3);
402
403        Model *wayPoint1 = new Model(newController);
404        wayPoint1->setMeshSource("crate.mesh");
405        wayPoint1->setPosition(7,-7,5);
406        wayPoint1->setScale(0.2);
407
408        Model *wayPoint2 = new Model(newController);
409        wayPoint2->setMeshSource("crate.mesh");
410        wayPoint2->setPosition(7,7,5);
411        wayPoint2->setScale(0.2);
412
413        newController->addWaypoint(wayPoint1);
414        newController->addWaypoint(wayPoint2);
415
416        // The following line causes the game to crash
417
418        newShip->setController(newController);
419//        newController -> getPlayer() -> startControl(newShip);
420        newShip->setPosition(-7,-7,5);
421        newShip->setScale(0.1);
422        //newShip->addSpeed(1);
423
424
425
426//      this->center_->attach(newShip);
427    }
428    */
429    /*
430    void TowerDefense::playerEntered(PlayerInfo* player)
431    {
432        Deathmatch::playerEntered(player);
433
434        const std::string& message = player->getName() + " entered the game";
435        ChatManager::message(message);
436    }
437
438    bool TowerDefense::playerLeft(PlayerInfo* player)
439    {
440        bool valid_player = Deathmatch::playerLeft(player);
441
442        if (valid_player)
443        {
444            const std::string& message = player->getName() + " left the game";
445            ChatManager::message(message);
446        }
447
448        return valid_player;
449    }
450
451
452    void TowerDefense::pawnKilled(Pawn* victim, Pawn* killer)
453    {
454        if (victim && victim->getPlayer())
455        {
456            std::string message;
457            if (killer)
458            {
459                if (killer->getPlayer())
460                    message = victim->getPlayer()->getName() + " was killed by " + killer->getPlayer()->getName();
461                else
462                    message = victim->getPlayer()->getName() + " was killed";
463            }
464            else
465                message = victim->getPlayer()->getName() + " died";
466
467            ChatManager::message(message);
468        }
469
470        Deathmatch::pawnKilled(victim, killer);
471    }
472
[9347]473    void TowerDefense::playerScored(PlayerInfo* player, int score)
[9272]474    {
[9347]475        Gametype::playerScored(player, score);
[9272]476    }*/
[9098]477}
Note: See TracBrowser for help on using the repository browser.