2006-02-24 04:48:15 +00:00
|
|
|
// Emacs style mode select -*- C++ -*-
|
|
|
|
//-----------------------------------------------------------------------------
|
|
|
|
//
|
|
|
|
// $Id:$
|
|
|
|
//
|
|
|
|
// Copyright (C) 1993-1996 by id Software, Inc.
|
|
|
|
//
|
|
|
|
// This source is available for distribution and/or modification
|
|
|
|
// only under the terms of the DOOM Source Code License as
|
|
|
|
// published by id Software. All rights reserved.
|
|
|
|
//
|
|
|
|
// The source is distributed in the hope that it will be useful,
|
|
|
|
// but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
|
|
// FITNESS FOR A PARTICULAR PURPOSE. See the DOOM Source Code License
|
|
|
|
// for more details.
|
|
|
|
//
|
|
|
|
// $Log:$
|
|
|
|
//
|
|
|
|
// DESCRIPTION:
|
|
|
|
// Cheat sequence checking.
|
|
|
|
//
|
|
|
|
//-----------------------------------------------------------------------------
|
|
|
|
|
|
|
|
|
|
|
|
#include <stdlib.h>
|
|
|
|
#include <math.h>
|
|
|
|
|
|
|
|
#include "m_cheat.h"
|
|
|
|
#include "d_player.h"
|
|
|
|
#include "doomstat.h"
|
|
|
|
#include "gstrings.h"
|
|
|
|
#include "p_local.h"
|
|
|
|
#include "a_doomglobal.h"
|
|
|
|
#include "a_strifeglobal.h"
|
|
|
|
#include "gi.h"
|
|
|
|
#include "p_enemy.h"
|
|
|
|
#include "sbar.h"
|
|
|
|
#include "c_dispatch.h"
|
|
|
|
#include "v_video.h"
|
|
|
|
#include "w_wad.h"
|
|
|
|
#include "a_keys.h"
|
|
|
|
#include "templates.h"
|
|
|
|
#include "p_lnspec.h"
|
2007-12-23 14:23:52 +00:00
|
|
|
#include "c_console.h"
|
2007-12-26 16:06:03 +00:00
|
|
|
#include "r_translate.h"
|
2006-02-24 04:48:15 +00:00
|
|
|
|
|
|
|
// [RH] Actually handle the cheat. The cheat code in st_stuff.c now just
|
|
|
|
// writes some bytes to the network data stream, and the network code
|
|
|
|
// later calls us.
|
|
|
|
|
|
|
|
void cht_DoCheat (player_t *player, int cheat)
|
|
|
|
{
|
2006-05-10 02:40:43 +00:00
|
|
|
static const PClass *BeholdPowers[9] =
|
2006-02-24 04:48:15 +00:00
|
|
|
{
|
|
|
|
RUNTIME_CLASS(APowerInvulnerable),
|
|
|
|
RUNTIME_CLASS(APowerStrength),
|
|
|
|
RUNTIME_CLASS(APowerInvisibility),
|
|
|
|
RUNTIME_CLASS(APowerIronFeet),
|
|
|
|
NULL, // MapRevealer
|
|
|
|
RUNTIME_CLASS(APowerLightAmp),
|
|
|
|
RUNTIME_CLASS(APowerShadow),
|
|
|
|
RUNTIME_CLASS(APowerMask),
|
|
|
|
RUNTIME_CLASS(APowerTargeter)
|
|
|
|
};
|
2006-05-10 02:40:43 +00:00
|
|
|
const PClass *type;
|
2006-02-24 04:48:15 +00:00
|
|
|
AInventory *item;
|
|
|
|
const char *msg = "";
|
|
|
|
char msgbuild[32];
|
|
|
|
int i;
|
|
|
|
|
|
|
|
switch (cheat)
|
|
|
|
{
|
|
|
|
case CHT_IDDQD:
|
2006-06-22 20:52:49 +00:00
|
|
|
if (!(player->cheats & CF_GODMODE) && player->playerstate == PST_LIVE)
|
2006-02-24 04:48:15 +00:00
|
|
|
{
|
|
|
|
if (player->mo)
|
|
|
|
player->mo->health = deh.GodHealth;
|
|
|
|
|
|
|
|
player->health = deh.GodHealth;
|
|
|
|
}
|
|
|
|
// fall through to CHT_GOD
|
|
|
|
case CHT_GOD:
|
|
|
|
player->cheats ^= CF_GODMODE;
|
|
|
|
if (player->cheats & CF_GODMODE)
|
|
|
|
msg = GStrings("STSTR_DQDON");
|
|
|
|
else
|
|
|
|
msg = GStrings("STSTR_DQDOFF");
|
|
|
|
SB_state = screen->GetPageCount ();
|
|
|
|
break;
|
|
|
|
|
|
|
|
case CHT_NOCLIP:
|
|
|
|
player->cheats ^= CF_NOCLIP;
|
|
|
|
if (player->cheats & CF_NOCLIP)
|
|
|
|
msg = GStrings("STSTR_NCON");
|
|
|
|
else
|
|
|
|
msg = GStrings("STSTR_NCOFF");
|
|
|
|
break;
|
|
|
|
|
|
|
|
case CHT_NOMOMENTUM:
|
|
|
|
player->cheats ^= CF_NOMOMENTUM;
|
|
|
|
if (player->cheats & CF_NOMOMENTUM)
|
|
|
|
msg = "LEAD BOOTS ON";
|
|
|
|
else
|
|
|
|
msg = "LEAD BOOTS OFF";
|
|
|
|
break;
|
|
|
|
|
|
|
|
case CHT_FLY:
|
|
|
|
if (player->mo != NULL)
|
|
|
|
{
|
|
|
|
player->cheats ^= CF_FLY;
|
|
|
|
if (player->cheats & CF_FLY)
|
|
|
|
{
|
|
|
|
player->mo->flags |= MF_NOGRAVITY;
|
|
|
|
player->mo->flags2 |= MF2_FLY;
|
|
|
|
msg = "You feel lighter";
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
player->mo->flags &= ~MF_NOGRAVITY;
|
|
|
|
player->mo->flags2 &= ~MF2_FLY;
|
|
|
|
msg = "Gravity weighs you down";
|
|
|
|
}
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
|
|
|
|
case CHT_MORPH:
|
2006-10-27 03:03:34 +00:00
|
|
|
msg = cht_Morph (player, PClass::FindClass (gameinfo.gametype == GAME_Heretic ? NAME_ChickenPlayer : NAME_PigPlayer), true);
|
2006-02-24 04:48:15 +00:00
|
|
|
break;
|
|
|
|
|
|
|
|
case CHT_NOTARGET:
|
|
|
|
player->cheats ^= CF_NOTARGET;
|
|
|
|
if (player->cheats & CF_NOTARGET)
|
|
|
|
msg = "notarget ON";
|
|
|
|
else
|
|
|
|
msg = "notarget OFF";
|
|
|
|
break;
|
|
|
|
|
|
|
|
case CHT_ANUBIS:
|
|
|
|
player->cheats ^= CF_FRIGHTENING;
|
|
|
|
if (player->cheats & CF_FRIGHTENING)
|
|
|
|
msg = "\"Quake with fear!\"";
|
|
|
|
else
|
|
|
|
msg = "No more ogre armor";
|
|
|
|
break;
|
|
|
|
|
|
|
|
case CHT_CHASECAM:
|
|
|
|
player->cheats ^= CF_CHASECAM;
|
|
|
|
if (player->cheats & CF_CHASECAM)
|
|
|
|
msg = "chasecam ON";
|
|
|
|
else
|
|
|
|
msg = "chasecam OFF";
|
|
|
|
R_ResetViewInterpolation ();
|
|
|
|
break;
|
|
|
|
|
|
|
|
case CHT_CHAINSAW:
|
2006-06-22 20:52:49 +00:00
|
|
|
if (player->mo != NULL && player->health >= 0)
|
2006-02-24 04:48:15 +00:00
|
|
|
{
|
2006-05-10 02:40:43 +00:00
|
|
|
type = PClass::FindClass ("Chainsaw");
|
- Fixed: ActorFlagSetOrReset() wasn't receiving the + or - character from
ParseActorProperties().
- Fixed: The decorate FindFlag() function returned flags from ActorFlags
instead of the passed flags set.
- Fixed: The CHT_CHAINSAW, CHT_POWER, CHT_HEALTH, and CHT_RESSURECT needed
NULL player->mo checks.
- Fixed: The "give all" command didn't give the backpack in Doom, and it
must give the backpack before giving ammo.
- Fixed: P_SetPsprite() must not call the action function if the player is
not attached to an actor. This can happen, for instance, if the level is
destroyed while the player is holding a powered-up Phoenix Rod. As part
of its EndPowerup() function, it sets the psprite to the regular version,
but the player actor has already been destroyed.
- Fixed: FinishThingdef() needs to check for valid names, because weapons
could have inherited valid pointers from their superclass.
- Fixed: fuglyname didn't work.
- Fixed: Redefining $ambient sounds leaked memory.
- Added Jim's crashcatcher.c fix for better shell support.
- VC7.1 seems to have no trouble distinguishing between passing a (const
TypeInfo *) reference to operator<< and the generic, templated (object *)
version, so a few places that can benefit from it now use it. I believe
VC6 had problems with this, which is why I didn't do it all along. The
function's implementation was also moved out of dobject.cpp and into
farchive.cpp.
- Fixed: UnpackPixels() unpacked all chunks in a byte, which is wrong for the
last byte in a row if the image width is not an even multiple of the number
pixels per byte.
- Fixed: P_TranslateLineDef() should only clear monster activation for secret
useable lines, not crossable lines.
- Fixed: Some leftover P_IsHostile() calls still needed to be rewritten.
- Fixed: AWeaponHolder::Serialize() wrote the class type in all circumstances.
SVN r20 (trunk)
2006-03-14 06:11:39 +00:00
|
|
|
if (player->mo->FindInventory (type) == NULL)
|
|
|
|
{
|
|
|
|
player->mo->GiveInventoryType (type);
|
|
|
|
}
|
|
|
|
msg = GStrings("STSTR_CHOPPERS");
|
2006-02-24 04:48:15 +00:00
|
|
|
}
|
|
|
|
// [RH] The original cheat also set powers[pw_invulnerability] to true.
|
|
|
|
// Since this is a timer and not a boolean, it effectively turned off
|
|
|
|
// the invulnerability powerup, although it looks like it was meant to
|
|
|
|
// turn it on.
|
|
|
|
break;
|
|
|
|
|
|
|
|
case CHT_POWER:
|
2006-06-22 20:52:49 +00:00
|
|
|
if (player->mo != NULL && player->health >= 0)
|
2006-02-24 04:48:15 +00:00
|
|
|
{
|
- Fixed: ActorFlagSetOrReset() wasn't receiving the + or - character from
ParseActorProperties().
- Fixed: The decorate FindFlag() function returned flags from ActorFlags
instead of the passed flags set.
- Fixed: The CHT_CHAINSAW, CHT_POWER, CHT_HEALTH, and CHT_RESSURECT needed
NULL player->mo checks.
- Fixed: The "give all" command didn't give the backpack in Doom, and it
must give the backpack before giving ammo.
- Fixed: P_SetPsprite() must not call the action function if the player is
not attached to an actor. This can happen, for instance, if the level is
destroyed while the player is holding a powered-up Phoenix Rod. As part
of its EndPowerup() function, it sets the psprite to the regular version,
but the player actor has already been destroyed.
- Fixed: FinishThingdef() needs to check for valid names, because weapons
could have inherited valid pointers from their superclass.
- Fixed: fuglyname didn't work.
- Fixed: Redefining $ambient sounds leaked memory.
- Added Jim's crashcatcher.c fix for better shell support.
- VC7.1 seems to have no trouble distinguishing between passing a (const
TypeInfo *) reference to operator<< and the generic, templated (object *)
version, so a few places that can benefit from it now use it. I believe
VC6 had problems with this, which is why I didn't do it all along. The
function's implementation was also moved out of dobject.cpp and into
farchive.cpp.
- Fixed: UnpackPixels() unpacked all chunks in a byte, which is wrong for the
last byte in a row if the image width is not an even multiple of the number
pixels per byte.
- Fixed: P_TranslateLineDef() should only clear monster activation for secret
useable lines, not crossable lines.
- Fixed: Some leftover P_IsHostile() calls still needed to be rewritten.
- Fixed: AWeaponHolder::Serialize() wrote the class type in all circumstances.
SVN r20 (trunk)
2006-03-14 06:11:39 +00:00
|
|
|
item = player->mo->FindInventory (RUNTIME_CLASS(APowerWeaponLevel2));
|
|
|
|
if (item != NULL)
|
|
|
|
{
|
|
|
|
item->Destroy ();
|
|
|
|
msg = GStrings("TXT_CHEATPOWEROFF");
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
player->mo->GiveInventoryType (RUNTIME_CLASS(APowerWeaponLevel2));
|
|
|
|
msg = GStrings("TXT_CHEATPOWERON");
|
|
|
|
}
|
2006-02-24 04:48:15 +00:00
|
|
|
}
|
|
|
|
break;
|
|
|
|
|
|
|
|
case CHT_IDKFA:
|
|
|
|
cht_Give (player, "backpack");
|
|
|
|
cht_Give (player, "weapons");
|
|
|
|
cht_Give (player, "ammo");
|
|
|
|
cht_Give (player, "keys");
|
|
|
|
cht_Give (player, "armor");
|
|
|
|
msg = GStrings("STSTR_KFAADDED");
|
|
|
|
break;
|
|
|
|
|
|
|
|
case CHT_IDFA:
|
|
|
|
cht_Give (player, "backpack");
|
|
|
|
cht_Give (player, "weapons");
|
|
|
|
cht_Give (player, "ammo");
|
|
|
|
cht_Give (player, "armor");
|
|
|
|
msg = GStrings("STSTR_FAADDED");
|
|
|
|
break;
|
|
|
|
|
|
|
|
case CHT_BEHOLDV:
|
|
|
|
case CHT_BEHOLDS:
|
|
|
|
case CHT_BEHOLDI:
|
|
|
|
case CHT_BEHOLDR:
|
|
|
|
case CHT_BEHOLDA:
|
|
|
|
case CHT_BEHOLDL:
|
|
|
|
case CHT_PUMPUPI:
|
|
|
|
case CHT_PUMPUPM:
|
|
|
|
case CHT_PUMPUPT:
|
|
|
|
i = cheat - CHT_BEHOLDV;
|
|
|
|
|
|
|
|
if (i == 4)
|
|
|
|
{
|
|
|
|
level.flags ^= LEVEL_ALLMAP;
|
|
|
|
}
|
2006-06-22 20:52:49 +00:00
|
|
|
else if (player->mo != NULL && player->health >= 0)
|
2006-02-24 04:48:15 +00:00
|
|
|
{
|
|
|
|
item = player->mo->FindInventory (BeholdPowers[i]);
|
|
|
|
if (item == NULL)
|
|
|
|
{
|
|
|
|
player->mo->GiveInventoryType (BeholdPowers[i]);
|
|
|
|
if (cheat == CHT_BEHOLDS)
|
|
|
|
{
|
|
|
|
P_GiveBody (player->mo, -100);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
item->Destroy ();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
msg = GStrings("STSTR_BEHOLDX");
|
|
|
|
break;
|
|
|
|
|
|
|
|
case CHT_MASSACRE:
|
|
|
|
{
|
|
|
|
int killcount = P_Massacre ();
|
|
|
|
// killough 3/22/98: make more intelligent about plural
|
|
|
|
// Ty 03/27/98 - string(s) *not* externalized
|
About a week's worth of changes here. As a heads-up, I wouldn't be
surprised if this doesn't build in Linux right now. The CMakeLists.txt
were checked with MinGW and NMake, but how they fair under Linux is an
unknown to me at this time.
- Converted most sprintf (and all wsprintf) calls to either mysnprintf or
FStrings, depending on the situation.
- Changed the strings in the wbstartstruct to be FStrings.
- Changed myvsnprintf() to output nothing if count is greater than INT_MAX.
This is so that I can use a series of mysnprintf() calls and advance the
pointer for each one. Once the pointer goes beyond the end of the buffer,
the count will go negative, but since it's an unsigned type it will be
seen as excessively huge instead. This should not be a problem, as there's
no reason for ZDoom to be using text buffers larger than 2 GB anywhere.
- Ripped out the disabled bit from FGameConfigFile::MigrateOldConfig().
- Changed CalcMapName() to return an FString instead of a pointer to a static
buffer.
- Changed startmap in d_main.cpp into an FString.
- Changed CheckWarpTransMap() to take an FString& as the first argument.
- Changed d_mapname in g_level.cpp into an FString.
- Changed DoSubstitution() in ct_chat.cpp to place the substitutions in an
FString.
- Fixed: The MAPINFO parser wrote into the string buffer to construct a map
name when given a Hexen map number. This was fine with the old scanner
code, but only a happy coincidence prevents it from crashing with the new
code
- Added the 'B' conversion specifier to StringFormat::VWorker() for printing
binary numbers.
- Added CMake support for building with MinGW, MSYS, and NMake. Linux support
is probably broken until I get around to booting into Linux again. Niceties
provided over the existing Makefiles they're replacing:
* All command-line builds can use the same build system, rather than having
a separate one for MinGW and another for Linux.
* Microsoft's NMake tool is supported as a target.
* Progress meters.
* Parallel makes work from a fresh checkout without needing to be primed
first with a single-threaded make.
* Porting to other architectures should be simplified, whenever that day
comes.
- Replaced the makewad tool with zipdir. This handles the dependency tracking
itself instead of generating an external makefile to do it, since I couldn't
figure out how to generate a makefile with an external tool and include it
with a CMake-generated makefile. Where makewad used a master list of files
to generate the package file, zipdir just zips the entire contents of one or
more directories.
- Added the gdtoa package from netlib's fp library so that ZDoom's printf-style
formatting can be entirely independant of the CRT.
SVN r1082 (trunk)
2008-07-23 04:57:26 +00:00
|
|
|
mysnprintf (msgbuild, countof(msgbuild), "%d Monster%s Killed", killcount, killcount==1 ? "" : "s");
|
2006-02-24 04:48:15 +00:00
|
|
|
msg = msgbuild;
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
|
|
|
|
case CHT_HEALTH:
|
2006-06-22 20:52:49 +00:00
|
|
|
if (player->mo != NULL && player->playerstate == PST_LIVE)
|
- Fixed: ActorFlagSetOrReset() wasn't receiving the + or - character from
ParseActorProperties().
- Fixed: The decorate FindFlag() function returned flags from ActorFlags
instead of the passed flags set.
- Fixed: The CHT_CHAINSAW, CHT_POWER, CHT_HEALTH, and CHT_RESSURECT needed
NULL player->mo checks.
- Fixed: The "give all" command didn't give the backpack in Doom, and it
must give the backpack before giving ammo.
- Fixed: P_SetPsprite() must not call the action function if the player is
not attached to an actor. This can happen, for instance, if the level is
destroyed while the player is holding a powered-up Phoenix Rod. As part
of its EndPowerup() function, it sets the psprite to the regular version,
but the player actor has already been destroyed.
- Fixed: FinishThingdef() needs to check for valid names, because weapons
could have inherited valid pointers from their superclass.
- Fixed: fuglyname didn't work.
- Fixed: Redefining $ambient sounds leaked memory.
- Added Jim's crashcatcher.c fix for better shell support.
- VC7.1 seems to have no trouble distinguishing between passing a (const
TypeInfo *) reference to operator<< and the generic, templated (object *)
version, so a few places that can benefit from it now use it. I believe
VC6 had problems with this, which is why I didn't do it all along. The
function's implementation was also moved out of dobject.cpp and into
farchive.cpp.
- Fixed: UnpackPixels() unpacked all chunks in a byte, which is wrong for the
last byte in a row if the image width is not an even multiple of the number
pixels per byte.
- Fixed: P_TranslateLineDef() should only clear monster activation for secret
useable lines, not crossable lines.
- Fixed: Some leftover P_IsHostile() calls still needed to be rewritten.
- Fixed: AWeaponHolder::Serialize() wrote the class type in all circumstances.
SVN r20 (trunk)
2006-03-14 06:11:39 +00:00
|
|
|
{
|
|
|
|
player->health = player->mo->health = player->mo->GetDefault()->health;
|
|
|
|
msg = GStrings("TXT_CHEATHEALTH");
|
|
|
|
}
|
2006-02-24 04:48:15 +00:00
|
|
|
break;
|
|
|
|
|
|
|
|
case CHT_KEYS:
|
|
|
|
cht_Give (player, "keys");
|
|
|
|
msg = GStrings("TXT_CHEATKEYS");
|
|
|
|
break;
|
|
|
|
|
|
|
|
// [GRB]
|
|
|
|
case CHT_RESSURECT:
|
- Fixed: ActorFlagSetOrReset() wasn't receiving the + or - character from
ParseActorProperties().
- Fixed: The decorate FindFlag() function returned flags from ActorFlags
instead of the passed flags set.
- Fixed: The CHT_CHAINSAW, CHT_POWER, CHT_HEALTH, and CHT_RESSURECT needed
NULL player->mo checks.
- Fixed: The "give all" command didn't give the backpack in Doom, and it
must give the backpack before giving ammo.
- Fixed: P_SetPsprite() must not call the action function if the player is
not attached to an actor. This can happen, for instance, if the level is
destroyed while the player is holding a powered-up Phoenix Rod. As part
of its EndPowerup() function, it sets the psprite to the regular version,
but the player actor has already been destroyed.
- Fixed: FinishThingdef() needs to check for valid names, because weapons
could have inherited valid pointers from their superclass.
- Fixed: fuglyname didn't work.
- Fixed: Redefining $ambient sounds leaked memory.
- Added Jim's crashcatcher.c fix for better shell support.
- VC7.1 seems to have no trouble distinguishing between passing a (const
TypeInfo *) reference to operator<< and the generic, templated (object *)
version, so a few places that can benefit from it now use it. I believe
VC6 had problems with this, which is why I didn't do it all along. The
function's implementation was also moved out of dobject.cpp and into
farchive.cpp.
- Fixed: UnpackPixels() unpacked all chunks in a byte, which is wrong for the
last byte in a row if the image width is not an even multiple of the number
pixels per byte.
- Fixed: P_TranslateLineDef() should only clear monster activation for secret
useable lines, not crossable lines.
- Fixed: Some leftover P_IsHostile() calls still needed to be rewritten.
- Fixed: AWeaponHolder::Serialize() wrote the class type in all circumstances.
SVN r20 (trunk)
2006-03-14 06:11:39 +00:00
|
|
|
if (player->playerstate != PST_LIVE && player->mo != NULL)
|
2006-02-24 04:48:15 +00:00
|
|
|
{
|
2006-04-16 13:29:50 +00:00
|
|
|
if (player->mo->IsKindOf(RUNTIME_CLASS(APlayerChunk)))
|
2006-04-13 22:40:43 +00:00
|
|
|
{
|
2006-04-16 13:29:50 +00:00
|
|
|
Printf("Unable to resurrect. Player is no longer connected to its body.\n");
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
player->playerstate = PST_LIVE;
|
|
|
|
player->health = player->mo->health = player->mo->GetDefault()->health;
|
2006-07-13 10:17:56 +00:00
|
|
|
player->viewheight = ((APlayerPawn *)player->mo->GetDefault())->ViewHeight;
|
2006-04-16 13:29:50 +00:00
|
|
|
player->mo->flags = player->mo->GetDefault()->flags;
|
2007-05-10 22:22:38 +00:00
|
|
|
player->mo->flags2 = player->mo->GetDefault()->flags2;
|
|
|
|
player->mo->flags3 = player->mo->GetDefault()->flags3;
|
|
|
|
player->mo->flags4 = player->mo->GetDefault()->flags4;
|
|
|
|
player->mo->flags5 = player->mo->GetDefault()->flags5;
|
2006-04-16 13:29:50 +00:00
|
|
|
player->mo->height = player->mo->GetDefault()->height;
|
2007-05-26 10:50:32 +00:00
|
|
|
player->mo->special1 = 0; // required for the Hexen fighter's fist attack.
|
|
|
|
// This gets set by AActor::Die as flag for the wimpy death and must be reset here.
|
2006-04-16 13:29:50 +00:00
|
|
|
player->mo->SetState (player->mo->SpawnState);
|
|
|
|
player->mo->Translation = TRANSLATION(TRANSLATION_Players, BYTE(player-players));
|
2006-10-31 14:53:21 +00:00
|
|
|
player->mo->DamageType = NAME_None;
|
2006-06-18 04:10:47 +00:00
|
|
|
// player->mo->GiveDefaultInventory();
|
2006-07-31 10:22:53 +00:00
|
|
|
if (player->ReadyWeapon != NULL)
|
|
|
|
{
|
2006-10-31 14:53:21 +00:00
|
|
|
P_SetPsprite(player, ps_weapon, player->ReadyWeapon->GetUpState());
|
2006-07-31 10:22:53 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
if (player->morphTics > 0)
|
|
|
|
{
|
2008-05-22 19:35:38 +00:00
|
|
|
P_UndoPlayerMorph(player, player);
|
2006-07-31 10:22:53 +00:00
|
|
|
}
|
|
|
|
|
2006-04-13 22:40:43 +00:00
|
|
|
}
|
2006-02-24 04:48:15 +00:00
|
|
|
}
|
|
|
|
break;
|
|
|
|
|
|
|
|
case CHT_TAKEWEAPS:
|
2008-01-07 08:58:18 +00:00
|
|
|
if (player->morphTics || player->mo == NULL || player->mo->health <= 0)
|
2006-02-24 04:48:15 +00:00
|
|
|
{
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
// Take away all weapons that are either non-wimpy or use ammo.
|
|
|
|
for (item = player->mo->Inventory; item != NULL; )
|
|
|
|
{
|
|
|
|
AInventory *next = item->Inventory;
|
|
|
|
if (item->IsKindOf (RUNTIME_CLASS(AWeapon)))
|
|
|
|
{
|
|
|
|
AWeapon *weap = static_cast<AWeapon *> (item);
|
|
|
|
if (!(weap->WeaponFlags & WIF_WIMPY_WEAPON) ||
|
|
|
|
weap->AmmoType1 != NULL)
|
|
|
|
{
|
|
|
|
item->Destroy ();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
item = next;
|
|
|
|
}
|
|
|
|
msg = GStrings("TXT_CHEATIDKFA");
|
|
|
|
break;
|
|
|
|
|
|
|
|
case CHT_NOWUDIE:
|
|
|
|
cht_Suicide (player);
|
|
|
|
msg = GStrings("TXT_CHEATIDDQD");
|
|
|
|
break;
|
|
|
|
|
|
|
|
case CHT_ALLARTI:
|
2006-04-11 16:27:41 +00:00
|
|
|
for (int i=0;i<25;i++)
|
|
|
|
{
|
|
|
|
cht_Give (player, "artifacts");
|
|
|
|
}
|
2006-02-24 04:48:15 +00:00
|
|
|
msg = GStrings("TXT_CHEATARTIFACTS3");
|
|
|
|
break;
|
|
|
|
|
|
|
|
case CHT_PUZZLE:
|
|
|
|
cht_Give (player, "puzzlepieces");
|
|
|
|
msg = GStrings("TXT_CHEATARTIFACTS3");
|
|
|
|
break;
|
|
|
|
|
|
|
|
case CHT_MDK:
|
|
|
|
if (player->mo == NULL)
|
|
|
|
{
|
- Fixed: ActorFlagSetOrReset() wasn't receiving the + or - character from
ParseActorProperties().
- Fixed: The decorate FindFlag() function returned flags from ActorFlags
instead of the passed flags set.
- Fixed: The CHT_CHAINSAW, CHT_POWER, CHT_HEALTH, and CHT_RESSURECT needed
NULL player->mo checks.
- Fixed: The "give all" command didn't give the backpack in Doom, and it
must give the backpack before giving ammo.
- Fixed: P_SetPsprite() must not call the action function if the player is
not attached to an actor. This can happen, for instance, if the level is
destroyed while the player is holding a powered-up Phoenix Rod. As part
of its EndPowerup() function, it sets the psprite to the regular version,
but the player actor has already been destroyed.
- Fixed: FinishThingdef() needs to check for valid names, because weapons
could have inherited valid pointers from their superclass.
- Fixed: fuglyname didn't work.
- Fixed: Redefining $ambient sounds leaked memory.
- Added Jim's crashcatcher.c fix for better shell support.
- VC7.1 seems to have no trouble distinguishing between passing a (const
TypeInfo *) reference to operator<< and the generic, templated (object *)
version, so a few places that can benefit from it now use it. I believe
VC6 had problems with this, which is why I didn't do it all along. The
function's implementation was also moved out of dobject.cpp and into
farchive.cpp.
- Fixed: UnpackPixels() unpacked all chunks in a byte, which is wrong for the
last byte in a row if the image width is not an even multiple of the number
pixels per byte.
- Fixed: P_TranslateLineDef() should only clear monster activation for secret
useable lines, not crossable lines.
- Fixed: Some leftover P_IsHostile() calls still needed to be rewritten.
- Fixed: AWeaponHolder::Serialize() wrote the class type in all circumstances.
SVN r20 (trunk)
2006-03-14 06:11:39 +00:00
|
|
|
Printf ("What do you want to kill outside of a game?\n");
|
2006-02-24 04:48:15 +00:00
|
|
|
}
|
|
|
|
else if (!deathmatch)
|
|
|
|
{
|
|
|
|
// Don't allow this in deathmatch even with cheats enabled, because it's
|
|
|
|
// a very very cheap kill.
|
|
|
|
P_LineAttack (player->mo, player->mo->angle, PLAYERMISSILERANGE,
|
|
|
|
P_AimLineAttack (player->mo, player->mo->angle, PLAYERMISSILERANGE), 1000000,
|
2006-11-04 13:06:42 +00:00
|
|
|
NAME_None, NAME_BulletPuff);
|
2006-02-24 04:48:15 +00:00
|
|
|
}
|
|
|
|
break;
|
|
|
|
|
|
|
|
case CHT_DONNYTRUMP:
|
|
|
|
cht_Give (player, "HealthTraining");
|
|
|
|
msg = "YOU GOT THE MIDAS TOUCH, BABY";
|
|
|
|
break;
|
|
|
|
|
|
|
|
case CHT_LEGO:
|
2006-06-22 20:52:49 +00:00
|
|
|
if (player->mo != NULL && player->health >= 0)
|
2006-02-24 04:48:15 +00:00
|
|
|
{
|
|
|
|
int oldpieces = ASigil::GiveSigilPiece (player->mo);
|
|
|
|
item = player->mo->FindInventory (RUNTIME_CLASS(ASigil));
|
|
|
|
|
|
|
|
if (item != NULL)
|
|
|
|
{
|
|
|
|
if (oldpieces == 5)
|
|
|
|
{
|
|
|
|
item->Destroy ();
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
player->PendingWeapon = static_cast<AWeapon *> (item);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
|
|
|
|
case CHT_PUMPUPH:
|
|
|
|
cht_Give (player, "MedPatch");
|
|
|
|
cht_Give (player, "MedicalKit");
|
|
|
|
cht_Give (player, "SurgeryKit");
|
|
|
|
msg = "you got the stuff!";
|
|
|
|
break;
|
|
|
|
|
|
|
|
case CHT_PUMPUPP:
|
|
|
|
cht_Give (player, "AmmoSatchel");
|
|
|
|
msg = "you got the stuff!";
|
|
|
|
break;
|
|
|
|
|
|
|
|
case CHT_PUMPUPS:
|
2006-06-24 08:03:15 +00:00
|
|
|
cht_Give (player, "UpgradeStamina", 10);
|
2006-02-24 04:48:15 +00:00
|
|
|
cht_Give (player, "UpgradeAccuracy");
|
|
|
|
msg = "you got the stuff!";
|
|
|
|
break;
|
2006-10-24 02:32:12 +00:00
|
|
|
|
|
|
|
case CHT_CLEARFROZENPROPS:
|
|
|
|
player->cheats &= ~(CF_FROZEN|CF_TOTALLYFROZEN);
|
|
|
|
msg = "Frozen player properties turned off";
|
|
|
|
break;
|
2007-10-29 20:27:40 +00:00
|
|
|
|
|
|
|
case CHT_FREEZE:
|
2008-03-28 00:38:17 +00:00
|
|
|
bglobal.changefreeze ^= 1;
|
|
|
|
if (bglobal.freeze ^ bglobal.changefreeze)
|
2007-10-29 20:27:40 +00:00
|
|
|
{
|
2007-12-06 23:17:38 +00:00
|
|
|
msg = "Freeze mode on";
|
2007-10-29 20:27:40 +00:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2007-12-06 23:17:38 +00:00
|
|
|
msg = "Freeze mode off";
|
2007-10-29 20:27:40 +00:00
|
|
|
}
|
|
|
|
break;
|
2006-02-24 04:48:15 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
if (!*msg) // [SO] Don't print blank lines!
|
|
|
|
return;
|
|
|
|
|
|
|
|
if (player == &players[consoleplayer])
|
|
|
|
Printf ("%s\n", msg);
|
|
|
|
else
|
|
|
|
Printf ("%s is a cheater: %s\n", player->userinfo.netname, msg);
|
|
|
|
}
|
|
|
|
|
2006-10-27 03:03:34 +00:00
|
|
|
const char *cht_Morph (player_t *player, const PClass *morphclass, bool quickundo)
|
|
|
|
{
|
|
|
|
if (player->mo == NULL)
|
|
|
|
{
|
|
|
|
return "";
|
|
|
|
}
|
|
|
|
PClass *oldclass = player->mo->GetClass();
|
2008-04-08 09:52:50 +00:00
|
|
|
|
|
|
|
// Set the standard morph style for the current game
|
|
|
|
int style = MORPH_UNDOBYTOMEOFPOWER;
|
|
|
|
if (gameinfo.gametype == GAME_Hexen) style |= MORPH_UNDOBYCHAOSDEVICE;
|
|
|
|
|
2006-10-27 03:03:34 +00:00
|
|
|
if (player->morphTics)
|
|
|
|
{
|
2008-05-22 19:35:38 +00:00
|
|
|
if (P_UndoPlayerMorph (player, player))
|
2006-10-27 03:03:34 +00:00
|
|
|
{
|
2008-04-12 15:31:18 +00:00
|
|
|
if (!quickundo && oldclass != morphclass && P_MorphPlayer (player, player, morphclass, 0, style))
|
2006-10-27 03:03:34 +00:00
|
|
|
{
|
|
|
|
return "You feel even stranger.";
|
|
|
|
}
|
|
|
|
return "You feel like yourself again.";
|
|
|
|
}
|
|
|
|
}
|
2008-04-12 15:31:18 +00:00
|
|
|
else if (P_MorphPlayer (player, player, morphclass, 0, style))
|
2006-10-27 03:03:34 +00:00
|
|
|
{
|
|
|
|
return "You feel strange...";
|
|
|
|
}
|
|
|
|
return "";
|
|
|
|
}
|
|
|
|
|
2006-05-10 02:40:43 +00:00
|
|
|
void GiveSpawner (player_t *player, const PClass *type, int amount)
|
2006-02-24 04:48:15 +00:00
|
|
|
{
|
2006-06-22 20:52:49 +00:00
|
|
|
if (player->mo == NULL || player->health <= 0)
|
|
|
|
{
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2006-02-24 04:48:15 +00:00
|
|
|
AInventory *item = static_cast<AInventory *>
|
2006-07-16 09:10:45 +00:00
|
|
|
(Spawn (type, player->mo->x, player->mo->y, player->mo->z, NO_REPLACE));
|
2006-02-24 04:48:15 +00:00
|
|
|
if (item != NULL)
|
|
|
|
{
|
|
|
|
if (amount > 0)
|
|
|
|
{
|
|
|
|
if (type->IsDescendantOf (RUNTIME_CLASS(ABasicArmorPickup)))
|
|
|
|
{
|
|
|
|
if (static_cast<ABasicArmorPickup*>(item)->SaveAmount != 0)
|
|
|
|
{
|
|
|
|
static_cast<ABasicArmorPickup*>(item)->SaveAmount *= amount;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
static_cast<ABasicArmorPickup*>(item)->SaveAmount *= amount;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else if (type->IsDescendantOf (RUNTIME_CLASS(ABasicArmorBonus)))
|
|
|
|
{
|
|
|
|
static_cast<ABasicArmorBonus*>(item)->SaveAmount *= amount;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
item->Amount = MIN (amount, item->MaxAmount);
|
|
|
|
}
|
|
|
|
}
|
2008-07-05 10:17:10 +00:00
|
|
|
if(item->flags & MF_COUNTITEM) // Given items shouldn't count against the map's total,
|
|
|
|
{ // since they aren't added to the player's total.
|
|
|
|
level.total_items--;
|
|
|
|
item->flags &= ~MF_COUNTITEM;
|
|
|
|
}
|
2006-02-24 04:48:15 +00:00
|
|
|
if (!item->TryPickup (player->mo))
|
|
|
|
{
|
|
|
|
item->Destroy ();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2006-12-29 03:38:37 +00:00
|
|
|
void cht_Give (player_t *player, const char *name, int amount)
|
2006-02-24 04:48:15 +00:00
|
|
|
{
|
2006-09-14 00:02:31 +00:00
|
|
|
bool giveall;
|
2006-02-24 04:48:15 +00:00
|
|
|
int i;
|
2006-05-10 02:40:43 +00:00
|
|
|
const PClass *type;
|
2006-02-24 04:48:15 +00:00
|
|
|
|
|
|
|
if (player != &players[consoleplayer])
|
|
|
|
Printf ("%s is a cheater: give %s\n", player->userinfo.netname, name);
|
|
|
|
|
2006-04-13 22:40:43 +00:00
|
|
|
if (player->mo == NULL || player->health <= 0)
|
2006-02-24 04:48:15 +00:00
|
|
|
{
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
giveall = (stricmp (name, "all") == 0);
|
|
|
|
|
|
|
|
if (giveall || stricmp (name, "health") == 0)
|
|
|
|
{
|
|
|
|
if (amount > 0)
|
|
|
|
{
|
|
|
|
if (player->mo)
|
|
|
|
{
|
|
|
|
player->mo->health += amount;
|
|
|
|
player->health = player->mo->health;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
player->health += amount;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
if (player->mo)
|
|
|
|
player->mo->health = deh.GodHealth;
|
|
|
|
|
|
|
|
player->health = deh.GodHealth;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!giveall)
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
- Fixed: ActorFlagSetOrReset() wasn't receiving the + or - character from
ParseActorProperties().
- Fixed: The decorate FindFlag() function returned flags from ActorFlags
instead of the passed flags set.
- Fixed: The CHT_CHAINSAW, CHT_POWER, CHT_HEALTH, and CHT_RESSURECT needed
NULL player->mo checks.
- Fixed: The "give all" command didn't give the backpack in Doom, and it
must give the backpack before giving ammo.
- Fixed: P_SetPsprite() must not call the action function if the player is
not attached to an actor. This can happen, for instance, if the level is
destroyed while the player is holding a powered-up Phoenix Rod. As part
of its EndPowerup() function, it sets the psprite to the regular version,
but the player actor has already been destroyed.
- Fixed: FinishThingdef() needs to check for valid names, because weapons
could have inherited valid pointers from their superclass.
- Fixed: fuglyname didn't work.
- Fixed: Redefining $ambient sounds leaked memory.
- Added Jim's crashcatcher.c fix for better shell support.
- VC7.1 seems to have no trouble distinguishing between passing a (const
TypeInfo *) reference to operator<< and the generic, templated (object *)
version, so a few places that can benefit from it now use it. I believe
VC6 had problems with this, which is why I didn't do it all along. The
function's implementation was also moved out of dobject.cpp and into
farchive.cpp.
- Fixed: UnpackPixels() unpacked all chunks in a byte, which is wrong for the
last byte in a row if the image width is not an even multiple of the number
pixels per byte.
- Fixed: P_TranslateLineDef() should only clear monster activation for secret
useable lines, not crossable lines.
- Fixed: Some leftover P_IsHostile() calls still needed to be rewritten.
- Fixed: AWeaponHolder::Serialize() wrote the class type in all circumstances.
SVN r20 (trunk)
2006-03-14 06:11:39 +00:00
|
|
|
if (giveall || stricmp (name, "backpack") == 0)
|
|
|
|
{
|
|
|
|
// Select the correct type of backpack based on the game
|
|
|
|
if (gameinfo.gametype == GAME_Heretic)
|
|
|
|
{
|
2006-05-10 02:40:43 +00:00
|
|
|
type = PClass::FindClass ("BagOfHolding");
|
- Fixed: ActorFlagSetOrReset() wasn't receiving the + or - character from
ParseActorProperties().
- Fixed: The decorate FindFlag() function returned flags from ActorFlags
instead of the passed flags set.
- Fixed: The CHT_CHAINSAW, CHT_POWER, CHT_HEALTH, and CHT_RESSURECT needed
NULL player->mo checks.
- Fixed: The "give all" command didn't give the backpack in Doom, and it
must give the backpack before giving ammo.
- Fixed: P_SetPsprite() must not call the action function if the player is
not attached to an actor. This can happen, for instance, if the level is
destroyed while the player is holding a powered-up Phoenix Rod. As part
of its EndPowerup() function, it sets the psprite to the regular version,
but the player actor has already been destroyed.
- Fixed: FinishThingdef() needs to check for valid names, because weapons
could have inherited valid pointers from their superclass.
- Fixed: fuglyname didn't work.
- Fixed: Redefining $ambient sounds leaked memory.
- Added Jim's crashcatcher.c fix for better shell support.
- VC7.1 seems to have no trouble distinguishing between passing a (const
TypeInfo *) reference to operator<< and the generic, templated (object *)
version, so a few places that can benefit from it now use it. I believe
VC6 had problems with this, which is why I didn't do it all along. The
function's implementation was also moved out of dobject.cpp and into
farchive.cpp.
- Fixed: UnpackPixels() unpacked all chunks in a byte, which is wrong for the
last byte in a row if the image width is not an even multiple of the number
pixels per byte.
- Fixed: P_TranslateLineDef() should only clear monster activation for secret
useable lines, not crossable lines.
- Fixed: Some leftover P_IsHostile() calls still needed to be rewritten.
- Fixed: AWeaponHolder::Serialize() wrote the class type in all circumstances.
SVN r20 (trunk)
2006-03-14 06:11:39 +00:00
|
|
|
}
|
|
|
|
else if (gameinfo.gametype == GAME_Strife)
|
|
|
|
{
|
2006-05-10 02:40:43 +00:00
|
|
|
type = PClass::FindClass ("AmmoSatchel");
|
- Fixed: ActorFlagSetOrReset() wasn't receiving the + or - character from
ParseActorProperties().
- Fixed: The decorate FindFlag() function returned flags from ActorFlags
instead of the passed flags set.
- Fixed: The CHT_CHAINSAW, CHT_POWER, CHT_HEALTH, and CHT_RESSURECT needed
NULL player->mo checks.
- Fixed: The "give all" command didn't give the backpack in Doom, and it
must give the backpack before giving ammo.
- Fixed: P_SetPsprite() must not call the action function if the player is
not attached to an actor. This can happen, for instance, if the level is
destroyed while the player is holding a powered-up Phoenix Rod. As part
of its EndPowerup() function, it sets the psprite to the regular version,
but the player actor has already been destroyed.
- Fixed: FinishThingdef() needs to check for valid names, because weapons
could have inherited valid pointers from their superclass.
- Fixed: fuglyname didn't work.
- Fixed: Redefining $ambient sounds leaked memory.
- Added Jim's crashcatcher.c fix for better shell support.
- VC7.1 seems to have no trouble distinguishing between passing a (const
TypeInfo *) reference to operator<< and the generic, templated (object *)
version, so a few places that can benefit from it now use it. I believe
VC6 had problems with this, which is why I didn't do it all along. The
function's implementation was also moved out of dobject.cpp and into
farchive.cpp.
- Fixed: UnpackPixels() unpacked all chunks in a byte, which is wrong for the
last byte in a row if the image width is not an even multiple of the number
pixels per byte.
- Fixed: P_TranslateLineDef() should only clear monster activation for secret
useable lines, not crossable lines.
- Fixed: Some leftover P_IsHostile() calls still needed to be rewritten.
- Fixed: AWeaponHolder::Serialize() wrote the class type in all circumstances.
SVN r20 (trunk)
2006-03-14 06:11:39 +00:00
|
|
|
}
|
|
|
|
else if (gameinfo.gametype == GAME_Doom)
|
|
|
|
{
|
2006-05-10 02:40:43 +00:00
|
|
|
type = PClass::FindClass ("Backpack");
|
- Fixed: ActorFlagSetOrReset() wasn't receiving the + or - character from
ParseActorProperties().
- Fixed: The decorate FindFlag() function returned flags from ActorFlags
instead of the passed flags set.
- Fixed: The CHT_CHAINSAW, CHT_POWER, CHT_HEALTH, and CHT_RESSURECT needed
NULL player->mo checks.
- Fixed: The "give all" command didn't give the backpack in Doom, and it
must give the backpack before giving ammo.
- Fixed: P_SetPsprite() must not call the action function if the player is
not attached to an actor. This can happen, for instance, if the level is
destroyed while the player is holding a powered-up Phoenix Rod. As part
of its EndPowerup() function, it sets the psprite to the regular version,
but the player actor has already been destroyed.
- Fixed: FinishThingdef() needs to check for valid names, because weapons
could have inherited valid pointers from their superclass.
- Fixed: fuglyname didn't work.
- Fixed: Redefining $ambient sounds leaked memory.
- Added Jim's crashcatcher.c fix for better shell support.
- VC7.1 seems to have no trouble distinguishing between passing a (const
TypeInfo *) reference to operator<< and the generic, templated (object *)
version, so a few places that can benefit from it now use it. I believe
VC6 had problems with this, which is why I didn't do it all along. The
function's implementation was also moved out of dobject.cpp and into
farchive.cpp.
- Fixed: UnpackPixels() unpacked all chunks in a byte, which is wrong for the
last byte in a row if the image width is not an even multiple of the number
pixels per byte.
- Fixed: P_TranslateLineDef() should only clear monster activation for secret
useable lines, not crossable lines.
- Fixed: Some leftover P_IsHostile() calls still needed to be rewritten.
- Fixed: AWeaponHolder::Serialize() wrote the class type in all circumstances.
SVN r20 (trunk)
2006-03-14 06:11:39 +00:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{ // Hexen doesn't have a backpack, foo!
|
|
|
|
type = NULL;
|
|
|
|
}
|
|
|
|
if (type != NULL)
|
|
|
|
{
|
|
|
|
GiveSpawner (player, type, 1);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!giveall)
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2006-02-24 04:48:15 +00:00
|
|
|
if (giveall || stricmp (name, "ammo") == 0)
|
|
|
|
{
|
|
|
|
// Find every unique type of ammo. Give it to the player if
|
|
|
|
// he doesn't have it already, and set each to its maximum.
|
2006-05-10 02:40:43 +00:00
|
|
|
for (unsigned int i = 0; i < PClass::m_Types.Size(); ++i)
|
2006-02-24 04:48:15 +00:00
|
|
|
{
|
2006-05-10 02:40:43 +00:00
|
|
|
const PClass *type = PClass::m_Types[i];
|
2006-02-24 04:48:15 +00:00
|
|
|
|
2006-05-10 02:40:43 +00:00
|
|
|
if (type->ParentClass == RUNTIME_CLASS(AAmmo))
|
2006-02-24 04:48:15 +00:00
|
|
|
{
|
|
|
|
AInventory *ammo = player->mo->FindInventory (type);
|
|
|
|
if (ammo == NULL)
|
|
|
|
{
|
2006-07-16 09:10:45 +00:00
|
|
|
ammo = static_cast<AInventory *>(Spawn (type, 0, 0, 0, NO_REPLACE));
|
2006-02-24 04:48:15 +00:00
|
|
|
ammo->AttachToOwner (player->mo);
|
|
|
|
ammo->Amount = ammo->MaxAmount;
|
|
|
|
}
|
|
|
|
else if (ammo->Amount < ammo->MaxAmount)
|
|
|
|
{
|
|
|
|
ammo->Amount = ammo->MaxAmount;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!giveall)
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (giveall || stricmp (name, "armor") == 0)
|
|
|
|
{
|
|
|
|
if (gameinfo.gametype != GAME_Hexen)
|
|
|
|
{
|
2006-07-16 09:10:45 +00:00
|
|
|
ABasicArmorPickup *armor = Spawn<ABasicArmorPickup> (0,0,0, NO_REPLACE);
|
2006-02-24 04:48:15 +00:00
|
|
|
armor->SaveAmount = 100*deh.BlueAC;
|
|
|
|
armor->SavePercent = gameinfo.gametype != GAME_Heretic ? FRACUNIT/2 : FRACUNIT*3/4;
|
|
|
|
if (!armor->TryPickup (player->mo))
|
|
|
|
{
|
|
|
|
armor->Destroy ();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
for (i = 0; i < 4; ++i)
|
|
|
|
{
|
2006-07-16 09:10:45 +00:00
|
|
|
AHexenArmor *armor = Spawn<AHexenArmor> (0,0,0, NO_REPLACE);
|
2006-02-24 04:48:15 +00:00
|
|
|
armor->health = i;
|
|
|
|
armor->Amount = 0;
|
|
|
|
if (!armor->TryPickup (player->mo))
|
|
|
|
{
|
|
|
|
armor->Destroy ();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!giveall)
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (giveall || stricmp (name, "keys") == 0)
|
|
|
|
{
|
2006-05-10 02:40:43 +00:00
|
|
|
for (unsigned int i = 0; i < PClass::m_Types.Size(); ++i)
|
2006-02-24 04:48:15 +00:00
|
|
|
{
|
2006-05-10 02:40:43 +00:00
|
|
|
if (PClass::m_Types[i]->IsDescendantOf (RUNTIME_CLASS(AKey)))
|
2006-02-24 04:48:15 +00:00
|
|
|
{
|
2006-05-10 02:40:43 +00:00
|
|
|
AKey *key = (AKey *)GetDefaultByType (PClass::m_Types[i]);
|
2006-02-24 04:48:15 +00:00
|
|
|
if (key->KeyNumber != 0)
|
|
|
|
{
|
2006-07-16 09:10:45 +00:00
|
|
|
key = static_cast<AKey *>(Spawn (PClass::m_Types[i], 0,0,0, NO_REPLACE));
|
2006-02-24 04:48:15 +00:00
|
|
|
if (!key->TryPickup (player->mo))
|
|
|
|
{
|
|
|
|
key->Destroy ();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if (!giveall)
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (giveall || stricmp (name, "weapons") == 0)
|
|
|
|
{
|
|
|
|
AWeapon *savedpending = player->PendingWeapon;
|
2006-05-10 02:40:43 +00:00
|
|
|
for (unsigned int i = 0; i < PClass::m_Types.Size(); ++i)
|
2006-02-24 04:48:15 +00:00
|
|
|
{
|
2006-05-10 02:40:43 +00:00
|
|
|
type = PClass::m_Types[i];
|
2006-02-24 04:48:15 +00:00
|
|
|
if (type != RUNTIME_CLASS(AWeapon) &&
|
|
|
|
type->IsDescendantOf (RUNTIME_CLASS(AWeapon)))
|
|
|
|
{
|
|
|
|
AWeapon *def = (AWeapon*)GetDefaultByType (type);
|
|
|
|
if (!(def->WeaponFlags & WIF_CHEATNOTWEAPON))
|
|
|
|
{
|
|
|
|
GiveSpawner (player, type, 1);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
player->PendingWeapon = savedpending;
|
|
|
|
|
|
|
|
if (!giveall)
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (giveall || stricmp (name, "artifacts") == 0)
|
|
|
|
{
|
2006-05-10 02:40:43 +00:00
|
|
|
for (unsigned int i = 0; i < PClass::m_Types.Size(); ++i)
|
2006-02-24 04:48:15 +00:00
|
|
|
{
|
2006-05-10 02:40:43 +00:00
|
|
|
type = PClass::m_Types[i];
|
2006-02-24 04:48:15 +00:00
|
|
|
if (type->IsDescendantOf (RUNTIME_CLASS(AInventory)))
|
|
|
|
{
|
|
|
|
AInventory *def = (AInventory*)GetDefaultByType (type);
|
2008-06-15 18:36:26 +00:00
|
|
|
if (def->Icon.isValid() && def->MaxAmount > 1 &&
|
2006-02-24 04:48:15 +00:00
|
|
|
!type->IsDescendantOf (RUNTIME_CLASS(APuzzleItem)) &&
|
|
|
|
!type->IsDescendantOf (RUNTIME_CLASS(APowerup)) &&
|
|
|
|
!type->IsDescendantOf (RUNTIME_CLASS(AArmor)))
|
|
|
|
{
|
|
|
|
GiveSpawner (player, type, 1);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if (!giveall)
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (giveall || stricmp (name, "puzzlepieces") == 0)
|
|
|
|
{
|
2006-05-10 02:40:43 +00:00
|
|
|
for (unsigned int i = 0; i < PClass::m_Types.Size(); ++i)
|
2006-02-24 04:48:15 +00:00
|
|
|
{
|
2006-05-10 02:40:43 +00:00
|
|
|
type = PClass::m_Types[i];
|
2006-02-24 04:48:15 +00:00
|
|
|
if (type->IsDescendantOf (RUNTIME_CLASS(APuzzleItem)))
|
|
|
|
{
|
|
|
|
AInventory *def = (AInventory*)GetDefaultByType (type);
|
2008-06-15 18:36:26 +00:00
|
|
|
if (def->Icon.isValid())
|
2006-02-24 04:48:15 +00:00
|
|
|
{
|
|
|
|
GiveSpawner (player, type, 1);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if (!giveall)
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2006-04-11 08:36:23 +00:00
|
|
|
if (giveall)
|
|
|
|
return;
|
|
|
|
|
2006-05-10 02:40:43 +00:00
|
|
|
type = PClass::FindClass (name);
|
2006-02-24 04:48:15 +00:00
|
|
|
if (type == NULL || !type->IsDescendantOf (RUNTIME_CLASS(AInventory)))
|
|
|
|
{
|
|
|
|
if (player == &players[consoleplayer])
|
|
|
|
Printf ("Unknown item \"%s\"\n", name);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
GiveSpawner (player, type, amount);
|
|
|
|
}
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2007-12-23 14:23:52 +00:00
|
|
|
void cht_Take (player_t *player, const char *name, int amount)
|
|
|
|
{
|
|
|
|
bool takeall;
|
|
|
|
const PClass *type;
|
|
|
|
|
|
|
|
if (player->mo == NULL || player->health <= 0)
|
|
|
|
{
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
takeall = (stricmp (name, "all") == 0);
|
|
|
|
|
|
|
|
if (!takeall && stricmp (name, "health") == 0)
|
|
|
|
{
|
|
|
|
if (player->mo->health - amount <= 0
|
|
|
|
|| player->health - amount <= 0
|
|
|
|
|| amount == 0)
|
|
|
|
{
|
|
|
|
|
|
|
|
cht_Suicide (player);
|
|
|
|
|
|
|
|
if (player == &players[consoleplayer])
|
|
|
|
C_HideConsole ();
|
|
|
|
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (amount > 0)
|
|
|
|
{
|
|
|
|
if (player->mo)
|
|
|
|
{
|
|
|
|
player->mo->health -= amount;
|
|
|
|
player->health = player->mo->health;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
player->health -= amount;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!takeall)
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (takeall || stricmp (name, "backpack") == 0)
|
|
|
|
{
|
|
|
|
// Select the correct type of backpack based on the game
|
|
|
|
if (gameinfo.gametype == GAME_Heretic)
|
|
|
|
{
|
|
|
|
type = PClass::FindClass ("BagOfHolding");
|
|
|
|
}
|
|
|
|
else if (gameinfo.gametype == GAME_Strife)
|
|
|
|
{
|
|
|
|
type = PClass::FindClass ("AmmoSatchel");
|
|
|
|
}
|
|
|
|
else if (gameinfo.gametype == GAME_Doom)
|
|
|
|
{
|
|
|
|
type = PClass::FindClass ("Backpack");
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{ // Hexen doesn't have a backpack, foo!
|
|
|
|
type = NULL;
|
|
|
|
}
|
|
|
|
if (type != NULL)
|
|
|
|
{
|
|
|
|
AActor *backpack = player->mo->FindInventory (type);
|
|
|
|
|
|
|
|
if (backpack)
|
|
|
|
backpack->Destroy ();
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!takeall)
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (takeall || stricmp (name, "ammo") == 0)
|
|
|
|
{
|
|
|
|
for (unsigned int i = 0; i < PClass::m_Types.Size(); ++i)
|
|
|
|
{
|
|
|
|
const PClass *type = PClass::m_Types[i];
|
|
|
|
|
|
|
|
if (type->ParentClass == RUNTIME_CLASS (AAmmo))
|
|
|
|
{
|
|
|
|
AInventory *ammo = player->mo->FindInventory (type);
|
|
|
|
|
|
|
|
if (ammo)
|
|
|
|
ammo->Amount = 0;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!takeall)
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (takeall || stricmp (name, "armor") == 0)
|
|
|
|
{
|
|
|
|
for (unsigned int i = 0; i < PClass::m_Types.Size(); ++i)
|
|
|
|
{
|
|
|
|
type = PClass::m_Types[i];
|
|
|
|
|
|
|
|
if (type->IsDescendantOf (RUNTIME_CLASS (AArmor)))
|
|
|
|
{
|
|
|
|
AActor *armor = player->mo->FindInventory (type);
|
|
|
|
|
|
|
|
if (armor)
|
|
|
|
armor->Destroy ();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!takeall)
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (takeall || stricmp (name, "keys") == 0)
|
|
|
|
{
|
|
|
|
for (unsigned int i = 0; i < PClass::m_Types.Size(); ++i)
|
|
|
|
{
|
|
|
|
type = PClass::m_Types[i];
|
|
|
|
|
|
|
|
if (type->IsDescendantOf (RUNTIME_CLASS (AKey)))
|
|
|
|
{
|
|
|
|
AActor *key = player->mo->FindInventory (type);
|
|
|
|
|
|
|
|
if (key)
|
|
|
|
key->Destroy ();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!takeall)
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (takeall || stricmp (name, "weapons") == 0)
|
|
|
|
{
|
|
|
|
for (unsigned int i = 0; i < PClass::m_Types.Size(); ++i)
|
|
|
|
{
|
|
|
|
type = PClass::m_Types[i];
|
|
|
|
|
|
|
|
if (type != RUNTIME_CLASS(AWeapon) &&
|
|
|
|
type->IsDescendantOf (RUNTIME_CLASS (AWeapon)))
|
|
|
|
{
|
|
|
|
AActor *weapon = player->mo->FindInventory (type);
|
|
|
|
|
|
|
|
if (weapon)
|
|
|
|
weapon->Destroy ();
|
|
|
|
|
|
|
|
player->ReadyWeapon = NULL;
|
|
|
|
player->PendingWeapon = WP_NOCHANGE;
|
|
|
|
player->psprites[ps_weapon].state = NULL;
|
|
|
|
player->psprites[ps_flash].state = NULL;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!takeall)
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (takeall || stricmp (name, "artifacts") == 0)
|
|
|
|
{
|
|
|
|
for (unsigned int i = 0; i < PClass::m_Types.Size(); ++i)
|
|
|
|
{
|
|
|
|
type = PClass::m_Types[i];
|
|
|
|
|
|
|
|
if (type->IsDescendantOf (RUNTIME_CLASS (AInventory)))
|
|
|
|
{
|
|
|
|
if (!type->IsDescendantOf (RUNTIME_CLASS (APuzzleItem)) &&
|
|
|
|
!type->IsDescendantOf (RUNTIME_CLASS (APowerup)) &&
|
|
|
|
!type->IsDescendantOf (RUNTIME_CLASS (AArmor)) &&
|
|
|
|
!type->IsDescendantOf (RUNTIME_CLASS (AWeapon)) &&
|
|
|
|
!type->IsDescendantOf (RUNTIME_CLASS (AKey)))
|
|
|
|
{
|
|
|
|
AActor *artifact = player->mo->FindInventory (type);
|
|
|
|
|
|
|
|
if (artifact)
|
|
|
|
artifact->Destroy ();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!takeall)
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (takeall || stricmp (name, "puzzlepieces") == 0)
|
|
|
|
{
|
|
|
|
for (unsigned int i = 0; i < PClass::m_Types.Size(); ++i)
|
|
|
|
{
|
|
|
|
type = PClass::m_Types[i];
|
|
|
|
|
|
|
|
if (type->IsDescendantOf (RUNTIME_CLASS (APuzzleItem)))
|
|
|
|
{
|
|
|
|
AActor *puzzlepiece = player->mo->FindInventory (type);
|
|
|
|
|
|
|
|
if (puzzlepiece)
|
|
|
|
puzzlepiece->Destroy ();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!takeall)
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (takeall)
|
|
|
|
return;
|
|
|
|
|
|
|
|
type = PClass::FindClass (name);
|
|
|
|
if (type == NULL || !type->IsDescendantOf (RUNTIME_CLASS (AInventory)))
|
|
|
|
{
|
|
|
|
if (player == &players[consoleplayer])
|
|
|
|
Printf ("Unknown item \"%s\"\n", name);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
AInventory *inventory = player->mo->FindInventory (type);
|
|
|
|
|
|
|
|
if (inventory != NULL)
|
|
|
|
{
|
|
|
|
inventory->Amount -= amount ? amount : 1;
|
|
|
|
|
|
|
|
if (inventory->Amount <= 0)
|
|
|
|
{
|
|
|
|
if (inventory->ItemFlags & IF_KEEPDEPLETED)
|
|
|
|
{
|
|
|
|
inventory->Amount = 0;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
inventory->Destroy ();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2006-02-24 04:48:15 +00:00
|
|
|
void cht_Suicide (player_t *plyr)
|
|
|
|
{
|
|
|
|
if (plyr->mo != NULL)
|
|
|
|
{
|
|
|
|
plyr->mo->flags |= MF_SHOOTABLE;
|
|
|
|
plyr->mo->flags2 &= ~MF2_INVULNERABLE;
|
2006-10-31 14:53:21 +00:00
|
|
|
P_DamageMobj (plyr->mo, plyr->mo, plyr->mo, 1000000, NAME_Suicide);
|
2006-02-24 04:48:15 +00:00
|
|
|
plyr->mo->flags &= ~MF_SHOOTABLE;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2006-09-14 00:02:31 +00:00
|
|
|
bool CheckCheatmode ();
|
2006-02-24 04:48:15 +00:00
|
|
|
|
|
|
|
CCMD (mdk)
|
|
|
|
{
|
|
|
|
if (CheckCheatmode ())
|
|
|
|
return;
|
|
|
|
|
|
|
|
Net_WriteByte (DEM_GENERICCHEAT);
|
|
|
|
Net_WriteByte (CHT_MDK);
|
|
|
|
}
|