dxx-rebirth/similar/main/gamecntl.cpp

1896 lines
52 KiB
C++
Raw Normal View History

2006-03-20 17:12:09 +00:00
/*
2014-06-01 17:55:23 +00:00
* Portions of this file are copyright Rebirth contributors and licensed as
* described in COPYING.txt.
* Portions of this file are copyright Parallax Software and licensed
* according to the Parallax license below.
* See COPYING.txt for license details.
2006-03-20 17:12:09 +00:00
THE COMPUTER CODE CONTAINED HEREIN IS THE SOLE PROPERTY OF PARALLAX
SOFTWARE CORPORATION ("PARALLAX"). PARALLAX, IN DISTRIBUTING THE CODE TO
END-USERS, AND SUBJECT TO ALL OF THE TERMS AND CONDITIONS HEREIN, GRANTS A
ROYALTY-FREE, PERPETUAL LICENSE TO SUCH END-USERS FOR USE BY SUCH END-USERS
IN USING, DISPLAYING, AND CREATING DERIVATIVE WORKS THEREOF, SO LONG AS
SUCH USE, DISPLAY OR CREATION IS FOR NON-COMMERCIAL, ROYALTY OR REVENUE
FREE PURPOSES. IN NO EVENT SHALL THE END-USER USE THE COMPUTER CODE
CONTAINED HEREIN FOR REVENUE-BEARING PURPOSES. THE END-USER UNDERSTANDS
AND AGREES TO THE TERMS HEREIN AND ACCEPTS THE SAME BY USE OF THIS FILE.
COPYRIGHT 1993-1999 PARALLAX SOFTWARE CORPORATION. ALL RIGHTS RESERVED.
*/
/*
*
* Game Controls Stuff
*
*/
//#define DOOR_DEBUGGING
#include <algorithm>
2006-03-20 17:12:09 +00:00
#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <stdarg.h>
#include "pstypes.h"
#include "window.h"
2006-03-20 17:12:09 +00:00
#include "console.h"
2013-12-26 04:18:28 +00:00
#include "strutil.h"
2006-03-20 17:12:09 +00:00
#include "game.h"
#include "player.h"
#include "key.h"
#include "object.h"
#include "menu.h"
#include "physics.h"
#include "dxxerror.h"
2006-03-20 17:12:09 +00:00
#include "joy.h"
#include "iff.h"
#include "pcx.h"
#include "timer.h"
#include "render.h"
#include "laser.h"
#include "screens.h"
#include "textures.h"
#include "slew.h"
#include "gauges.h"
#include "texmap.h"
#include "3d.h"
#include "effects.h"
#include "gameseg.h"
#include "wall.h"
#include "ai.h"
#include "rbaudio.h"
2006-03-20 17:12:09 +00:00
#include "digi.h"
#include "u_mem.h"
#include "palette.h"
#include "morph.h"
#include "lighting.h"
#include "newdemo.h"
#include "weapon.h"
#include "sounds.h"
#include "args.h"
#include "gameseq.h"
#include "automap.h"
#include "text.h"
#include "powerup.h"
#include "songs.h"
2006-03-20 17:12:09 +00:00
#include "newmenu.h"
#include "gamefont.h"
#include "endlevel.h"
#include "config.h"
2006-03-20 17:12:09 +00:00
#include "kconfig.h"
#include "mouse.h"
#include "titles.h"
#include "gr.h"
#include "playsave.h"
#include "scores.h"
#include "multi.h"
#include "cntrlcen.h"
#include "fuelcen.h"
2006-03-20 17:12:09 +00:00
#include "pcx.h"
#include "state.h"
#include "piggy.h"
#include "multibot.h"
#include "ai.h"
#include "rbaudio.h"
#include "switch.h"
#if defined(DXX_BUILD_DESCENT_II)
2006-03-20 17:12:09 +00:00
#include "escort.h"
#include "movie.h"
#endif
#include "window.h"
2006-03-20 17:12:09 +00:00
#ifdef EDITOR
#include "editor/editor.h"
#include "editor/esegment.h"
2006-03-20 17:12:09 +00:00
#endif
2014-09-07 04:04:08 +00:00
#include "compiler-array.h"
2014-04-27 23:12:34 +00:00
#include "compiler-range_for.h"
2014-10-12 23:05:46 +00:00
#include "highest_valid.h"
2014-04-27 23:12:34 +00:00
#include "partial_range.h"
2013-06-30 02:22:56 +00:00
#include <SDL.h>
2006-03-20 17:12:09 +00:00
using std::min;
2006-03-20 17:12:09 +00:00
// Global Variables -----------------------------------------------------------
int Debug_spew;
2006-03-20 17:12:09 +00:00
// Function prototypes --------------------------------------------------------
#ifndef RELEASE
2013-09-22 22:26:27 +00:00
static void do_cheat_menu();
static void advance_sound();
static void play_test_sound();
2006-03-20 17:12:09 +00:00
#endif
#define key_isfunc(k) (((k&0xff)>=KEY_F1 && (k&0xff)<=KEY_F10) || (k&0xff)==KEY_F11 || (k&0xff)==KEY_F12)
// Functions ------------------------------------------------------------------
#if defined(DXX_BUILD_DESCENT_II)
2006-03-20 17:12:09 +00:00
#define CONVERTER_RATE 20 //10 units per second xfer rate
#define CONVERTER_SCALE 2 //2 units energy -> 1 unit shields
#define CONVERTER_SOUND_DELAY (f1_0/2) //play every half second
2013-09-22 22:26:27 +00:00
static void transfer_energy_to_shield()
2006-03-20 17:12:09 +00:00
{
fix e; //how much energy gets transfered
static fix64 last_play_time=0;
2006-03-20 17:12:09 +00:00
e = min(min(FrameTime*CONVERTER_RATE,Players[Player_num].energy - INITIAL_ENERGY),(MAX_SHIELDS-Players[Player_num].shields)*CONVERTER_SCALE);
2006-03-20 17:12:09 +00:00
if (e <= 0) {
if (Players[Player_num].energy <= INITIAL_ENERGY) {
HUD_init_message(HM_DEFAULT, "Need more than %i energy to enable transfer", f2i(INITIAL_ENERGY));
}
else if (Players[Player_num].shields >= MAX_SHIELDS) {
HUD_init_message_literal(HM_DEFAULT, "No transfer: Shields already at max");
}
2006-03-20 17:12:09 +00:00
return;
}
Players[Player_num].energy -= e;
Players[Player_num].shields += e/CONVERTER_SCALE;
if (last_play_time > GameTime64)
2006-03-20 17:12:09 +00:00
last_play_time = 0;
if (GameTime64 > last_play_time+CONVERTER_SOUND_DELAY) {
2006-03-20 17:12:09 +00:00
digi_play_sample_once(SOUND_CONVERT_ENERGY, F1_0);
last_play_time = GameTime64;
2006-03-20 17:12:09 +00:00
}
}
#endif
2006-03-20 17:12:09 +00:00
// Control Functions
2013-09-22 22:26:27 +00:00
static fix64 newdemo_single_frame_time;
2006-03-20 17:12:09 +00:00
2013-09-22 22:26:27 +00:00
static void update_vcr_state(void)
2006-03-20 17:12:09 +00:00
{
if ((keyd_pressed[KEY_LSHIFT] || keyd_pressed[KEY_RSHIFT]) && keyd_pressed[KEY_RIGHT] && d_tick_step)
2006-03-20 17:12:09 +00:00
Newdemo_vcr_state = ND_STATE_FASTFORWARD;
else if ((keyd_pressed[KEY_LSHIFT] || keyd_pressed[KEY_RSHIFT]) && keyd_pressed[KEY_LEFT] && d_tick_step)
2006-03-20 17:12:09 +00:00
Newdemo_vcr_state = ND_STATE_REWINDING;
else if (!(keyd_pressed[KEY_LCTRL] || keyd_pressed[KEY_RCTRL]) && keyd_pressed[KEY_RIGHT] && ((GameTime64 - newdemo_single_frame_time) >= F1_0) && d_tick_step)
2006-03-20 17:12:09 +00:00
Newdemo_vcr_state = ND_STATE_ONEFRAMEFORWARD;
else if (!(keyd_pressed[KEY_LCTRL] || keyd_pressed[KEY_RCTRL]) && keyd_pressed[KEY_LEFT] && ((GameTime64 - newdemo_single_frame_time) >= F1_0) && d_tick_step)
2006-03-20 17:12:09 +00:00
Newdemo_vcr_state = ND_STATE_ONEFRAMEBACKWARD;
else if ((Newdemo_vcr_state == ND_STATE_FASTFORWARD) || (Newdemo_vcr_state == ND_STATE_REWINDING))
Newdemo_vcr_state = ND_STATE_PLAYBACK;
}
#if defined(DXX_BUILD_DESCENT_II)
2006-03-20 17:12:09 +00:00
//returns which bomb will be dropped next time the bomb key is pressed
int which_bomb()
{
int bomb;
//use the last one selected, unless there aren't any, in which case use
//the other if there are any
// If hoard game, only let the player drop smart mines
if (game_mode_hoard())
2006-03-20 17:12:09 +00:00
return SMART_MINE_INDEX;
bomb = Secondary_last_was_super[PROXIMITY_INDEX]?SMART_MINE_INDEX:PROXIMITY_INDEX;
if (Players[Player_num].secondary_ammo[bomb] == 0 &&
Players[Player_num].secondary_ammo[SMART_MINE_INDEX+PROXIMITY_INDEX-bomb] != 0) {
bomb = SMART_MINE_INDEX+PROXIMITY_INDEX-bomb;
Secondary_last_was_super[bomb%SUPER_WEAPON] = (bomb == SMART_MINE_INDEX);
}
2006-03-20 17:12:09 +00:00
return bomb;
}
#endif
2006-03-20 17:12:09 +00:00
2013-09-22 22:26:27 +00:00
static void do_weapon_n_item_stuff()
2006-03-20 17:12:09 +00:00
{
if (Controls.state.fire_flare > 0)
{
Controls.state.fire_flare = 0;
2006-03-20 17:12:09 +00:00
if (allowed_to_fire_flare())
Flare_create(ConsoleObject);
}
2006-03-20 17:12:09 +00:00
if (allowed_to_fire_missile() && Controls.state.fire_secondary)
Global_missile_firing_count += Weapon_info[Secondary_weapon_to_weapon_info[Secondary_weapon]].fire_count;
2006-03-20 17:12:09 +00:00
if (Global_missile_firing_count) {
do_missile_firing(0);
2006-03-20 17:12:09 +00:00
Global_missile_firing_count--;
}
if (Controls.state.cycle_primary > 0)
{
for (int i=0;i < Controls.state.cycle_primary;i++)
CyclePrimary ();
Controls.state.cycle_primary = 0;
}
if (Controls.state.cycle_secondary > 0)
{
for (int i=0;i < Controls.state.cycle_secondary;i++)
CycleSecondary ();
Controls.state.cycle_secondary = 0;
}
if (Controls.state.select_weapon > 0)
{
Controls.state.select_weapon--;
do_weapon_select(Controls.state.select_weapon>4?Controls.state.select_weapon-5:Controls.state.select_weapon,Controls.state.select_weapon>4?1:0);
Controls.state.select_weapon = 0;
}
#if defined(DXX_BUILD_DESCENT_II)
if (Controls.state.headlight > 0)
{
for (int i=0;i < Controls.state.headlight;i++)
toggle_headlight_active ();
Controls.state.headlight = 0;
}
#endif
2006-03-20 17:12:09 +00:00
if (Global_missile_firing_count < 0)
Global_missile_firing_count = 0;
// Drop proximity bombs.
while (Controls.state.drop_bomb > 0)
{
do_missile_firing(1);
Controls.state.drop_bomb--;
2006-03-20 17:12:09 +00:00
}
#if defined(DXX_BUILD_DESCENT_II)
if (Controls.state.toggle_bomb > 0)
{
int bomb = Secondary_last_was_super[PROXIMITY_INDEX]?PROXIMITY_INDEX:SMART_MINE_INDEX;
if (!Players[Player_num].secondary_ammo[PROXIMITY_INDEX] && !Players[Player_num].secondary_ammo[SMART_MINE_INDEX])
{
digi_play_sample_once( SOUND_BAD_SELECTION, F1_0 );
HUD_init_message_literal(HM_DEFAULT, "No bombs available!");
}
else
{
if (Players[Player_num].secondary_ammo[bomb]==0)
{
digi_play_sample_once( SOUND_BAD_SELECTION, F1_0 );
HUD_init_message(HM_DEFAULT, "No %s available!",(bomb==SMART_MINE_INDEX)?"Smart mines":"Proximity bombs");
}
else
{
Secondary_last_was_super[PROXIMITY_INDEX]=!Secondary_last_was_super[PROXIMITY_INDEX];
digi_play_sample_once( SOUND_GOOD_SELECTION_SECONDARY, F1_0 );
}
}
Controls.state.toggle_bomb = 0;
}
if (Controls.state.energy_to_shield && (Players[Player_num].flags & PLAYER_FLAGS_CONVERTER))
transfer_energy_to_shield();
#endif
2006-03-20 17:12:09 +00:00
}
2013-09-22 22:26:27 +00:00
static void format_time(char *str, int secs_int)
2006-03-20 17:12:09 +00:00
{
int h, m, s;
h = secs_int/3600;
s = secs_int%3600;
m = s / 60;
s = s % 60;
sprintf(str, "%1d:%02d:%02d", h, m, s );
}
2014-09-07 18:06:59 +00:00
struct pause_window : ignore_window_pointer_t
2014-09-07 04:04:08 +00:00
{
array<char, 1024> msg;
};
2010-01-09 09:19:26 +00:00
//Process selected keys until game unpaused
2014-10-04 21:47:13 +00:00
static window_event_result pause_handler(window *wind,const d_event &event, pause_window *p)
2006-03-20 17:12:09 +00:00
{
int key;
2010-01-09 09:19:26 +00:00
2014-10-04 21:47:13 +00:00
switch (event.type)
2010-01-09 09:19:26 +00:00
{
case EVENT_WINDOW_ACTIVATED:
game_flush_inputs();
break;
case EVENT_KEY_COMMAND:
key = event_key_get(event);
switch (key)
{
case 0:
break;
case KEY_ESC:
window_close(wind);
2014-08-06 02:10:49 +00:00
return window_event_result::close;
case KEY_F1:
show_help();
2014-08-06 02:10:49 +00:00
return window_event_result::handled;
case KEY_PAUSE:
window_close(wind);
2014-08-06 02:10:49 +00:00
return window_event_result::close;
default:
break;
}
break;
case EVENT_IDLE:
timer_delay2(50);
2010-01-09 09:19:26 +00:00
break;
case EVENT_WINDOW_DRAW:
2014-09-07 04:04:08 +00:00
show_boxed_message(&p->msg[0], 1);
2010-01-09 09:19:26 +00:00
break;
case EVENT_WINDOW_CLOSE:
songs_resume();
2014-09-07 04:04:08 +00:00
delete p;
2010-01-09 09:19:26 +00:00
break;
2010-01-09 09:19:26 +00:00
default:
break;
}
2014-08-06 02:10:49 +00:00
return window_event_result::ignored;
2010-01-09 09:19:26 +00:00
}
2013-09-22 22:26:27 +00:00
static int do_game_pause()
2010-01-09 09:19:26 +00:00
{
2006-03-20 17:12:09 +00:00
char total_time[9],level_time[9];
if (Game_mode & GM_MULTI)
2006-03-20 17:12:09 +00:00
{
netplayerinfo_on= !netplayerinfo_on;
return(KEY_PAUSE);
2006-03-20 17:12:09 +00:00
}
2014-09-07 04:04:08 +00:00
pause_window *p = new pause_window;
songs_pause();
2006-03-20 17:12:09 +00:00
format_time(total_time, f2i(Players[Player_num].time_total) + Players[Player_num].hours_total*3600);
format_time(level_time, f2i(Players[Player_num].time_level) + Players[Player_num].hours_level*3600);
2007-10-14 16:25:55 +00:00
if (Newdemo_state!=ND_STATE_PLAYBACK)
2014-09-07 04:04:08 +00:00
snprintf(&p->msg[0], p->msg.size(),"PAUSE\n\nSkill level: %s\nHostages on board: %d\nTime on level: %s\nTotal time in game: %s",MENU_DIFFICULTY_TEXT(Difficulty_level),Players[Player_num].hostages_on_board,level_time,total_time);
2007-10-14 16:25:55 +00:00
else
2014-09-07 04:04:08 +00:00
snprintf(&p->msg[0], p->msg.size(),"PAUSE\n\nSkill level: %s\nHostages on board: %d\n",MENU_DIFFICULTY_TEXT(Difficulty_level),Players[Player_num].hostages_on_board);
set_screen_mode(SCREEN_MENU);
2014-09-07 04:04:08 +00:00
if (!window_create(&grd_curscreen->sc_canvas, 0, 0, SWIDTH, SHEIGHT, pause_handler, p))
delete p;
2010-01-09 09:19:26 +00:00
return 0 /*key*/; // Keycode returning ripped out (kreatordxx)
2006-03-20 17:12:09 +00:00
}
2013-09-22 22:26:27 +00:00
static int HandleEndlevelKey(int key)
2006-03-20 17:12:09 +00:00
{
switch (key)
{
case KEY_COMMAND+KEY_P:
case KEY_PAUSE:
do_game_pause();
return 1;
case KEY_ESC:
stop_endlevel_sequence();
last_drawn_cockpit=-1;
return 1;
2006-03-20 17:12:09 +00:00
}
return 0;
2006-03-20 17:12:09 +00:00
}
2014-10-04 21:47:13 +00:00
static int HandleDeathInput(const d_event &event)
2006-03-20 17:12:09 +00:00
{
2014-10-04 21:47:13 +00:00
if (event.type == EVENT_KEY_COMMAND)
{
int key = event_key_get(event);
2006-03-20 17:12:09 +00:00
if (Player_exploded && !key_isfunc(key) && key != KEY_PAUSE && key)
Death_sequence_aborted = 1; //Any key but func or modifier aborts
if (key == KEY_ESC)
if (ConsoleObject->flags & OF_EXPLODING)
Death_sequence_aborted = 1;
2006-03-20 17:12:09 +00:00
}
2014-10-04 21:47:13 +00:00
if (Player_exploded && (event.type == EVENT_JOYSTICK_BUTTON_UP || event.type == EVENT_MOUSE_BUTTON_UP))
Death_sequence_aborted = 1;
2006-03-20 17:12:09 +00:00
if (Death_sequence_aborted)
{
2006-03-20 17:12:09 +00:00
game_flush_inputs();
return 1;
}
return 0;
2006-03-20 17:12:09 +00:00
}
2013-09-22 22:26:27 +00:00
static int HandleDemoKey(int key)
2006-03-20 17:12:09 +00:00
{
switch (key) {
KEY_MAC(case KEY_COMMAND+KEY_1:)
case KEY_F1: show_newdemo_help(); break;
KEY_MAC(case KEY_COMMAND+KEY_2:)
case KEY_F2: do_options_menu(); break;
KEY_MAC(case KEY_COMMAND+KEY_3:)
2006-03-20 17:12:09 +00:00
case KEY_F3:
if (Viewer->type == OBJ_PLAYER)
2006-03-20 17:12:09 +00:00
toggle_cockpit();
break;
KEY_MAC(case KEY_COMMAND+KEY_4:)
Giving credits function ability to use custom creditfile (again); Made laser-offset for laser exclusive so Prox mines won't go tru doors; Preventing cycling tru cockpit modes while dead, but allowing to load a state; Implemented D2X' lighting code to D1X (faster, better, sexier - weeee); Try to hop over some errors regarding walls/doors in levels instead of using -1 indexes for arrays; Made the briefing text ptr a bit more failsafe in case the file is corrupt/non-standard; Made scores use the menu screen even in GAME OVER; Fixed bug in neighbour fields of Weapon Keys table; Added the Weapon Keys stuff to TABLE_CREATION; Fixed bug where D2X did not recall applied resolution in the resolutions menu; Simpler check to create DEMO_DIR; Seperated X/Y sensitivity for mouse and joystick; Flush controls when Automap toggles so keypress won't deactivate it again; Made FrameCount in Demos aligned to the Dropframe condition; Added KEy to ttoggle playback text off; Gracefully exit demo code if demo is corrupt; Removed that new percent counter because many old demos seem to have corrupted last frames; Closing endlevel data file if IFF error so the mission still can be freed; Fixed Cruising for keyboard which was not aligned to FPS correctly; Used mouse delta scaling in kconfig.c instead of mouse.c to not screw up when delta is requested in non-ingame situations - it actually belongs to the controls IMHO; Now support up to 8 joysticks; Changed some leftover malloc's to d_malloc and free to d_free
2008-10-16 17:27:02 +00:00
case KEY_F4: Newdemo_show_percentage = !Newdemo_show_percentage; break;
KEY_MAC(case KEY_COMMAND+KEY_7:)
2006-03-20 17:12:09 +00:00
case KEY_F7:
Show_kill_list = (Show_kill_list+1) % ((Newdemo_game_mode & GM_TEAM) ? 4 : 3);
2006-03-20 17:12:09 +00:00
break;
case KEY_ESC:
if (GameArg.SysAutoDemo)
{
int choice;
choice = nm_messagebox( NULL, 2, TXT_YES, TXT_NO, TXT_ABORT_AUTODEMO );
if (choice == 0)
GameArg.SysAutoDemo = 0;
else
break;
}
newdemo_stop_playback();
2006-03-20 17:12:09 +00:00
break;
case KEY_UP:
Newdemo_vcr_state = ND_STATE_PLAYBACK;
break;
case KEY_DOWN:
Newdemo_vcr_state = ND_STATE_PAUSED;
break;
case KEY_LEFT:
newdemo_single_frame_time = GameTime64;
2006-03-20 17:12:09 +00:00
Newdemo_vcr_state = ND_STATE_ONEFRAMEBACKWARD;
break;
case KEY_RIGHT:
newdemo_single_frame_time = GameTime64;
2006-03-20 17:12:09 +00:00
Newdemo_vcr_state = ND_STATE_ONEFRAMEFORWARD;
break;
case KEY_CTRLED + KEY_RIGHT:
newdemo_goto_end(0);
2006-03-20 17:12:09 +00:00
break;
case KEY_CTRLED + KEY_LEFT:
newdemo_goto_beginning();
break;
KEY_MAC(case KEY_COMMAND+KEY_P:)
2006-03-20 17:12:09 +00:00
case KEY_PAUSE:
do_game_pause();
break;
#ifdef macintosh
case KEY_COMMAND + KEY_SHIFTED + KEY_3:
#endif
Giving credits function ability to use custom creditfile (again); Made laser-offset for laser exclusive so Prox mines won't go tru doors; Preventing cycling tru cockpit modes while dead, but allowing to load a state; Implemented D2X' lighting code to D1X (faster, better, sexier - weeee); Try to hop over some errors regarding walls/doors in levels instead of using -1 indexes for arrays; Made the briefing text ptr a bit more failsafe in case the file is corrupt/non-standard; Made scores use the menu screen even in GAME OVER; Fixed bug in neighbour fields of Weapon Keys table; Added the Weapon Keys stuff to TABLE_CREATION; Fixed bug where D2X did not recall applied resolution in the resolutions menu; Simpler check to create DEMO_DIR; Seperated X/Y sensitivity for mouse and joystick; Flush controls when Automap toggles so keypress won't deactivate it again; Made FrameCount in Demos aligned to the Dropframe condition; Added KEy to ttoggle playback text off; Gracefully exit demo code if demo is corrupt; Removed that new percent counter because many old demos seem to have corrupted last frames; Closing endlevel data file if IFF error so the mission still can be freed; Fixed Cruising for keyboard which was not aligned to FPS correctly; Used mouse delta scaling in kconfig.c instead of mouse.c to not screw up when delta is requested in non-ingame situations - it actually belongs to the controls IMHO; Now support up to 8 joysticks; Changed some leftover malloc's to d_malloc and free to d_free
2008-10-16 17:27:02 +00:00
case KEY_PRINT_SCREEN:
{
if (PlayerCfg.PRShot)
{
gr_set_current_canvas(NULL);
render_frame(0, 0);
gr_set_curfont(MEDIUM2_FONT);
gr_string(SWIDTH-FSPACX(92),SHEIGHT-LINE_SPACING,"DXX-Rebirth\n");
gr_flip();
save_screen_shot(0);
}
else
{
int old_state;
old_state = Newdemo_show_percentage;
Newdemo_show_percentage = 0;
game_render_frame_mono(0);
2013-11-10 17:10:11 +00:00
if (!GameArg.DbgNoDoubleBuffer)
gr_flip();
save_screen_shot(0);
Newdemo_show_percentage = old_state;
}
2006-03-20 17:12:09 +00:00
break;
}
#ifndef NDEBUG
2006-03-20 17:12:09 +00:00
case KEY_DEBUGGED + KEY_I:
Newdemo_do_interpolate = !Newdemo_do_interpolate;
HUD_init_message(HM_DEFAULT, "Demo playback interpolation %s", Newdemo_do_interpolate?"ON":"OFF");
2006-03-20 17:12:09 +00:00
break;
case KEY_DEBUGGED + KEY_K: {
int how_many, c;
char filename[FILENAME_LEN], num[16];
newmenu_item m[6];
filename[0] = '\0';
nm_set_item_text(& m[ 0], "output file name");
nm_set_item_input(&m[ 1], 8, filename);
c = newmenu_do( NULL, NULL, 2, m, unused_newmenu_subfunction, unused_newmenu_userdata);
2006-03-20 17:12:09 +00:00
if (c == -2)
break;
strcat(filename, DEMO_EXT);
2006-03-20 17:12:09 +00:00
num[0] = '\0';
nm_set_item_text(& m[ 0], "strip how many bytes");
nm_set_item_input(&m[ 1], 16, num);
c = newmenu_do( NULL, NULL, 2, m, unused_newmenu_subfunction, unused_newmenu_userdata);
2006-03-20 17:12:09 +00:00
if (c == -2)
break;
how_many = atoi(num);
if (how_many <= 0)
break;
newdemo_strip_frames(filename, how_many);
break;
}
#endif
default:
return 0;
2006-03-20 17:12:09 +00:00
}
return 1;
2006-03-20 17:12:09 +00:00
}
#if defined(DXX_BUILD_DESCENT_II)
2006-03-20 17:12:09 +00:00
//switch a cockpit window to the next function
2013-09-22 22:26:27 +00:00
static int select_next_window_function(int w)
2006-03-20 17:12:09 +00:00
{
Assert(w==0 || w==1);
switch (PlayerCfg.Cockpit3DView[w]) {
2006-03-20 17:12:09 +00:00
case CV_NONE:
PlayerCfg.Cockpit3DView[w] = CV_REAR;
2006-03-20 17:12:09 +00:00
break;
case CV_REAR:
if (find_escort()) {
PlayerCfg.Cockpit3DView[w] = CV_ESCORT;
2006-03-20 17:12:09 +00:00
break;
}
//if no ecort, fall through
case CV_ESCORT:
Coop_view_player[w] = -1; //force first player
//fall through
case CV_COOP:
Marker_viewer_num[w] = -1;
if ((Game_mode & GM_MULTI_COOP) || (Game_mode & GM_TEAM)) {
PlayerCfg.Cockpit3DView[w] = CV_COOP;
2006-03-20 17:12:09 +00:00
while (1) {
Coop_view_player[w]++;
if (Coop_view_player[w] == (MAX_PLAYERS-1)) {
PlayerCfg.Cockpit3DView[w] = CV_MARKER;
2006-03-20 17:12:09 +00:00
goto case_marker;
}
if (Players[Coop_view_player[w]].connected != CONNECT_PLAYING)
continue;
2006-03-20 17:12:09 +00:00
if (Coop_view_player[w]==Player_num)
continue;
if (Game_mode & GM_MULTI_COOP)
break;
else if (get_team(Coop_view_player[w]) == get_team(Player_num))
break;
}
break;
}
//if not multi, fall through
case CV_MARKER:
case_marker:;
if ((Game_mode & GM_MULTI) && !(Game_mode & GM_MULTI_COOP) && Netgame.Allow_marker_view) { //anarchy only
PlayerCfg.Cockpit3DView[w] = CV_MARKER;
2006-03-20 17:12:09 +00:00
if (Marker_viewer_num[w] == -1)
Marker_viewer_num[w] = Player_num * 2;
else if (Marker_viewer_num[w] == Player_num * 2)
Marker_viewer_num[w]++;
else
PlayerCfg.Cockpit3DView[w] = CV_NONE;
2006-03-20 17:12:09 +00:00
}
else
PlayerCfg.Cockpit3DView[w] = CV_NONE;
2006-03-20 17:12:09 +00:00
break;
}
write_player_file();
return 1; //screen_changed
}
#endif
2006-03-20 17:12:09 +00:00
#ifdef DOOR_DEBUGGING
dump_door_debugging_info()
{
object *obj;
vms_vector new_pos;
fvi_query fq;
fvi_info hit_info;
int fate;
PHYSFS_file *dfile;
2006-03-20 17:12:09 +00:00
obj = &Objects[Players[Player_num].objnum];
vm_vec_scale_add(&new_pos,&obj->pos,&obj->orient.fvec,i2f(100));
fq.p0 = &obj->pos;
fq.startseg = obj->segnum;
fq.p1 = &new_pos;
fq.rad = 0;
fq.thisobjnum = Players[Player_num].objnum;
fq.ignore_obj_list = NULL;
fq.flags = 0;
fate = find_vector_intersection(&fq,&hit_info);
dfile = PHYSFSX_openWriteBuffered("door.out");
2006-03-20 17:12:09 +00:00
PHYSFSX_printf(dfile,"FVI hit_type = %d\n",fate);
PHYSFSX_printf(dfile," hit_seg = %d\n",hit_info.hit_seg);
PHYSFSX_printf(dfile," hit_side = %d\n",hit_info.hit_side);
PHYSFSX_printf(dfile," hit_side_seg = %d\n",hit_info.hit_side_seg);
PHYSFSX_printf(dfile,"\n");
2006-03-20 17:12:09 +00:00
if (fate == HIT_WALL) {
2014-09-21 22:11:51 +00:00
auto wall_num = Segments[hit_info.hit_seg].sides[hit_info.hit_side].wall_num;
PHYSFSX_printf(dfile,"wall_num = %d\n",wall_num);
2014-09-21 22:11:51 +00:00
if (wall_num != wall_none) {
2006-03-20 17:12:09 +00:00
wall *wall = &Walls[wall_num];
active_door *d;
PHYSFSX_printf(dfile," segnum = %d\n",wall->segnum);
PHYSFSX_printf(dfile," sidenum = %d\n",wall->sidenum);
PHYSFSX_printf(dfile," hps = %x\n",wall->hps);
PHYSFSX_printf(dfile," linked_wall = %d\n",wall->linked_wall);
PHYSFSX_printf(dfile," type = %d\n",wall->type);
PHYSFSX_printf(dfile," flags = %x\n",wall->flags);
PHYSFSX_printf(dfile," state = %d\n",wall->state);
PHYSFSX_printf(dfile," trigger = %d\n",wall->trigger);
PHYSFSX_printf(dfile," clip_num = %d\n",wall->clip_num);
PHYSFSX_printf(dfile," keys = %x\n",wall->keys);
PHYSFSX_printf(dfile," controlling_trigger = %d\n",wall->controlling_trigger);
PHYSFSX_printf(dfile," cloak_value = %d\n",wall->cloak_value);
PHYSFSX_printf(dfile,"\n");
2014-09-20 23:47:27 +00:00
range_for (auto &i, partial_range(ActiveDoors, Num_open_doors)) { //find door
d = &i;
2006-03-20 17:12:09 +00:00
if (d->front_wallnum[0]==wall-Walls || d->back_wallnum[0]==wall-Walls || (d->n_parts==2 && (d->front_wallnum[1]==wall-Walls || d->back_wallnum[1]==wall-Walls)))
break;
}
2006-03-20 17:12:09 +00:00
if (i>=Num_open_doors)
PHYSFSX_printf(dfile,"No active door.\n");
2006-03-20 17:12:09 +00:00
else {
PHYSFSX_printf(dfile,"Active door %d:\n",i);
PHYSFSX_printf(dfile," n_parts = %d\n",d->n_parts);
PHYSFSX_printf(dfile," front_wallnum = %d,%d\n",d->front_wallnum[0],d->front_wallnum[1]);
PHYSFSX_printf(dfile," back_wallnum = %d,%d\n",d->back_wallnum[0],d->back_wallnum[1]);
PHYSFSX_printf(dfile," time = %x\n",d->time);
2006-03-20 17:12:09 +00:00
}
2006-03-20 17:12:09 +00:00
}
}
PHYSFSX_printf(dfile,"\n");
PHYSFSX_printf(dfile,"\n");
2006-03-20 17:12:09 +00:00
PHYSFS_close(dfile);
2006-03-20 17:12:09 +00:00
}
#endif
//this is for system-level keys, such as help, etc.
//returns 1 if screen changed
2014-08-06 02:10:49 +00:00
static window_event_result HandleSystemKey(int key)
2006-03-20 17:12:09 +00:00
{
if (!Player_is_dead)
switch (key)
{
2006-03-20 17:12:09 +00:00
#ifdef DOOR_DEBUGGING
case KEY_LAPOSTRO+KEY_SHIFTED:
dump_door_debugging_info();
return 1;
2006-03-20 17:12:09 +00:00
#endif
case KEY_ESC:
{
int choice;
choice=nm_messagebox( NULL, 2, TXT_YES, TXT_NO, TXT_ABORT_GAME );
if (choice == 0)
2014-08-06 02:10:49 +00:00
{
window_close(Game_wind);
2014-08-06 02:10:49 +00:00
return window_event_result::close;
}
return window_event_result::handled;
}
#if defined(DXX_BUILD_DESCENT_II)
2006-03-20 17:12:09 +00:00
// fleshed these out because F1 and F2 aren't sequenctial keycodes on mac -- MWA
KEY_MAC(case KEY_COMMAND+KEY_SHIFTED+KEY_1:)
2006-03-20 17:12:09 +00:00
case KEY_SHIFTED+KEY_F1:
select_next_window_function(0);
2014-08-06 02:10:49 +00:00
return window_event_result::handled;
KEY_MAC(case KEY_COMMAND+KEY_SHIFTED+KEY_2:)
2006-03-20 17:12:09 +00:00
case KEY_SHIFTED+KEY_F2:
select_next_window_function(1);
2014-08-06 02:10:49 +00:00
return window_event_result::handled;
#endif
2006-03-20 17:12:09 +00:00
}
switch (key)
{
KEY_MAC( case KEY_COMMAND+KEY_P: )
case KEY_PAUSE:
do_game_pause(); break;
2006-03-20 17:12:09 +00:00
#ifdef macintosh
case KEY_COMMAND + KEY_SHIFTED + KEY_3:
#endif
case KEY_PRINT_SCREEN:
{
if (PlayerCfg.PRShot)
{
gr_set_current_canvas(NULL);
render_frame(0, 0);
gr_set_curfont(MEDIUM2_FONT);
gr_string(SWIDTH-FSPACX(92),SHEIGHT-LINE_SPACING,"DXX-Rebirth\n");
gr_flip();
save_screen_shot(0);
}
else
{
game_render_frame_mono(0);
2013-11-10 17:10:11 +00:00
if(!GameArg.DbgNoDoubleBuffer)
gr_flip();
save_screen_shot(0);
}
break;
}
2006-03-20 17:12:09 +00:00
KEY_MAC(case KEY_COMMAND+KEY_1:)
case KEY_F1: if (Game_mode & GM_MULTI) show_netgame_help(); else show_help(); break;
2006-03-20 17:12:09 +00:00
KEY_MAC(case KEY_COMMAND+KEY_2:)
case KEY_F2:
2006-03-20 17:12:09 +00:00
{
do_options_menu();
break;
}
KEY_MAC(case KEY_COMMAND+KEY_3:)
2006-03-20 17:12:09 +00:00
case KEY_F3:
if (!Player_is_dead && Viewer->type==OBJ_PLAYER) //if (!(Guided_missile[Player_num] && Guided_missile[Player_num]->type==OBJ_WEAPON && Guided_missile[Player_num]->id==GUIDEDMISS_ID && Guided_missile[Player_num]->signature==Guided_missile_sig[Player_num] && PlayerCfg.GuidedInBigWindow))
2006-03-20 17:12:09 +00:00
{
toggle_cockpit();
2006-03-20 17:12:09 +00:00
}
break;
KEY_MAC(case KEY_COMMAND+KEY_5:)
2006-03-20 17:12:09 +00:00
case KEY_F5:
if ( Newdemo_state == ND_STATE_RECORDING )
newdemo_stop_recording();
else if ( Newdemo_state == ND_STATE_NORMAL )
2007-10-14 16:25:55 +00:00
newdemo_start_recording();
2006-03-20 17:12:09 +00:00
break;
KEY_MAC(case KEY_COMMAND+KEY_ALTED+KEY_4:)
case KEY_ALTED + KEY_F4:
2006-03-20 17:12:09 +00:00
Show_reticle_name = (Show_reticle_name+1)%2;
break;
KEY_MAC(case KEY_COMMAND+KEY_7:)
2006-03-20 17:12:09 +00:00
case KEY_F7:
Show_kill_list = (Show_kill_list+1) % ((Game_mode & GM_TEAM) ? 4 : 3);
if (Game_mode & GM_MULTI)
multi_sort_kill_list();
break;
KEY_MAC(case KEY_COMMAND+KEY_8:)
2006-03-20 17:12:09 +00:00
case KEY_F8:
multi_send_message_start();
break;
case KEY_F9:
case KEY_F10:
case KEY_F11:
case KEY_F12:
multi_send_macro(key);
break; // send taunt macros
#if defined(__APPLE__) || defined(macintosh)
2006-03-20 17:12:09 +00:00
case KEY_9 + KEY_COMMAND:
multi_send_macro(KEY_F9);
break;
case KEY_0 + KEY_COMMAND:
multi_send_macro(KEY_F10);
break;
case KEY_1 + KEY_COMMAND + KEY_CTRLED:
multi_send_macro(KEY_F11);
break;
case KEY_2 + KEY_COMMAND + KEY_CTRLED:
multi_send_macro(KEY_F12);
break;
#endif
2006-03-20 17:12:09 +00:00
case KEY_SHIFTED + KEY_F9:
case KEY_SHIFTED + KEY_F10:
case KEY_SHIFTED + KEY_F11:
case KEY_SHIFTED + KEY_F12:
multi_define_macro(key);
break; // redefine taunt macros
#if defined(__APPLE__) || defined(macintosh)
2006-03-20 17:12:09 +00:00
case KEY_9 + KEY_SHIFTED + KEY_COMMAND:
multi_define_macro(KEY_F9);
break;
case KEY_0 + KEY_SHIFTED + KEY_COMMAND:
multi_define_macro(KEY_F10);
break;
case KEY_1 + KEY_SHIFTED + KEY_COMMAND + KEY_CTRLED:
multi_define_macro(KEY_F11);
break;
case KEY_2 + KEY_SHIFTED + KEY_COMMAND + KEY_CTRLED:
multi_define_macro(KEY_F12);
break;
#endif
2006-03-20 17:12:09 +00:00
KEY_MAC(case KEY_COMMAND+KEY_SHIFTED+KEY_S:)
KEY_MAC(case KEY_COMMAND+KEY_ALTED+KEY_2:)
2006-03-20 17:12:09 +00:00
case KEY_ALTED+KEY_F2:
if (!Player_is_dead)
state_save_all(0, NULL, 0); // 0 means not between levels.
2006-03-20 17:12:09 +00:00
break;
KEY_MAC(case KEY_COMMAND+KEY_S:)
case KEY_ALTED+KEY_SHIFTED+KEY_F2:
if (!Player_is_dead)
state_save_all(0, NULL, 1);
break;
KEY_MAC(case KEY_COMMAND+KEY_SHIFTED+KEY_O:)
KEY_MAC(case KEY_COMMAND+KEY_ALTED+KEY_3:)
2006-03-20 17:12:09 +00:00
case KEY_ALTED+KEY_F3:
if (!((Game_mode & GM_MULTI) && !(Game_mode & GM_MULTI_COOP)))
state_restore_all(1, 0, NULL, 0);
break;
KEY_MAC(case KEY_COMMAND+KEY_O:)
case KEY_ALTED+KEY_SHIFTED+KEY_F3:
if (!((Game_mode & GM_MULTI) && !(Game_mode & GM_MULTI_COOP)))
state_restore_all(1, 0, NULL, 1);
2006-03-20 17:12:09 +00:00
break;
#if defined(DXX_BUILD_DESCENT_II)
KEY_MAC(case KEY_COMMAND+KEY_SHIFTED+KEY_4:)
2006-03-20 17:12:09 +00:00
case KEY_F4 + KEY_SHIFTED:
do_escort_menu();
break;
KEY_MAC(case KEY_COMMAND+KEY_SHIFTED+KEY_ALTED+KEY_4:)
2006-03-20 17:12:09 +00:00
case KEY_F4 + KEY_SHIFTED + KEY_ALTED:
change_guidebot_name();
break;
#endif
2006-03-20 17:12:09 +00:00
/*
* Jukebox hotkeys -- MD2211, 2007
* Now for all music
* ==============================================
*/
case KEY_ALTED + KEY_SHIFTED + KEY_F9:
KEY_MAC(case KEY_COMMAND+KEY_E:)
if (GameCfg.MusicType == MUSIC_TYPE_REDBOOK)
{
songs_stop_all();
RBAEjectDisk();
}
break;
case KEY_ALTED + KEY_SHIFTED + KEY_F10:
KEY_MAC(case KEY_COMMAND+KEY_UP:)
KEY_MAC(case KEY_COMMAND+KEY_DOWN:)
songs_pause_resume();
break;
case KEY_MINUS + KEY_ALTED:
case KEY_ALTED + KEY_SHIFTED + KEY_F11:
KEY_MAC(case KEY_COMMAND+KEY_LEFT:)
songs_play_level_song( Current_level_num, -1 );
2006-03-20 17:12:09 +00:00
break;
case KEY_EQUAL + KEY_ALTED:
case KEY_ALTED + KEY_SHIFTED + KEY_F12:
KEY_MAC(case KEY_COMMAND+KEY_RIGHT:)
songs_play_level_song( Current_level_num, 1 );
2006-03-20 17:12:09 +00:00
break;
2006-03-20 17:12:09 +00:00
default:
2014-08-06 02:10:49 +00:00
return window_event_result::ignored;
}
2014-08-06 02:10:49 +00:00
return window_event_result::handled;
2006-03-20 17:12:09 +00:00
}
2014-08-06 02:10:49 +00:00
static window_event_result HandleGameKey(int key)
2006-03-20 17:12:09 +00:00
{
switch (key) {
#if defined(DXX_BUILD_DESCENT_II)
2006-03-20 17:12:09 +00:00
case KEY_1 + KEY_SHIFTED:
case KEY_2 + KEY_SHIFTED:
case KEY_3 + KEY_SHIFTED:
case KEY_4 + KEY_SHIFTED:
case KEY_5 + KEY_SHIFTED:
case KEY_6 + KEY_SHIFTED:
case KEY_7 + KEY_SHIFTED:
case KEY_8 + KEY_SHIFTED:
case KEY_9 + KEY_SHIFTED:
case KEY_0 + KEY_SHIFTED:
if (PlayerCfg.EscortHotKeys)
{
if (!(Game_mode & GM_MULTI))
set_escort_special_goal(key);
else
HUD_init_message_literal(HM_DEFAULT, "No Guide-Bot in Multiplayer!");
game_flush_inputs();
2014-08-06 02:10:49 +00:00
return window_event_result::handled;
}
else
2014-08-06 02:10:49 +00:00
return window_event_result::ignored;
#endif
2006-03-20 17:12:09 +00:00
case KEY_ALTED+KEY_F7:
KEY_MAC(case KEY_COMMAND+KEY_ALTED+KEY_7:)
PlayerCfg.HudMode=(PlayerCfg.HudMode+1)%GAUGE_HUD_NUMMODES;
write_player_file();
switch (PlayerCfg.HudMode)
{
case 0: HUD_init_message_literal(HM_DEFAULT, "Standard HUD"); break;
case 1: HUD_init_message_literal(HM_DEFAULT, "Alternative HUD #1"); break;
case 2: HUD_init_message_literal(HM_DEFAULT, "Alternative HUD #2"); break;
case 3: HUD_init_message_literal(HM_DEFAULT, "No HUD"); break;
}
2014-08-06 02:10:49 +00:00
return window_event_result::handled;
2006-03-20 17:12:09 +00:00
KEY_MAC(case KEY_COMMAND+KEY_6:)
2006-03-20 17:12:09 +00:00
case KEY_F6:
if (Netgame.RefusePlayers && WaitForRefuseAnswer && !(Game_mode & GM_TEAM))
{
RefuseThisPlayer=1;
HUD_init_message_literal(HM_MULTI, "Player accepted!");
}
2014-08-06 02:10:49 +00:00
return window_event_result::handled;
2006-03-20 17:12:09 +00:00
case KEY_ALTED + KEY_1:
if (Netgame.RefusePlayers && WaitForRefuseAnswer && (Game_mode & GM_TEAM))
{
RefuseThisPlayer=1;
HUD_init_message_literal(HM_MULTI, "Player accepted!");
2006-03-20 17:12:09 +00:00
RefuseTeam=1;
game_flush_inputs();
2006-03-20 17:12:09 +00:00
}
2014-08-06 02:10:49 +00:00
return window_event_result::handled;
2006-03-20 17:12:09 +00:00
case KEY_ALTED + KEY_2:
if (Netgame.RefusePlayers && WaitForRefuseAnswer && (Game_mode & GM_TEAM))
{
RefuseThisPlayer=1;
HUD_init_message_literal(HM_MULTI, "Player accepted!");
2006-03-20 17:12:09 +00:00
RefuseTeam=2;
game_flush_inputs();
2006-03-20 17:12:09 +00:00
}
2014-08-06 02:10:49 +00:00
return window_event_result::handled;
2006-03-20 17:12:09 +00:00
default:
#if defined(DXX_BUILD_DESCENT_I)
2014-08-06 02:10:49 +00:00
return window_event_result::ignored;
#endif
2006-03-20 17:12:09 +00:00
break;
} //switch (key)
#if defined(DXX_BUILD_DESCENT_II)
if (!Player_is_dead)
switch (key)
{
KEY_MAC(case KEY_COMMAND+KEY_SHIFTED+KEY_5:)
case KEY_F5 + KEY_SHIFTED:
DropCurrentWeapon();
break;
KEY_MAC(case KEY_COMMAND+KEY_SHIFTED+KEY_6:)
case KEY_F6 + KEY_SHIFTED:
DropSecondaryWeapon();
break;
case KEY_0 + KEY_ALTED:
DropFlag ();
game_flush_inputs();
break;
KEY_MAC(case KEY_COMMAND+KEY_4:)
case KEY_F4:
if (!DefiningMarkerMessage)
InitMarkerInput();
break;
default:
2014-08-06 02:10:49 +00:00
return window_event_result::ignored;
}
else
2014-08-06 02:10:49 +00:00
return window_event_result::ignored;
#endif
2014-08-06 02:10:49 +00:00
return window_event_result::handled;
2006-03-20 17:12:09 +00:00
}
#if defined(DXX_BUILD_DESCENT_II)
2013-09-22 22:26:27 +00:00
static void kill_all_robots(void)
2006-03-20 17:12:09 +00:00
{
int dead_count=0;
2006-03-20 17:12:09 +00:00
//int boss_index = -1;
// Kill all bots except for Buddy bot and boss. However, if only boss and buddy left, kill boss.
2014-10-12 23:05:46 +00:00
range_for (auto i, highest_valid(Objects))
2006-03-20 17:12:09 +00:00
if (Objects[i].type == OBJ_ROBOT) {
if (!Robot_info[get_robot_id(&Objects[i])].companion && !Robot_info[get_robot_id(&Objects[i])].boss_flag) {
2006-03-20 17:12:09 +00:00
dead_count++;
Objects[i].flags |= OF_EXPLODING|OF_SHOULD_BE_DEAD;
}
}
// -- // Now, if more than boss and buddy left, un-kill boss.
// -- if ((dead_count > 2) && (boss_index != -1)) {
// -- Objects[boss_index].flags &= ~(OF_EXPLODING|OF_SHOULD_BE_DEAD);
// -- dead_count--;
// -- } else if (boss_index != -1)
// -- HUD_init_message(HM_DEFAULT, "Toasted the BOSS!");
2006-03-20 17:12:09 +00:00
// Toast the buddy if nothing else toasted!
if (dead_count == 0)
2014-10-12 23:05:46 +00:00
range_for (auto i, highest_valid(Objects))
2006-03-20 17:12:09 +00:00
if (Objects[i].type == OBJ_ROBOT)
if (Robot_info[get_robot_id(&Objects[i])].companion) {
2006-03-20 17:12:09 +00:00
Objects[i].flags |= OF_EXPLODING|OF_SHOULD_BE_DEAD;
HUD_init_message_literal(HM_DEFAULT, "Toasted the Buddy! *sniff*");
2006-03-20 17:12:09 +00:00
dead_count++;
}
HUD_init_message(HM_DEFAULT, "%i robots toasted!", dead_count);
2006-03-20 17:12:09 +00:00
}
#endif
2006-03-20 17:12:09 +00:00
// --------------------------------------------------------------------------
// Detonate reactor.
// Award player all powerups in mine.
// Place player just outside exit.
// Kill all bots in mine.
// Yippee!!
2013-09-22 22:26:27 +00:00
static void kill_and_so_forth(void)
2006-03-20 17:12:09 +00:00
{
HUD_init_message_literal(HM_DEFAULT, "Killing, awarding, etc.!");
2006-03-20 17:12:09 +00:00
2014-10-12 23:05:46 +00:00
range_for (auto i, highest_valid(Objects))
{
2006-03-20 17:12:09 +00:00
switch (Objects[i].type) {
case OBJ_ROBOT:
Objects[i].flags |= OF_EXPLODING|OF_SHOULD_BE_DEAD;
break;
case OBJ_POWERUP:
do_powerup(&Objects[i]);
break;
}
}
do_controlcen_destroyed_stuff(object_none);
2006-03-20 17:12:09 +00:00
2014-09-20 23:47:27 +00:00
for (uint_fast32_t i = 0; i < Num_triggers; i++) {
if (trigger_is_exit(&Triggers[i])) {
2014-04-27 23:12:34 +00:00
range_for (auto &w, partial_range(Walls, Num_walls))
{
if (w.trigger == i) {
compute_segment_center(&ConsoleObject->pos, &Segments[w.segnum]);
obj_relink(ConsoleObject-Objects,w.segnum);
2006-03-20 17:12:09 +00:00
goto kasf_done;
}
}
}
}
kasf_done: ;
}
#ifndef RELEASE
#if defined(DXX_BUILD_DESCENT_II)
static void kill_all_snipers(void) __attribute_used;
static void kill_all_snipers(void)
2006-03-20 17:12:09 +00:00
{
int dead_count=0;
2006-03-20 17:12:09 +00:00
// Kill all snipers.
2014-10-12 23:05:46 +00:00
range_for (auto i, highest_valid(Objects))
2006-03-20 17:12:09 +00:00
if (Objects[i].type == OBJ_ROBOT)
if (Objects[i].ctype.ai_info.behavior == AIB_SNIPE) {
dead_count++;
Objects[i].flags |= OF_EXPLODING|OF_SHOULD_BE_DEAD;
}
HUD_init_message(HM_DEFAULT, "%i robots toasted!", dead_count);
2006-03-20 17:12:09 +00:00
}
static void kill_thief(void) __attribute_used;
static void kill_thief(void)
2006-03-20 17:12:09 +00:00
{
// Kill thief.
2014-10-12 23:05:46 +00:00
range_for (auto i, highest_valid(Objects))
2006-03-20 17:12:09 +00:00
if (Objects[i].type == OBJ_ROBOT)
if (Robot_info[get_robot_id(&Objects[i])].thief) {
2006-03-20 17:12:09 +00:00
Objects[i].flags |= OF_EXPLODING|OF_SHOULD_BE_DEAD;
HUD_init_message_literal(HM_DEFAULT, "Thief toasted!");
2006-03-20 17:12:09 +00:00
}
}
static void kill_buddy(void) __attribute_used;
static void kill_buddy(void)
2006-03-20 17:12:09 +00:00
{
// Kill buddy.
2014-10-12 23:05:46 +00:00
range_for (auto i, highest_valid(Objects))
2006-03-20 17:12:09 +00:00
if (Objects[i].type == OBJ_ROBOT)
if (Robot_info[get_robot_id(&Objects[i])].companion) {
2006-03-20 17:12:09 +00:00
Objects[i].flags |= OF_EXPLODING|OF_SHOULD_BE_DEAD;
HUD_init_message_literal(HM_DEFAULT, "Buddy toasted!");
2006-03-20 17:12:09 +00:00
}
}
#endif
2006-03-20 17:12:09 +00:00
2014-08-06 02:10:49 +00:00
static window_event_result HandleTestKey(int key)
2006-03-20 17:12:09 +00:00
{
switch (key)
{
2006-03-20 17:12:09 +00:00
#ifdef SHOW_EXIT_PATH
case KEY_DEBUGGED+KEY_1: create_special_path(); break;
#endif
2006-03-20 17:12:09 +00:00
case KEY_DEBUGGED+KEY_Y:
do_controlcen_destroyed_stuff(object_none);
2006-03-20 17:12:09 +00:00
break;
#if defined(DXX_BUILD_DESCENT_II)
2006-03-20 17:12:09 +00:00
case KEY_DEBUGGED+KEY_ALTED+KEY_D:
PlayerCfg.NetlifeKills=4000; PlayerCfg.NetlifeKilled=5;
2006-03-20 17:12:09 +00:00
multi_add_lifetime_kills();
break;
case KEY_DEBUGGED+KEY_R+KEY_SHIFTED:
kill_all_robots();
break;
#endif
2006-03-20 17:12:09 +00:00
case KEY_BACKSP:
case KEY_CTRLED+KEY_BACKSP:
case KEY_ALTED+KEY_BACKSP:
case KEY_SHIFTED+KEY_BACKSP:
case KEY_SHIFTED+KEY_ALTED+KEY_BACKSP:
case KEY_CTRLED+KEY_ALTED+KEY_BACKSP:
case KEY_SHIFTED+KEY_CTRLED+KEY_BACKSP:
case KEY_SHIFTED+KEY_CTRLED+KEY_ALTED+KEY_BACKSP:
Int3(); break;
2006-03-20 17:12:09 +00:00
case KEY_DEBUGGED+KEY_S: digi_reset(); break;
case KEY_DEBUGGED+KEY_P:
if (Game_suspended & SUSP_ROBOTS)
Game_suspended &= ~SUSP_ROBOTS; //robots move
else
Game_suspended |= SUSP_ROBOTS; //robots don't move
break;
case KEY_DEBUGGED+KEY_K: Players[Player_num].shields = 1; break; // a virtual kill
case KEY_DEBUGGED+KEY_SHIFTED + KEY_K: Players[Player_num].shields = -1; break; // an actual kill
case KEY_DEBUGGED+KEY_X: Players[Player_num].lives++; break; // Extra life cheat key.
case KEY_DEBUGGED+KEY_H:
if (Player_is_dead)
2014-08-06 02:10:49 +00:00
return window_event_result::ignored;
Players[Player_num].flags ^= PLAYER_FLAGS_CLOAKED;
if (Players[Player_num].flags & PLAYER_FLAGS_CLOAKED) {
if (Game_mode & GM_MULTI)
multi_send_cloak();
ai_do_cloak_stuff();
Players[Player_num].cloak_time = GameTime64;
}
2006-03-20 17:12:09 +00:00
break;
case KEY_DEBUGGED+KEY_R:
cheats.robotfiringsuspended = !cheats.robotfiringsuspended;
2006-03-20 17:12:09 +00:00
break;
#ifdef EDITOR //editor-specific functions
2006-03-20 17:12:09 +00:00
case KEY_E + KEY_DEBUGGED:
window_set_visible(Game_wind, 0); // don't let the game do anything while we set the editor up
init_editor();
window_close(Game_wind);
2014-08-06 02:10:49 +00:00
return window_event_result::close;
#if defined(DXX_BUILD_DESCENT_II)
2006-03-20 17:12:09 +00:00
case KEY_Q + KEY_SHIFTED + KEY_DEBUGGED:
{
2013-01-06 21:11:53 +00:00
palette_array_t save_pal;
save_pal = gr_palette;
2013-10-27 21:01:04 +00:00
PlayMovie ("end.tex", "end.mve",MOVIE_ABORT_ON);
2006-03-20 17:12:09 +00:00
Screen_mode = -1;
set_screen_mode(SCREEN_GAME);
reset_cockpit();
2013-01-06 21:11:53 +00:00
gr_palette = save_pal;
2006-03-20 17:12:09 +00:00
gr_palette_load(gr_palette);
break;
}
#endif
2006-03-20 17:12:09 +00:00
case KEY_C + KEY_SHIFTED + KEY_DEBUGGED:
if (!Player_is_dead && !( Game_mode & GM_MULTI ))
2006-03-20 17:12:09 +00:00
move_player_2_segment(Cursegp,Curside);
break; //move eye to curseg
case KEY_DEBUGGED+KEY_W: draw_world_from_game(); break;
#endif //#ifdef EDITOR
case KEY_DEBUGGED+KEY_LAPOSTRO: Show_view_text_timer = 0x30000; object_goto_next_viewer(); break;
case KEY_DEBUGGED+KEY_SHIFTED+KEY_LAPOSTRO: Viewer=ConsoleObject; break;
case KEY_DEBUGGED+KEY_O: toggle_outline_mode(); break;
case KEY_DEBUGGED+KEY_T:
#if defined(DXX_BUILD_DESCENT_II)
2013-09-08 04:25:05 +00:00
{
static int Toggle_var;
Toggle_var = !Toggle_var;
if (Toggle_var)
GameArg.SysMaxFPS = 300;
else
GameArg.SysMaxFPS = 30;
2013-09-08 04:25:05 +00:00
}
#endif
2006-03-20 17:12:09 +00:00
break;
case KEY_DEBUGGED + KEY_L:
if (++Lighting_on >= 2) Lighting_on = 0; break;
case KEY_PAD5: slew_stop(); break;
#ifndef NDEBUG
case KEY_DEBUGGED + KEY_F11: play_test_sound(); break;
case KEY_DEBUGGED + KEY_SHIFTED+KEY_F11: advance_sound(); play_test_sound(); break;
#endif
case KEY_DEBUGGED + KEY_M:
Debug_spew = !Debug_spew;
if (Debug_spew) {
HUD_init_message_literal(HM_DEFAULT, "Debug Spew: ON" );
2006-03-20 17:12:09 +00:00
} else {
HUD_init_message_literal(HM_DEFAULT, "Debug Spew: OFF" );
2006-03-20 17:12:09 +00:00
}
break;
case KEY_DEBUGGED + KEY_C:
do_cheat_menu();
break;
case KEY_DEBUGGED + KEY_SHIFTED + KEY_A:
do_megawow_powerup(10);
break;
case KEY_DEBUGGED + KEY_A: {
do_megawow_powerup(200);
break;
2006-03-20 17:12:09 +00:00
}
case KEY_DEBUGGED+KEY_SPACEBAR: //KEY_F7: // Toggle physics flying
slew_stop();
game_flush_inputs();
if ( ConsoleObject->control_type != CT_FLYING ) {
fly_init(ConsoleObject);
Game_suspended &= ~SUSP_ROBOTS; //robots move
} else {
slew_init(ConsoleObject); //start player slewing
Game_suspended |= SUSP_ROBOTS; //robots don't move
}
break;
case KEY_DEBUGGED+KEY_COMMA: Render_zoom = fixmul(Render_zoom,62259); break;
case KEY_DEBUGGED+KEY_PERIOD: Render_zoom = fixmul(Render_zoom,68985); break;
#ifndef NDEBUG
case KEY_DEBUGGED+KEY_D:
2013-11-10 17:10:11 +00:00
if ((GameArg.DbgNoDoubleBuffer = !GameArg.DbgNoDoubleBuffer)!=0)
2006-03-20 17:12:09 +00:00
init_cockpit();
break;
#endif
#ifdef EDITOR
2006-03-20 17:12:09 +00:00
case KEY_DEBUGGED+KEY_Q:
stop_time();
dump_used_textures_all();
start_time();
break;
#endif
2006-03-20 17:12:09 +00:00
case KEY_DEBUGGED+KEY_B: {
newmenu_item m;
2014-07-23 02:27:22 +00:00
d_fname text{};
2006-03-20 17:12:09 +00:00
int item;
2014-07-26 22:45:01 +00:00
nm_set_item_input(&m, text.size(), &text[0u]);
item = newmenu_do( NULL, "Briefing to play?", 1, &m, unused_newmenu_subfunction, unused_newmenu_userdata);
2006-03-20 17:12:09 +00:00
if (item != -1) {
do_briefing_screens(text,1);
}
break;
}
case KEY_DEBUGGED+KEY_SHIFTED+KEY_B:
if (Player_is_dead)
2014-08-06 02:10:49 +00:00
return window_event_result::ignored;
2006-03-20 17:12:09 +00:00
kill_and_so_forth();
break;
case KEY_DEBUGGED+KEY_G:
GameTime64 = (0x7fffffffffffffffLL) - (F1_0*10);
HUD_init_message(HM_DEFAULT, "GameTime %" PRIi64 " - Reset in 10 seconds!", GameTime64);
break;
default:
2014-08-06 02:10:49 +00:00
return window_event_result::ignored;
2006-03-20 17:12:09 +00:00
}
2014-08-06 02:10:49 +00:00
return window_event_result::handled;
2006-03-20 17:12:09 +00:00
}
#endif //#ifndef RELEASE
#define CHEAT_MAX_LEN 15
struct cheat_code
2006-03-20 17:12:09 +00:00
{
2013-10-05 21:05:09 +00:00
const char string[CHEAT_MAX_LEN];
2013-11-28 02:16:05 +00:00
int (game_cheats::*stateptr);
2014-06-26 02:18:48 +00:00
};
static const cheat_code cheat_codes[] = {
#if defined(DXX_BUILD_DESCENT_I)
2013-11-28 02:16:05 +00:00
{ "gabbagabbahey", &game_cheats::enabled },
{ "scourge", &game_cheats::wowie },
{ "bigred", &game_cheats::wowie2 },
{ "mitzi", &game_cheats::allkeys },
{ "racerx", &game_cheats::invul },
{ "guile", &game_cheats::cloak },
{ "twilight", &game_cheats::shields },
{ "poboys", &game_cheats::killreactor },
{ "farmerjoe", &game_cheats::levelwarp },
{ "bruin", &game_cheats::extralife },
{ "porgys", &game_cheats::rapidfire },
{ "ahimsa", &game_cheats::robotfiringsuspended },
#elif defined(DXX_BUILD_DESCENT_II)
2013-11-28 02:16:05 +00:00
{ "gabbagabbahey", &game_cheats::lamer },
{ "motherlode", &game_cheats::lamer },
{ "currygoat", &game_cheats::lamer },
{ "zingermans", &game_cheats::lamer },
{ "eatangelos", &game_cheats::lamer },
{ "ericaanne", &game_cheats::lamer },
{ "joshuaakira", &game_cheats::lamer },
{ "whammazoom", &game_cheats::lamer },
{ "honestbob", &game_cheats::wowie },
{ "algroove", &game_cheats::allkeys },
{ "alifalafel", &game_cheats::accessory },
{ "almighty", &game_cheats::invul },
{ "blueorb", &game_cheats::shields },
{ "delshiftb", &game_cheats::killreactor },
{ "freespace", &game_cheats::levelwarp },
{ "rockrgrl", &game_cheats::fullautomap },
{ "wildfire", &game_cheats::rapidfire },
{ "duddaboo", &game_cheats::bouncyfire },
{ "imagespace", &game_cheats::robotfiringsuspended },
{ "spaniard", &game_cheats::killallrobots },
{ "silkwing", &game_cheats::robotskillrobots },
{ "godzilla", &game_cheats::monsterdamage },
{ "helpvishnu", &game_cheats::buddyclone },
{ "gowingnut", &game_cheats::buddyangry },
#endif
2013-11-28 02:16:05 +00:00
{ "flash", &game_cheats::exitpath },
{ "astral", &game_cheats::ghostphysics },
{ "buggin", &game_cheats::turbo },
{ "bittersweet", &game_cheats::acid },
};
2014-08-06 02:10:49 +00:00
static window_event_result FinalCheats(int key)
{
2013-10-03 02:33:04 +00:00
static char cheat_buffer[CHEAT_MAX_LEN];
2013-11-28 02:16:05 +00:00
int (game_cheats::*gotcha);
2006-03-20 17:12:09 +00:00
if (Game_mode & GM_MULTI)
2014-08-06 02:10:49 +00:00
return window_event_result::ignored;
2008-11-18 14:34:52 +00:00
for (unsigned i = 1; i < CHEAT_MAX_LEN; i++)
cheat_buffer[i-1] = cheat_buffer[i];
cheat_buffer[CHEAT_MAX_LEN-1] = key_ascii();
for (unsigned i = 0;; i++)
{
if (i >= sizeof(cheat_codes) / sizeof(cheat_codes[0]))
2014-08-06 02:10:49 +00:00
return window_event_result::ignored;
int cheatlen = strlen(cheat_codes[i].string);
Assert(cheatlen <= CHEAT_MAX_LEN);
if (d_strnicmp(cheat_codes[i].string, cheat_buffer+CHEAT_MAX_LEN-cheatlen, cheatlen)==0)
{
2013-11-28 02:16:05 +00:00
gotcha = cheat_codes[i].stateptr;
#if defined(DXX_BUILD_DESCENT_I)
2013-11-28 02:16:05 +00:00
if (!cheats.enabled && cheats.*gotcha != cheats.enabled)
2014-08-06 02:10:49 +00:00
return window_event_result::ignored;
if (!cheats.enabled)
HUD_init_message_literal(HM_DEFAULT, TXT_CHEATS_ENABLED);
#endif
2013-11-28 02:16:05 +00:00
cheats.*gotcha = !(cheats.*gotcha);
cheats.enabled = 1;
digi_play_sample( SOUND_CHEATER, F1_0);
Players[Player_num].score = 0;
break;
}
}
2006-03-20 17:12:09 +00:00
#if defined(DXX_BUILD_DESCENT_I)
2013-11-28 02:16:05 +00:00
if (gotcha == &game_cheats::wowie)
{
HUD_init_message_literal(HM_DEFAULT, TXT_WOWIE_ZOWIE);
2013-10-26 19:52:41 +00:00
Players[Player_num].primary_weapon_flags |= (HAS_LASER_FLAG | HAS_VULCAN_FLAG | HAS_SPREADFIRE_FLAG);
2013-10-26 19:52:41 +00:00
Players[Player_num].secondary_weapon_flags |= (HAS_CONCUSSION_FLAG | HAS_HOMING_FLAG | HAS_PROXIMITY_BOMB_FLAG);
2013-09-20 23:12:54 +00:00
Players[Player_num].vulcan_ammo = Primary_ammo_max[VULCAN_INDEX];
for (unsigned i=0; i<3; i++)
Players[Player_num].secondary_ammo[i] = Secondary_ammo_max[i];
if (Newdemo_state == ND_STATE_RECORDING)
newdemo_record_laser_level(Players[Player_num].laser_level, MAX_LASER_LEVEL);
Players[Player_num].energy = MAX_ENERGY;
Players[Player_num].laser_level = MAX_LASER_LEVEL;
Players[Player_num].flags |= PLAYER_FLAGS_QUAD_LASERS;
update_laser_weapon_info();
}
2013-11-28 02:16:05 +00:00
if (gotcha == &game_cheats::wowie2)
{
HUD_init_message(HM_DEFAULT, "SUPER %s",TXT_WOWIE_ZOWIE);
2013-10-26 19:52:41 +00:00
Players[Player_num].primary_weapon_flags |= (HAS_LASER_FLAG | HAS_VULCAN_FLAG | HAS_SPREADFIRE_FLAG | HAS_PLASMA_FLAG | HAS_FUSION_FLAG);
2013-10-26 19:52:41 +00:00
Players[Player_num].secondary_weapon_flags |= (HAS_CONCUSSION_FLAG | HAS_HOMING_FLAG | HAS_PROXIMITY_BOMB_FLAG | HAS_SMART_FLAG | HAS_MEGA_FLAG);
2013-09-20 23:12:54 +00:00
Players[Player_num].vulcan_ammo = Primary_ammo_max[VULCAN_INDEX];
for (unsigned i=0; i<MAX_SECONDARY_WEAPONS; i++)
Players[Player_num].secondary_ammo[i] = Secondary_ammo_max[i];
if (Newdemo_state == ND_STATE_RECORDING)
newdemo_record_laser_level(Players[Player_num].laser_level, MAX_LASER_LEVEL);
Players[Player_num].energy = MAX_ENERGY;
Players[Player_num].laser_level = MAX_LASER_LEVEL;
Players[Player_num].flags |= PLAYER_FLAGS_QUAD_LASERS;
update_laser_weapon_info();
}
#elif defined(DXX_BUILD_DESCENT_II)
2013-11-28 02:16:05 +00:00
if (gotcha == &game_cheats::lamer)
{
Players[Player_num].shields=Players[Player_num].energy=i2f(1);
HUD_init_message_literal(HM_DEFAULT, "Take that...cheater!");
}
2006-03-20 17:12:09 +00:00
2013-11-28 02:16:05 +00:00
if (gotcha == &game_cheats::wowie)
{
HUD_init_message_literal(HM_DEFAULT, TXT_WOWIE_ZOWIE);
2006-03-20 17:12:09 +00:00
if (Piggy_hamfile_version < 3) // SHAREWARE
{
2013-10-26 19:52:41 +00:00
Players[Player_num].primary_weapon_flags |= (HAS_LASER_FLAG | HAS_VULCAN_FLAG | HAS_SPREADFIRE_FLAG | HAS_PLASMA_FLAG) | (HAS_GAUSS_FLAG | HAS_HELIX_FLAG);
2013-10-26 19:52:41 +00:00
Players[Player_num].secondary_weapon_flags |= (HAS_CONCUSSION_FLAG | HAS_HOMING_FLAG | HAS_PROXIMITY_BOMB_FLAG | HAS_SMART_FLAG) | (HAS_FLASH_FLAG | HAS_GUIDED_FLAG | HAS_SMART_BOMB_FLAG);
2006-03-20 17:12:09 +00:00
}
else
2006-03-20 17:12:09 +00:00
{
2013-10-26 19:52:41 +00:00
Players[Player_num].primary_weapon_flags |= (HAS_LASER_FLAG | HAS_VULCAN_FLAG | HAS_SPREADFIRE_FLAG | HAS_PLASMA_FLAG | HAS_FUSION_FLAG) | (HAS_GAUSS_FLAG | HAS_HELIX_FLAG | HAS_PHOENIX_FLAG | HAS_OMEGA_FLAG);
2013-10-26 19:52:41 +00:00
Players[Player_num].secondary_weapon_flags |= (HAS_CONCUSSION_FLAG | HAS_HOMING_FLAG | HAS_PROXIMITY_BOMB_FLAG | HAS_SMART_FLAG | HAS_MEGA_FLAG) | (HAS_FLASH_FLAG | HAS_GUIDED_FLAG | HAS_SMART_BOMB_FLAG | HAS_MERCURY_FLAG | HAS_EARTHSHAKER_FLAG);
2006-03-20 17:12:09 +00:00
}
2013-09-20 23:12:54 +00:00
Players[Player_num].vulcan_ammo = Primary_ammo_max[VULCAN_INDEX];
for (unsigned i=0; i<MAX_SECONDARY_WEAPONS; i++)
Players[Player_num].secondary_ammo[i] = Secondary_ammo_max[i];
if (Piggy_hamfile_version < 3) // SHAREWARE
2006-03-20 17:12:09 +00:00
{
Players[Player_num].secondary_ammo[SMISSILE4_INDEX] = 0;
Players[Player_num].secondary_ammo[SMISSILE5_INDEX] = 0;
Players[Player_num].secondary_ammo[MEGA_INDEX] = 0;
2006-03-20 17:12:09 +00:00
}
if (Newdemo_state == ND_STATE_RECORDING)
newdemo_record_laser_level(Players[Player_num].laser_level, MAX_LASER_LEVEL);
Players[Player_num].energy = MAX_ENERGY;
Players[Player_num].laser_level = MAX_SUPER_LASER_LEVEL;
Players[Player_num].flags |= PLAYER_FLAGS_QUAD_LASERS;
update_laser_weapon_info();
}
2013-11-28 02:16:05 +00:00
if (gotcha == &game_cheats::accessory)
2006-03-20 17:12:09 +00:00
{
Players[Player_num].flags |=PLAYER_FLAGS_HEADLIGHT;
Players[Player_num].flags |=PLAYER_FLAGS_AFTERBURNER;
Players[Player_num].flags |=PLAYER_FLAGS_AMMO_RACK;
Players[Player_num].flags |=PLAYER_FLAGS_CONVERTER;
HUD_init_message_literal(HM_DEFAULT, "Accessories!!");
2006-03-20 17:12:09 +00:00
}
#endif
2013-11-28 02:16:05 +00:00
if (gotcha == &game_cheats::allkeys)
{
HUD_init_message_literal(HM_DEFAULT, TXT_ALL_KEYS);
Players[Player_num].flags |= PLAYER_FLAGS_BLUE_KEY | PLAYER_FLAGS_RED_KEY | PLAYER_FLAGS_GOLD_KEY;
}
2006-03-20 17:12:09 +00:00
2013-11-28 02:16:05 +00:00
if (gotcha == &game_cheats::invul)
{
Players[Player_num].flags ^= PLAYER_FLAGS_INVULNERABLE;
HUD_init_message(HM_DEFAULT, "%s %s!", TXT_INVULNERABILITY, (Players[Player_num].flags&PLAYER_FLAGS_INVULNERABLE)?TXT_ON:TXT_OFF);
Players[Player_num].invulnerable_time = GameTime64+i2f(1000);
}
2013-11-28 02:16:05 +00:00
if (gotcha == &game_cheats::shields)
{
HUD_init_message_literal(HM_DEFAULT, TXT_FULL_SHIELDS);
Players[Player_num].shields = MAX_SHIELDS;
}
#if defined(DXX_BUILD_DESCENT_I)
2013-11-28 02:16:05 +00:00
if (gotcha == &game_cheats::cloak)
{
Players[Player_num].flags ^= PLAYER_FLAGS_CLOAKED;
HUD_init_message(HM_DEFAULT, "%s %s!", TXT_CLOAK, (Players[Player_num].flags&PLAYER_FLAGS_CLOAKED)?TXT_ON:TXT_OFF);
if (Players[Player_num].flags & PLAYER_FLAGS_CLOAKED)
{
ai_do_cloak_stuff();
Players[Player_num].cloak_time = GameTime64;
}
}
2013-11-28 02:16:05 +00:00
if (gotcha == &game_cheats::extralife)
{
if (Players[Player_num].lives<50)
{
Players[Player_num].lives++;
HUD_init_message_literal(HM_DEFAULT, "Extra life!");
}
}
#endif
2013-11-28 02:16:05 +00:00
if (gotcha == &game_cheats::killreactor)
{
kill_and_so_forth();
}
2013-11-28 02:16:05 +00:00
if (gotcha == &game_cheats::exitpath)
{
if (create_special_path())
HUD_init_message_literal(HM_DEFAULT, "Exit path illuminated!");
}
2013-11-28 02:16:05 +00:00
if (gotcha == &game_cheats::levelwarp)
{
2006-03-20 17:12:09 +00:00
newmenu_item m;
char text[10]="";
int new_level_num;
int item;
nm_set_item_input(&m, 10, text);
item = newmenu_do( NULL, TXT_WARP_TO_LEVEL, 1, &m, unused_newmenu_subfunction, unused_newmenu_userdata);
2006-03-20 17:12:09 +00:00
if (item != -1) {
new_level_num = atoi(m.text);
if (new_level_num!=0 && new_level_num>=0 && new_level_num<=Last_level) {
window_set_visible(Game_wind, 0);
StartNewLevel(new_level_num);
window_set_visible(Game_wind, 1);
2006-03-20 17:12:09 +00:00
}
}
}
2006-03-20 17:12:09 +00:00
#if defined(DXX_BUILD_DESCENT_II)
if (gotcha == &game_cheats::fullautomap && cheats.fullautomap)
HUD_init_message(HM_DEFAULT, "FULL MAP!");
#endif
2013-11-28 02:16:05 +00:00
if (gotcha == &game_cheats::ghostphysics)
2006-03-20 17:12:09 +00:00
{
HUD_init_message(HM_DEFAULT, "%s %s!", "Ghosty mode", cheats.ghostphysics?TXT_ON:TXT_OFF);
2006-03-20 17:12:09 +00:00
}
2013-11-28 02:16:05 +00:00
if (gotcha == &game_cheats::rapidfire)
#if defined(DXX_BUILD_DESCENT_I)
{
do_megawow_powerup(200);
}
#elif defined(DXX_BUILD_DESCENT_II)
2006-03-20 17:12:09 +00:00
{
HUD_init_message(HM_DEFAULT, "Rapid fire %s!", cheats.rapidfire?TXT_ON:TXT_OFF);
2006-03-20 17:12:09 +00:00
}
2013-11-28 02:16:05 +00:00
if (gotcha == &game_cheats::bouncyfire)
{
HUD_init_message(HM_DEFAULT, "Bouncing weapons %s!", cheats.bouncyfire?TXT_ON:TXT_OFF);
}
#endif
2006-03-20 17:12:09 +00:00
2013-11-28 02:16:05 +00:00
if (gotcha == &game_cheats::turbo)
{
HUD_init_message(HM_DEFAULT, "%s %s!", "Turbo mode", cheats.turbo?TXT_ON:TXT_OFF);
}
2006-03-20 17:12:09 +00:00
2013-11-28 02:16:05 +00:00
if (gotcha == &game_cheats::robotfiringsuspended)
{
HUD_init_message(HM_DEFAULT, "Robot firing %s!", cheats.robotfiringsuspended?TXT_OFF:TXT_ON);
}
2006-03-20 17:12:09 +00:00
#if defined(DXX_BUILD_DESCENT_II)
2013-11-28 02:16:05 +00:00
if (gotcha == &game_cheats::killallrobots)
{
kill_all_robots();
}
2006-03-20 17:12:09 +00:00
2013-11-28 02:16:05 +00:00
if (gotcha == &game_cheats::robotskillrobots)
{
HUD_init_message_literal(HM_DEFAULT, cheats.robotskillrobots?"Rabid robots!":"Kill the player!");
}
2006-03-20 17:12:09 +00:00
2013-11-28 02:16:05 +00:00
if (gotcha == &game_cheats::monsterdamage)
{
HUD_init_message_literal(HM_DEFAULT, cheats.monsterdamage?"Oh no, there goes Tokyo!":"What have you done, I'm shrinking!!");
}
2006-03-20 17:12:09 +00:00
2013-11-28 02:16:05 +00:00
if (gotcha == &game_cheats::buddyclone)
{
HUD_init_message_literal(HM_DEFAULT, "What's this? Another buddy bot!");
create_buddy_bot();
}
2006-03-20 17:12:09 +00:00
2013-11-28 02:16:05 +00:00
if (gotcha == &game_cheats::buddyangry)
{
if (cheats.buddyangry)
2006-03-20 17:12:09 +00:00
{
HUD_init_message(HM_DEFAULT, "%s gets angry!",PlayerCfg.GuidebotName);
strcpy(PlayerCfg.GuidebotName,"Wingnut");
2006-03-20 17:12:09 +00:00
}
else
{
strcpy(PlayerCfg.GuidebotName,PlayerCfg.GuidebotNameReal);
HUD_init_message(HM_DEFAULT, "%s calms down",PlayerCfg.GuidebotName);
2006-03-20 17:12:09 +00:00
}
}
#endif
2006-03-20 17:12:09 +00:00
2013-11-28 02:16:05 +00:00
if (gotcha == &game_cheats::acid)
{
HUD_init_message_literal(HM_DEFAULT, cheats.acid?"Going up!":"Coming down!");
2006-03-20 17:12:09 +00:00
}
2014-08-06 02:10:49 +00:00
return window_event_result::handled;
2006-03-20 17:12:09 +00:00
}
// Internal Cheat Menu
#ifndef RELEASE
2013-09-22 22:26:27 +00:00
static void do_cheat_menu()
2006-03-20 17:12:09 +00:00
{
int mmn;
newmenu_item mm[16];
char score_text[21];
sprintf( score_text, "%d", Players[Player_num].score );
nm_set_item_checkbox(&mm[0],TXT_INVULNERABILITY,Players[Player_num].flags & PLAYER_FLAGS_INVULNERABLE);
nm_set_item_checkbox(&mm[1],TXT_CLOAKED,Players[Player_num].flags & PLAYER_FLAGS_CLOAKED);
nm_set_item_checkbox(&mm[2],"All keys",0);
nm_set_item_number(&mm[3], "% Energy", f2i(Players[Player_num].energy), 0, 200);
nm_set_item_number(&mm[4], "% Shields", f2i(Players[Player_num].shields), 0, 200);
nm_set_item_text(& mm[5], "Score:");
nm_set_item_input(&mm[6], 10, score_text);
#if defined(DXX_BUILD_DESCENT_I)
nm_set_item_radio(&mm[7], "Laser level 1", (Players[Player_num].laser_level==0), 0);
nm_set_item_radio(&mm[8], "Laser level 2", (Players[Player_num].laser_level==1), 0);
nm_set_item_radio(&mm[9], "Laser level 3", (Players[Player_num].laser_level==2), 0);
nm_set_item_radio(&mm[10], "Laser level 4", (Players[Player_num].laser_level==3), 0);
nm_set_item_number(&mm[11], "Missiles", Players[Player_num].secondary_ammo[CONCUSSION_INDEX], 0, 200);
mmn = newmenu_do("Wimp Menu",NULL,12, mm, unused_newmenu_subfunction, unused_newmenu_userdata);
#elif defined(DXX_BUILD_DESCENT_II)
nm_set_item_number(&mm[7], "Laser Level", Players[Player_num].laser_level+1, 0, MAX_SUPER_LASER_LEVEL+1);
nm_set_item_number(&mm[8], "Missiles", Players[Player_num].secondary_ammo[CONCUSSION_INDEX], 0, 200);
2006-03-20 17:12:09 +00:00
mmn = newmenu_do("Wimp Menu",NULL,9, mm, unused_newmenu_subfunction, unused_newmenu_userdata);
#endif
2006-03-20 17:12:09 +00:00
if (mmn > -1 ) {
if ( mm[0].value ) {
Players[Player_num].flags |= PLAYER_FLAGS_INVULNERABLE;
Players[Player_num].invulnerable_time = GameTime64+i2f(1000);
2006-03-20 17:12:09 +00:00
} else
Players[Player_num].flags &= ~PLAYER_FLAGS_INVULNERABLE;
if ( mm[1].value )
{
2006-03-20 17:12:09 +00:00
Players[Player_num].flags |= PLAYER_FLAGS_CLOAKED;
if (Game_mode & GM_MULTI)
multi_send_cloak();
ai_do_cloak_stuff();
Players[Player_num].cloak_time = GameTime64;
2006-03-20 17:12:09 +00:00
}
else
Players[Player_num].flags &= ~PLAYER_FLAGS_CLOAKED;
if (mm[2].value) Players[Player_num].flags |= PLAYER_FLAGS_BLUE_KEY | PLAYER_FLAGS_RED_KEY | PLAYER_FLAGS_GOLD_KEY;
Players[Player_num].energy=i2f(mm[3].value);
Players[Player_num].shields=i2f(mm[4].value);
Players[Player_num].score = atoi(mm[6].text);
#if defined(DXX_BUILD_DESCENT_I)
if (mm[7].value) Players[Player_num].laser_level=0;
if (mm[8].value) Players[Player_num].laser_level=1;
if (mm[9].value) Players[Player_num].laser_level=2;
if (mm[10].value) Players[Player_num].laser_level=3;
Players[Player_num].secondary_ammo[CONCUSSION_INDEX] = mm[11].value;
#elif defined(DXX_BUILD_DESCENT_II)
2006-03-20 17:12:09 +00:00
Players[Player_num].laser_level = mm[7].value-1;
Players[Player_num].secondary_ammo[CONCUSSION_INDEX] = mm[8].value;
#endif
2006-03-20 17:12:09 +00:00
init_gauges();
}
}
#endif
// Testing functions ----------------------------------------------------------
#ifndef NDEBUG
// Sounds for testing
int test_sound_num = 0;
int sound_nums[] = {10,11,20,21,30,31,32,33,40,41,50,51,60,61,62,70,80,81,82,83,90,91};
#define N_TEST_SOUNDS (sizeof(sound_nums) / sizeof(*sound_nums))
2013-09-22 22:26:27 +00:00
static void advance_sound()
2006-03-20 17:12:09 +00:00
{
if (++test_sound_num == N_TEST_SOUNDS)
test_sound_num=0;
}
int Test_sound = 251;
2013-09-22 22:26:27 +00:00
static void play_test_sound()
2006-03-20 17:12:09 +00:00
{
// -- digi_play_sample(sound_nums[test_sound_num], F1_0);
digi_play_sample(Test_sound, F1_0);
}
#endif //ifndef NDEBUG
2014-10-04 21:47:13 +00:00
window_event_result ReadControls(const d_event &event)
2006-03-20 17:12:09 +00:00
{
int key;
static ubyte exploding_flag=0;
Player_fired_laser_this_frame=object_none;
2006-03-20 17:12:09 +00:00
if (Player_exploded) {
2006-03-20 17:12:09 +00:00
if (exploding_flag==0) {
exploding_flag = 1; // When player starts exploding, clear all input devices...
game_flush_inputs();
}
} else {
exploding_flag=0;
}
if (Player_is_dead && !( (Game_mode & GM_MULTI) && (multi_sending_message[Player_num] || multi_defining_message) ))
if (HandleDeathInput(event))
2014-08-06 02:10:49 +00:00
return window_event_result::handled;
2006-03-20 17:12:09 +00:00
if (Newdemo_state == ND_STATE_PLAYBACK)
2006-03-20 17:12:09 +00:00
update_vcr_state();
2014-10-04 21:47:13 +00:00
if (event.type == EVENT_KEY_COMMAND)
{
key = event_key_get(event);
#if defined(DXX_BUILD_DESCENT_II)
if (DefiningMarkerMessage)
{
return MarkerInputMessage(key);
}
#endif
if ( (Game_mode & GM_MULTI) && (multi_sending_message[Player_num] || multi_defining_message) )
{
return multi_message_input_sub(key);
}
#ifndef RELEASE
if ((key&KEY_DEBUGGED)&&(Game_mode&GM_MULTI)) {
Network_message_reciever = 100; // Send to everyone...
sprintf( Network_message, "%s %s", TXT_I_AM_A, TXT_CHEATER);
}
#endif
2006-03-20 17:12:09 +00:00
if (Endlevel_sequence)
{
if (HandleEndlevelKey(key))
2014-08-06 02:10:49 +00:00
return window_event_result::handled;
}
else if (Newdemo_state == ND_STATE_PLAYBACK )
{
if (HandleDemoKey(key))
2014-08-06 02:10:49 +00:00
return window_event_result::handled;
}
else
{
2014-08-06 02:10:49 +00:00
window_event_result r = FinalCheats(key);
if (r == window_event_result::ignored)
r = HandleSystemKey(key);
if (r == window_event_result::ignored)
r = HandleGameKey(key);
if (r != window_event_result::ignored)
return r;
}
2006-03-20 17:12:09 +00:00
#ifndef RELEASE
2014-08-06 02:10:49 +00:00
{
window_event_result r = HandleTestKey(key);
if (r != window_event_result::ignored)
return r;
}
#endif
if (call_default_handler(event))
2014-08-06 02:10:49 +00:00
return window_event_result::handled;
}
if (!Endlevel_sequence && !Player_is_dead && (Newdemo_state != ND_STATE_PLAYBACK))
{
kconfig_read_controls(event, 0);
check_rear_view();
// If automap key pressed, enable automap unless you are in network mode, control center destroyed and < 10 seconds left
if ( Controls.state.automap )
{
Controls.state.automap = 0;
if (!((Game_mode & GM_MULTI) && Control_center_destroyed && (Countdown_seconds_left < 10)))
{
2014-07-16 02:35:14 +00:00
do_automap();
2014-08-06 02:10:49 +00:00
return window_event_result::handled;
}
}
do_weapon_n_item_stuff();
}
2014-08-06 02:10:49 +00:00
return window_event_result::ignored;
2006-03-20 17:12:09 +00:00
}