2007-05-27 12:01:53 -04:00
|
|
|
// $Id$
|
|
|
|
//
|
|
|
|
// SuperTuxKart - a fun racing game with go-kart
|
|
|
|
// Copyright (C) 2006 Joerg Henrichs
|
|
|
|
//
|
|
|
|
// This program is free software; you can redistribute it and/or
|
|
|
|
// modify it under the terms of the GNU General Public License
|
2008-06-12 20:53:52 -04:00
|
|
|
// as published by the Free Software Foundation; either version 3
|
2007-05-27 12:01:53 -04:00
|
|
|
// 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., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA.
|
|
|
|
|
|
|
|
#include "collectable.hpp"
|
|
|
|
#include "user_config.hpp"
|
1) Removed race_setup and race_mode data structures. All this
information is now only managed by the race_manager, no
more in-between objects to transfer information along.
2) The scores for grand prix are now defined in the stk_config.dat
file (10, 8, 6, 5, 4, .., 1, 0, 0) points
3) Bugfix: unlock information wasn't saved anymore. Added specific
saving after unlocking, plus re-inserted the 'generic' save
at the end of STK again.
4) bugfix/work around: Visual Studio complains about incompatible
iterators in sdldrv - apparently caused by using erase, and
then keep on using the iterator.
5) Fixed bug when running a race in a GP again (scores/times
were added each time).
git-svn-id: svn+ssh://svn.code.sf.net/p/supertuxkart/code/trunk/supertuxkart@1681 178a84e3-b1eb-0310-8ba1-8eac791a3b58
2008-04-09 09:52:48 -04:00
|
|
|
#include "race_manager.hpp"
|
2007-05-27 12:01:53 -04:00
|
|
|
#include "projectile_manager.hpp"
|
|
|
|
#include "kart.hpp"
|
|
|
|
#include "sound_manager.hpp"
|
|
|
|
#include "world.hpp"
|
|
|
|
#include "stk_config.hpp"
|
|
|
|
|
|
|
|
//-----------------------------------------------------------------------------
|
|
|
|
Collectable::Collectable(Kart* kart_)
|
|
|
|
{
|
2007-12-08 08:04:56 -05:00
|
|
|
m_owner = kart_;
|
2008-05-11 23:29:06 -04:00
|
|
|
reset();
|
2007-05-27 12:01:53 -04:00
|
|
|
} // Collectable
|
|
|
|
|
2008-05-11 20:21:34 -04:00
|
|
|
//-----------------------------------------------------------------------------
|
2008-05-11 23:29:06 -04:00
|
|
|
void Collectable::reset()
|
2008-05-11 20:21:34 -04:00
|
|
|
{
|
|
|
|
if(race_manager->getRaceMode()==RaceManager::RM_TIME_TRIAL)
|
|
|
|
{
|
|
|
|
m_type = COLLECT_ZIPPER;
|
|
|
|
m_number = race_manager->getNumLaps();
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
m_type = COLLECT_NOTHING;
|
|
|
|
m_number = 0;
|
|
|
|
}
|
2008-05-11 23:29:06 -04:00
|
|
|
} // reset
|
|
|
|
|
2007-05-27 12:01:53 -04:00
|
|
|
//-----------------------------------------------------------------------------
|
2007-12-08 08:04:56 -05:00
|
|
|
void Collectable::set(CollectableType type, int n)
|
2007-05-27 12:01:53 -04:00
|
|
|
{
|
2007-12-08 08:04:56 -05:00
|
|
|
if (m_type==type)
|
2007-05-27 12:01:53 -04:00
|
|
|
{
|
2007-12-08 08:04:56 -05:00
|
|
|
m_number+=n;
|
2007-05-27 12:01:53 -04:00
|
|
|
return;
|
|
|
|
}
|
2007-12-08 08:04:56 -05:00
|
|
|
m_type=type;
|
|
|
|
m_number=n;
|
2007-05-27 12:01:53 -04:00
|
|
|
} // set
|
|
|
|
|
|
|
|
//-----------------------------------------------------------------------------
|
|
|
|
Material *Collectable::getIcon()
|
|
|
|
{
|
|
|
|
// Check if it's one of the types which have a separate
|
|
|
|
// data file which includes the icon:
|
2007-12-08 08:04:56 -05:00
|
|
|
return collectable_manager->getIcon(m_type);
|
2007-05-27 12:01:53 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
//-----------------------------------------------------------------------------
|
|
|
|
void Collectable::use()
|
|
|
|
{
|
2007-12-08 08:04:56 -05:00
|
|
|
m_number--;
|
|
|
|
switch (m_type)
|
2007-05-27 12:01:53 -04:00
|
|
|
{
|
2007-12-08 08:04:56 -05:00
|
|
|
case COLLECT_ZIPPER: m_owner->handleZipper();
|
2007-05-27 12:01:53 -04:00
|
|
|
break ;
|
2007-12-08 08:04:56 -05:00
|
|
|
case COLLECT_HOMING:
|
2007-05-27 12:01:53 -04:00
|
|
|
case COLLECT_SPARK:
|
|
|
|
case COLLECT_MISSILE:
|
2007-12-08 08:04:56 -05:00
|
|
|
if(m_owner->isPlayerKart())
|
2007-05-27 12:01:53 -04:00
|
|
|
sound_manager->playSfx(SOUND_SHOT);
|
|
|
|
|
2007-12-08 08:04:56 -05:00
|
|
|
projectile_manager->newProjectile(m_owner, m_type);
|
2007-05-27 12:01:53 -04:00
|
|
|
break ;
|
|
|
|
|
|
|
|
case COLLECT_ANVIL:
|
|
|
|
//Attach an anvil(twice as good as the one given
|
|
|
|
//by the bananas) to the kart in the 1st position.
|
2008-04-15 09:57:18 -04:00
|
|
|
for(unsigned int i = 0 ; i < race_manager->getNumKarts(); ++i)
|
2007-05-27 12:01:53 -04:00
|
|
|
{
|
2008-04-15 09:57:18 -04:00
|
|
|
Kart *kart=world->getKart(i);
|
|
|
|
if(kart->isEliminated()) continue;
|
|
|
|
if(kart == m_owner) continue;
|
|
|
|
if(kart->getPosition() == 1)
|
2007-05-27 12:01:53 -04:00
|
|
|
{
|
2008-04-15 09:57:18 -04:00
|
|
|
kart->attach(ATTACH_ANVIL, stk_config->m_anvil_time);
|
|
|
|
kart->adjustSpeedWeight(stk_config->m_anvil_speed_factor*0.5f);
|
2007-05-27 12:01:53 -04:00
|
|
|
|
2008-04-15 09:57:18 -04:00
|
|
|
if(kart->isPlayerKart())
|
2007-05-27 12:01:53 -04:00
|
|
|
sound_manager->playSfx(SOUND_USE_ANVIL);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
break;
|
|
|
|
|
|
|
|
case COLLECT_PARACHUTE:
|
|
|
|
{
|
|
|
|
bool player_affected = false;
|
|
|
|
//Attach a parachutte(that last as twice as the
|
|
|
|
//one from the bananas) to all the karts that
|
|
|
|
//are in front of this one.
|
2008-04-15 09:57:18 -04:00
|
|
|
for(unsigned int i = 0 ; i < race_manager->getNumKarts(); ++i)
|
2007-05-27 12:01:53 -04:00
|
|
|
{
|
2008-04-15 09:57:18 -04:00
|
|
|
Kart *kart=world->getKart(i);
|
|
|
|
if(kart->isEliminated() || kart== m_owner) continue;
|
|
|
|
if(m_owner->getPosition() > kart->getPosition())
|
2007-05-27 12:01:53 -04:00
|
|
|
{
|
2008-04-15 09:57:18 -04:00
|
|
|
kart->attach(ATTACH_PARACHUTE, stk_config->m_parachute_time_other);
|
2007-05-27 12:01:53 -04:00
|
|
|
|
2008-04-15 09:57:18 -04:00
|
|
|
if(kart->isPlayerKart())
|
2007-05-27 12:01:53 -04:00
|
|
|
player_affected = true;
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
if(player_affected)
|
|
|
|
sound_manager->playSfx(SOUND_USE_PARACHUTE);
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
|
|
|
|
case COLLECT_NOTHING:
|
|
|
|
default : break ;
|
|
|
|
}
|
|
|
|
|
2007-12-08 08:04:56 -05:00
|
|
|
if ( m_number <= 0 )
|
2007-05-27 12:01:53 -04:00
|
|
|
{
|
2008-05-11 23:29:06 -04:00
|
|
|
m_number = 0;
|
|
|
|
m_type = COLLECT_NOTHING;
|
2007-05-27 12:01:53 -04:00
|
|
|
}
|
|
|
|
} // use
|
|
|
|
|
|
|
|
//-----------------------------------------------------------------------------
|
|
|
|
void Collectable::hitRedHerring(int n)
|
|
|
|
{
|
|
|
|
//The probabilities of getting the anvil or the parachute increase
|
|
|
|
//depending on how bad the owner's position is. For the first
|
|
|
|
//driver the posibility is none, for the last player is 15 %.
|
|
|
|
|
2007-12-08 08:04:56 -05:00
|
|
|
if(m_owner->getPosition() != 1 && m_type == COLLECT_NOTHING)
|
2007-05-27 12:01:53 -04:00
|
|
|
{
|
2008-04-15 09:57:18 -04:00
|
|
|
const int SPECIAL_PROB = (int)(15.0 / ((float)world->getCurrentNumKarts() /
|
2007-12-08 08:04:56 -05:00
|
|
|
(float)m_owner->getPosition()));
|
2007-05-27 12:01:53 -04:00
|
|
|
const int RAND_NUM = rand()%100;
|
|
|
|
if(RAND_NUM <= SPECIAL_PROB)
|
|
|
|
{
|
|
|
|
//If the driver in the first position has finished, give the driver
|
|
|
|
//the parachute.
|
2008-04-15 09:57:18 -04:00
|
|
|
for(unsigned int i=0; i < race_manager->getNumKarts(); ++i)
|
2007-05-27 12:01:53 -04:00
|
|
|
{
|
2008-04-15 09:57:18 -04:00
|
|
|
Kart *kart = world->getKart(i);
|
|
|
|
if(kart->isEliminated() || kart == m_owner) continue;
|
2008-05-19 23:33:48 -04:00
|
|
|
if(kart->getPosition() == 1 && kart->hasFinishedRace())
|
2007-05-27 12:01:53 -04:00
|
|
|
{
|
2007-12-08 08:04:56 -05:00
|
|
|
m_type = COLLECT_PARACHUTE;
|
|
|
|
m_number = 1;
|
2007-05-27 12:01:53 -04:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2007-12-08 08:04:56 -05:00
|
|
|
m_type = rand()%(2) == 0 ? COLLECT_ANVIL : COLLECT_PARACHUTE;
|
|
|
|
m_number = 1;
|
2007-05-27 12:01:53 -04:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
//rand() is moduled by COLLECT_MAX - 1 - 2 because because we have to
|
|
|
|
//exclude the anvil and the parachute, but later we have to add 1 to prevent
|
|
|
|
//having a value of 0 since that isn't a valid collectable.
|
2007-12-08 08:04:56 -05:00
|
|
|
CollectableType newC;
|
2007-05-27 12:01:53 -04:00
|
|
|
if(!user_config->m_profile)
|
|
|
|
{
|
2008-05-11 20:39:14 -04:00
|
|
|
newC = (CollectableType)(rand()%(COLLECT_MAX - 1 - 2) + 1);
|
2007-05-27 12:01:53 -04:00
|
|
|
}
|
|
|
|
else
|
2008-02-17 07:58:12 -05:00
|
|
|
{ // for now: no collectables when profiling
|
|
|
|
return;
|
2007-05-27 12:01:53 -04:00
|
|
|
// No random effects when profiling!
|
|
|
|
static int simpleCounter=-1;
|
|
|
|
simpleCounter++;
|
2007-12-08 08:04:56 -05:00
|
|
|
newC = (CollectableType)(simpleCounter%(COLLECT_MAX - 1 - 2) + 1);
|
2007-05-27 12:01:53 -04:00
|
|
|
}
|
2007-12-08 08:04:56 -05:00
|
|
|
if(m_type==COLLECT_NOTHING)
|
2007-05-27 12:01:53 -04:00
|
|
|
{
|
2007-12-08 08:04:56 -05:00
|
|
|
m_type=newC;
|
|
|
|
m_number = n;
|
2007-05-27 12:01:53 -04:00
|
|
|
}
|
2007-12-08 08:04:56 -05:00
|
|
|
else if(newC==m_type)
|
2007-05-27 12:01:53 -04:00
|
|
|
{
|
2007-12-08 08:04:56 -05:00
|
|
|
m_number+=n;
|
|
|
|
if(m_number > MAX_COLLECTABLES) m_number = MAX_COLLECTABLES;
|
2007-05-27 12:01:53 -04:00
|
|
|
}
|
|
|
|
// Ignore new collectable if it is different from the current one
|
|
|
|
} // hitRedHerring
|