Merge branch 'main' into code-readability-2

# Conflicts:
#	src/scripts.h
This commit is contained in:
phobos2077 2023-07-30 23:48:01 +02:00
commit 9798183c09
38 changed files with 3243 additions and 75 deletions

View File

@ -701,6 +701,30 @@ static int aiPacketWrite(File* stream, AiPacket* ai)
return 0;
}
// 0x428058
int combat_ai_num()
{
return gAiPacketsLength;
}
// 0x428060
char* combat_ai_name(int packet_num)
{
int index;
if (packet_num < 0 || packet_num >= gAiPacketsLength) {
return NULL;
}
for (index = 0; index < gAiPacketsLength; index++) {
if (gAiPackets[index].packet_num == packet_num) {
return gAiPackets[index].name;
}
}
return NULL;
}
// Get ai from object
//
// 0x4280B4

View File

@ -30,6 +30,8 @@ void aiReset();
int aiExit();
int aiLoad(File* stream);
int aiSave(File* stream);
int combat_ai_num();
char* combat_ai_name(int packet_num);
int aiGetAreaAttackMode(Object* obj);
int aiGetRunAwayMode(Object* obj);
int aiGetBestWeapon(Object* obj);

View File

@ -1396,4 +1396,40 @@ bool _critter_flag_check(int pid, int flag)
return (proto->critter.data.flags & flag) != 0;
}
// 0x42E6F0
void critter_flag_set(int pid, int flag)
{
Proto* proto;
if (pid == -1) {
return;
}
if (PID_TYPE(pid) != OBJ_TYPE_CRITTER) {
return;
}
protoGetProto(pid, &proto);
proto->critter.data.flags |= flag;
}
// 0x42E71C
void critter_flag_unset(int pid, int flag)
{
Proto* proto;
if (pid == -1) {
return;
}
if (PID_TYPE(pid) != OBJ_TYPE_CRITTER) {
return;
}
protoGetProto(pid, &proto);
proto->critter.data.flags &= ~flag;
}
} // namespace fallout

View File

@ -70,6 +70,8 @@ int critterGetMovementPointCostAdjustedForCrippledLegs(Object* critter, int a2);
bool critterIsEncumbered(Object* critter);
bool critterIsFleeing(Object* a1);
bool _critter_flag_check(int pid, int flag);
void critter_flag_set(int pid, int flag);
void critter_flag_unset(int pid, int flag);
} // namespace fallout

View File

@ -125,10 +125,26 @@ bool gameConfigInit(bool isMapper, int argc, char** argv)
char* ch = strrchr(executable, '\\');
if (ch != NULL) {
*ch = '\0';
snprintf(gGameConfigFilePath, sizeof(gGameConfigFilePath), "%s\\%s", executable, GAME_CONFIG_FILE_NAME);
if (isMapper) {
snprintf(gGameConfigFilePath,
sizeof(gGameConfigFilePath),
"%s\\%s",
executable,
MAPPER_CONFIG_FILE_NAME);
} else {
snprintf(gGameConfigFilePath,
sizeof(gGameConfigFilePath),
"%s\\%s",
executable,
GAME_CONFIG_FILE_NAME);
}
*ch = '\\';
} else {
strcpy(gGameConfigFilePath, GAME_CONFIG_FILE_NAME);
if (isMapper) {
strcpy(gGameConfigFilePath, MAPPER_CONFIG_FILE_NAME);
} else {
strcpy(gGameConfigFilePath, GAME_CONFIG_FILE_NAME);
}
}
// Read contents of `fallout2.cfg` into config. The values from the file

View File

@ -5,8 +5,8 @@
namespace fallout {
// The file name of the main config file.
#define GAME_CONFIG_FILE_NAME "fallout2.cfg"
#define MAPPER_CONFIG_FILE_NAME "mapper2.cfg"
#define GAME_CONFIG_SYSTEM_KEY "system"
#define GAME_CONFIG_PREFERENCES_KEY "preferences"

View File

@ -1328,6 +1328,12 @@ int gameMouseGetCursor()
return gGameMouseCursor;
}
// 0x44C9F0
void gmouse_set_mapper_mode(int mode)
{
_gmouse_mapper_mode = mode;
}
// 0x44C9F8
void _gmouse_3d_enable_modes()
{

View File

@ -86,6 +86,7 @@ void gameMouseRefresh();
void _gmouse_handle_event(int mouseX, int mouseY, int mouseState);
int gameMouseSetCursor(int cursor);
int gameMouseGetCursor();
void gmouse_set_mapper_mode(int mode);
void gameMouseSetMode(int a1);
int gameMouseGetMode();
void gameMouseCycleMode();

View File

@ -52,6 +52,14 @@ unsigned char HighRGB(unsigned char color)
return std::max(std::max(r, g), b);
}
// 0x44ED98
int load_lbm_to_buf(const char* path, unsigned char* buffer, int a3, int a4, int a5, int a6, int a7)
{
// TODO: Incomplete.
return -1;
}
// 0x44F250
int graphCompress(unsigned char* a1, unsigned char* a2, int a3)
{

View File

@ -4,6 +4,7 @@
namespace fallout {
unsigned char HighRGB(unsigned char color);
int load_lbm_to_buf(const char* path, unsigned char* buffer, int a3, int a4, int a5, int a6, int a7);
int graphCompress(unsigned char* a1, unsigned char* a2, int a3);
int graphDecompress(unsigned char* a1, unsigned char* a2, int a3);
void grayscalePaletteUpdate(int a1, int a2);

View File

@ -169,7 +169,6 @@ static int _GameMap2Slot(File* stream);
static int _SlotMap2Game(File* stream);
static int _mygets(char* dest, File* stream);
static int _copy_file(const char* existingFileName, const char* newFileName);
static int _MapDirErase(const char* path, const char* extension);
static int _SaveBackup();
static int _RestoreSave();
static int _LoadObjDudeCid(File* stream);
@ -340,9 +339,9 @@ void _InitLoadSave()
_slot_cursor = 0;
_patches = settings.system.master_patches_path.c_str();
_MapDirErase("MAPS\\", "SAV");
_MapDirErase(PROTO_DIR_NAME "\\" CRITTERS_DIR_NAME "\\", PROTO_FILE_EXT);
_MapDirErase(PROTO_DIR_NAME "\\" ITEMS_DIR_NAME "\\", PROTO_FILE_EXT);
MapDirErase("MAPS\\", "SAV");
MapDirErase(PROTO_DIR_NAME "\\" CRITTERS_DIR_NAME "\\", PROTO_FILE_EXT);
MapDirErase(PROTO_DIR_NAME "\\" ITEMS_DIR_NAME "\\", PROTO_FILE_EXT);
configGetInt(&gSfallConfig, SFALL_CONFIG_MISC_KEY, SFALL_CONFIG_AUTO_QUICK_SAVE, &quickSaveSlots);
if (quickSaveSlots > 0 && quickSaveSlots <= 10) {
@ -353,9 +352,9 @@ void _InitLoadSave()
// 0x47B85C
void _ResetLoadSave()
{
_MapDirErase("MAPS\\", "SAV");
_MapDirErase(PROTO_DIR_NAME "\\" CRITTERS_DIR_NAME "\\", PROTO_FILE_EXT);
_MapDirErase(PROTO_DIR_NAME "\\" ITEMS_DIR_NAME "\\", PROTO_FILE_EXT);
MapDirErase("MAPS\\", "SAV");
MapDirErase(PROTO_DIR_NAME "\\" CRITTERS_DIR_NAME "\\", PROTO_FILE_EXT);
MapDirErase(PROTO_DIR_NAME "\\" ITEMS_DIR_NAME "\\", PROTO_FILE_EXT);
}
// SaveGame
@ -1570,7 +1569,7 @@ static int lsgPerformSaveGame()
debugPrint("\nLOADSAVE: ** Error opening save game for writing! **\n");
_RestoreSave();
snprintf(_gmpath, sizeof(_gmpath), "%s\\%s%.2d\\", "SAVEGAME", "SLOT", _slot_cursor + 1);
_MapDirErase(_gmpath, "BAK");
MapDirErase(_gmpath, "BAK");
_partyMemberUnPrepSave();
backgroundSoundResume();
return -1;
@ -1583,7 +1582,7 @@ static int lsgPerformSaveGame()
fileClose(_flptr);
_RestoreSave();
snprintf(_gmpath, sizeof(_gmpath), "%s\\%s%.2d\\", "SAVEGAME", "SLOT", _slot_cursor + 1);
_MapDirErase(_gmpath, "BAK");
MapDirErase(_gmpath, "BAK");
_partyMemberUnPrepSave();
backgroundSoundResume();
return -1;
@ -1597,7 +1596,7 @@ static int lsgPerformSaveGame()
fileClose(_flptr);
_RestoreSave();
snprintf(_gmpath, sizeof(_gmpath), "%s\\%s%.2d\\", "SAVEGAME", "SLOT", _slot_cursor + 1);
_MapDirErase(_gmpath, "BAK");
MapDirErase(_gmpath, "BAK");
_partyMemberUnPrepSave();
backgroundSoundResume();
return -1;
@ -1678,7 +1677,7 @@ static int lsgPerformSaveGame()
}
snprintf(_gmpath, sizeof(_gmpath), "%s\\%s%.2d\\", "SAVEGAME", "SLOT", _slot_cursor + 1);
_MapDirErase(_gmpath, "BAK");
MapDirErase(_gmpath, "BAK");
gLoadSaveMessageListItem.num = 140;
if (messageListGetItem(&gLoadSaveMessageList, &gLoadSaveMessageListItem)) {
@ -1771,7 +1770,7 @@ static int lsgLoadGameInSlot(int slot)
}
snprintf(_str, sizeof(_str), "%s\\", "MAPS");
_MapDirErase(_str, "BAK");
MapDirErase(_str, "BAK");
_proto_dude_update_gender();
// Game Loaded.
@ -2487,7 +2486,7 @@ static int _GameMap2Slot(File* stream)
snprintf(_gmpath, sizeof(_gmpath), "%s\\%s%.2d\\", "SAVEGAME", "SLOT", _slot_cursor + 1);
if (_MapDirErase(_gmpath, "SAV") == -1) {
if (MapDirErase(_gmpath, "SAV") == -1) {
fileNameListFree(&fileNameList, 0);
return -1;
}
@ -2566,19 +2565,19 @@ static int _SlotMap2Game(File* stream)
snprintf(_str0, sizeof(_str0), "%s\\", PROTO_DIR_NAME "\\" CRITTERS_DIR_NAME);
if (_MapDirErase(_str0, PROTO_FILE_EXT) == -1) {
if (MapDirErase(_str0, PROTO_FILE_EXT) == -1) {
debugPrint("LOADSAVE: returning 3\n");
return -1;
}
snprintf(_str0, sizeof(_str0), "%s\\", PROTO_DIR_NAME "\\" ITEMS_DIR_NAME);
if (_MapDirErase(_str0, PROTO_FILE_EXT) == -1) {
if (MapDirErase(_str0, PROTO_FILE_EXT) == -1) {
debugPrint("LOADSAVE: returning 4\n");
return -1;
}
snprintf(_str0, sizeof(_str0), "%s\\", "MAPS");
if (_MapDirErase(_str0, "SAV") == -1) {
if (MapDirErase(_str0, "SAV") == -1) {
debugPrint("LOADSAVE: returning 5\n");
return -1;
}
@ -2749,11 +2748,11 @@ void lsgInit()
{
char path[COMPAT_MAX_PATH];
snprintf(path, sizeof(path), "%s\\", "MAPS");
_MapDirErase(path, "SAV");
MapDirErase(path, "SAV");
}
// 0x480040
static int _MapDirErase(const char* relativePath, const char* extension)
int MapDirErase(const char* relativePath, const char* extension)
{
char path[COMPAT_MAX_PATH];
snprintf(path, sizeof(path), "%s*.%s", relativePath, extension);

View File

@ -20,6 +20,7 @@ int lsgSaveGame(int mode);
int lsgLoadGame(int mode);
bool _isLoadingGame();
void lsgInit();
int MapDirErase(const char* path, const char* extension);
int _MapDirEraseFile_(const char* a1, const char* a2);
} // namespace fallout

View File

@ -108,7 +108,7 @@ int* gMapLocalVars = NULL;
// map_vars
// 0x51956C
static int* gMapGlobalVars = NULL;
int* gMapGlobalVars = NULL;
// local_vars_num
// 0x519570
@ -116,7 +116,7 @@ int gMapLocalVarsLength = 0;
// map_vars_num
// 0x519574
static int gMapGlobalVarsLength = 0;
int gMapGlobalVarsLength = 0;
// Current elevation.
//

View File

@ -69,7 +69,9 @@ typedef void IsoWindowRefreshProc(Rect* rect);
extern int gMapSid;
extern int* gMapLocalVars;
extern int* gMapGlobalVars;
extern int gMapLocalVarsLength;
extern int gMapGlobalVarsLength;
extern int gElevation;
extern MessageList gMapMessageList;

17
src/mapper/map_func.cc Normal file
View File

@ -0,0 +1,17 @@
#include "mapper/map_func.h"
namespace fallout {
// 0x4825B0
void setup_map_dirs()
{
// TODO: Incomplete.
}
// 0x4826B4
void copy_proto_lists()
{
// TODO: Incomplete.
}
} // namespace fallout

11
src/mapper/map_func.h Normal file
View File

@ -0,0 +1,11 @@
#ifndef FALLOUT_MAPPER_MAP_FUNC_H_
#define FALLOUT_MAPPER_MAP_FUNC_H_
namespace fallout {
void setup_map_dirs();
void copy_proto_lists();
} // namespace fallout
#endif /* FALLOUT_MAPPER_MAP_FUNC_H_ */

1495
src/mapper/mapper.cc Normal file

File diff suppressed because it is too large Load Diff

23
src/mapper/mapper.h Normal file
View File

@ -0,0 +1,23 @@
#ifndef FALLOUT_MAPPER_MAPPER_H_
#define FALLOUT_MAPPER_MAPPER_H_
#include "map.h"
#include "obj_types.h"
namespace fallout {
extern MapTransition mapInfo;
extern int menu_val_0[8];
extern int menu_val_2[8];
extern int menu_val_1[21];
extern unsigned char* tool;
extern int tool_win;
int mapper_main(int argc, char** argv);
void print_toolbar_name(int object_type);
int mapper_inven_unwield(Object* obj, int right_hand);
} // namespace fallout
#endif /* FALLOUT_MAPPER_MAPPER_H_ */

438
src/mapper/mp_proto.cc Normal file
View File

@ -0,0 +1,438 @@
#include "mapper/mp_proto.h"
#include <string.h>
#include "color.h"
#include "combat_ai.h"
#include "critter.h"
#include "input.h"
#include "kb.h"
#include "mapper/mp_targt.h"
#include "memory.h"
#include "proto.h"
#include "svga.h"
#include "window_manager.h"
#include "window_manager_private.h"
namespace fallout {
#define CRITTER_FLAG_COUNT 10
#define YES 0
#define NO 1
static int proto_choose_container_flags(Proto* proto);
static void proto_critter_flags_redraw(int win, int pid);
static int proto_critter_flags_modify(int pid);
static int mp_pick_kill_type();
static char kYes[] = "YES";
static char kNo[] = "NO";
// 0x559B94
static const char* wall_light_strs[] = {
"North/South",
"East/West",
"North Corner",
"South Corner",
"East Corner",
"West Corner",
};
// 0x559C50
static char* yesno[] = {
kYes,
kNo,
};
// 0x559C58
int edit_window_color = 1;
// 0x559C60
bool can_modify_protos = false;
// 0x559C6C
static int critFlagList[CRITTER_FLAG_COUNT] = {
CRITTER_NO_STEAL,
CRITTER_NO_DROP,
CRITTER_NO_LIMBS,
CRITTER_NO_AGE,
CRITTER_NO_HEAL,
CRITTER_INVULNERABLE,
CRITTER_FLAT,
CRITTER_SPECIAL_DEATH,
CRITTER_LONG_LIMBS,
CRITTER_NO_KNOCKBACK,
};
// 0x559C94
static const char* critFlagStrs[CRITTER_FLAG_COUNT] = {
"_Steal",
"_Drop",
"_Limbs",
"_Ages",
"_Heal",
"Invuln.,",
"_Flattens",
"Special",
"Rng",
"_Knock",
};
// 0x4922F8
void init_mapper_protos()
{
edit_window_color = _colorTable[10570];
can_modify_protos = target_overriden();
}
// 0x492840
int proto_choose_container_flags(Proto* proto)
{
int win = windowCreate(320,
185,
220,
205,
edit_window_color,
WINDOW_MOVE_ON_TOP);
if (win == -1) {
return -1;
}
_win_register_text_button(win,
10,
11,
-1,
-1,
-1,
'1',
"Magic Hands Grnd",
0);
if ((proto->item.data.container.openFlags & 0x1) != 0) {
windowDrawText(win,
yesno[YES],
50,
125,
15,
_colorTable[32747] | 0x10000);
} else {
windowDrawText(win,
yesno[NO],
50,
125,
15,
_colorTable[32747] | 0x10000);
}
_win_register_text_button(win,
10,
32,
-1,
-1,
-1,
'2',
"Cannot Pick Up",
0);
if (_proto_action_can_pickup(proto->pid)) {
windowDrawText(win,
yesno[YES],
50,
125,
36,
_colorTable[32747] | 0x10000);
} else {
windowDrawText(win,
yesno[NO],
50,
125,
36,
_colorTable[32747] | 0x10000);
}
windowDrawBorder(win);
windowRefresh(win);
while (1) {
sharedFpsLimiter.mark();
int input = inputGetInput();
if (input == KEY_ESCAPE
|| input == KEY_BAR
|| input == KEY_RETURN) {
break;
}
if (input == '1') {
proto->item.data.container.openFlags ^= 0x1;
if ((proto->item.data.container.openFlags & 0x1) != 0) {
windowDrawText(win,
yesno[YES],
50,
125,
15,
_colorTable[32747] | 0x10000);
} else {
windowDrawText(win,
yesno[NO],
50,
125,
15,
_colorTable[32747] | 0x10000);
}
windowRefresh(win);
} else if (input == '2') {
proto->item.extendedFlags ^= 0x8000;
if (_proto_action_can_pickup(proto->pid)) {
windowDrawText(win,
yesno[YES],
50,
125,
36,
_colorTable[32747] | 0x10000);
} else {
windowDrawText(win,
yesno[NO],
50,
125,
36,
_colorTable[32747] | 0x10000);
}
windowRefresh(win);
}
renderPresent();
sharedFpsLimiter.throttle();
}
windowDestroy(win);
return 0;
}
// 0x495438
const char* proto_wall_light_str(int flags)
{
if ((flags & 0x8000000) != 0) {
return wall_light_strs[1];
}
if ((flags & 0x10000000) != 0) {
return wall_light_strs[2];
}
if ((flags & 0x20000000) != 0) {
return wall_light_strs[3];
}
if ((flags & 0x40000000) != 0) {
return wall_light_strs[4];
}
if ((flags & 0x80000000) != 0) {
return wall_light_strs[5];
}
return wall_light_strs[0];
}
// 0x4960B8
void proto_critter_flags_redraw(int win, int pid)
{
int index;
int color;
int x = 110;
for (index = 0; index < CRITTER_FLAG_COUNT; index++) {
if (_critter_flag_check(pid, critFlagList[index])) {
color = _colorTable[992];
} else {
color = _colorTable[10570];
}
windowDrawText(win, critFlagStrs[index], 44, x, 195, color | 0x10000);
x += 48;
}
}
// 0x496120
int proto_critter_flags_modify(int pid)
{
Proto* proto;
int rc;
int flags = 0;
int index;
if (protoGetProto(pid, &proto) == -1) {
return -1;
}
rc = win_yes_no("Can't be stolen from?", 340, 200, _colorTable[15855]);
if (rc == -1) {
return -1;
}
if (rc == 1) {
flags |= CRITTER_NO_STEAL;
}
rc = win_yes_no("Can't Drop items?", 340, 200, _colorTable[15855]);
if (rc == -1) {
return -1;
}
if (rc == 1) {
flags |= CRITTER_NO_DROP;
}
rc = win_yes_no("Can't lose limbs?", 340, 200, _colorTable[15855]);
if (rc == -1) {
return -1;
}
if (rc == 1) {
flags |= CRITTER_NO_LIMBS;
}
rc = win_yes_no("Dead Bodies Can't Age?", 340, 200, _colorTable[15855]);
if (rc == -1) {
return -1;
}
if (rc == 1) {
flags |= CRITTER_NO_AGE;
}
rc = win_yes_no("Can't Heal by Aging?", 340, 200, _colorTable[15855]);
if (rc == -1) {
return -1;
}
if (rc == 1) {
flags |= CRITTER_NO_HEAL;
}
rc = win_yes_no("Is Invlunerable????", 340, 200, _colorTable[15855]);
if (rc == -1) {
return -1;
}
if (rc == 1) {
flags |= CRITTER_INVULNERABLE;
}
rc = win_yes_no("Can't Flatten on Death?", 340, 200, _colorTable[15855]);
if (rc == -1) {
return -1;
}
if (rc == 1) {
flags |= CRITTER_FLAT;
}
rc = win_yes_no("Has Special Death?", 340, 200, _colorTable[15855]);
if (rc == -1) {
return -1;
}
if (rc == 1) {
flags |= CRITTER_SPECIAL_DEATH;
}
rc = win_yes_no("Has Extra Hand-To-Hand Range?", 340, 200, _colorTable[15855]);
if (rc == -1) {
return -1;
}
if (rc == 1) {
flags |= CRITTER_LONG_LIMBS;
}
rc = win_yes_no("Can't be knocked back?", 340, 200, _colorTable[15855]);
if (rc == -1) {
return -1;
}
if (rc == 1) {
flags |= CRITTER_NO_KNOCKBACK;
}
if (!can_modify_protos) {
win_timed_msg("Can't modify protos!", _colorTable[31744] | 0x10000);
return -1;
}
for (index = 0; index < CRITTER_FLAG_COUNT; index++) {
if ((critFlagList[index] & flags) != 0) {
critter_flag_set(pid, critFlagList[index]);
} else {
critter_flag_unset(pid, critFlagList[index]);
}
}
return 0;
}
// 0x497520
int mp_pick_kill_type()
{
char* names[KILL_TYPE_COUNT];
int index;
for (index = 0; index < KILL_TYPE_COUNT; index++) {
names[index] = killTypeGetName(index);
}
return _win_list_select("Kill Type",
names,
KILL_TYPE_COUNT,
NULL,
50,
100,
_colorTable[15855]);
}
// 0x497568
int proto_pick_ai_packet(int* value)
{
int count;
char** names;
int index;
int rc;
count = combat_ai_num();
if (count <= 0) {
return -1;
}
names = (char**)internal_malloc(sizeof(char*) * count);
for (index = 0; index < count; index++) {
names[index] = (char*)internal_malloc(strlen(combat_ai_name(index)) + 1);
strcpy(names[index], combat_ai_name(index));
}
rc = _win_list_select("AI Packet",
names,
count,
NULL,
50,
100,
_colorTable[15855]);
if (rc != -1) {
*value = rc;
}
for (index = 0; index < count; index++) {
internal_free(names[index]);
}
internal_free(names);
return 0;
}
} // namespace fallout

14
src/mapper/mp_proto.h Normal file
View File

@ -0,0 +1,14 @@
#ifndef FALLOUT_MAPPER_MP_PROTO_H_
#define FALLOUT_MAPPER_MP_PROTO_H_
namespace fallout {
extern bool can_modify_protos;
void init_mapper_protos();
const char* proto_wall_light_str(int flags);
int proto_pick_ai_packet(int* value);
} // namespace fallout
#endif /* FALLOUT_MAPPER_MP_PROTO_H_ */

90
src/mapper/mp_scrpt.cc Normal file
View File

@ -0,0 +1,90 @@
#include "mapper/mp_scrpt.h"
#include "art.h"
#include "object.h"
#include "scripts.h"
#include "tile.h"
namespace fallout {
// 0x49B170
int map_scr_remove_spatial(int tile, int elevation)
{
Script* scr;
Object* obj;
Rect rect;
scr = scriptGetFirstSpatialScript(elevation);
while (scr != NULL) {
if (builtTileGetTile(scr->sp.built_tile) == tile) {
scriptRemove(scr->sid);
scr = scriptGetFirstSpatialScript(elevation);
continue;
}
scr = scriptGetNextSpatialScript();
}
obj = objectFindFirstAtElevation(elevation);
while (obj != NULL) {
if (obj->tile == tile && buildFid(OBJ_TYPE_INTERFACE, 3, 0, 0, 0) == obj->fid) {
objectDestroy(obj, &rect);
tileWindowRefreshRect(&rect, elevation);
obj = objectFindFirstAtElevation(elevation);
continue;
}
obj = objectFindNextAtElevation();
}
return 0;
}
// 0x49B214
int map_scr_remove_all_spatials()
{
int elevation;
Script* scr;
Object* obj;
int sid;
for (elevation = 0; elevation < ELEVATION_COUNT; elevation++) {
scr = scriptGetFirstSpatialScript(elevation);
while (scr != NULL) {
scriptRemove(scr->sid);
scr = scriptGetFirstSpatialScript(elevation);
}
obj = objectFindFirstAtElevation(elevation);
while (obj != NULL) {
if (buildFid(OBJ_TYPE_INTERFACE, 3, 0, 0, 0) == obj->fid) {
objectDestroy(obj, NULL);
obj = objectFindFirstAtElevation(elevation);
continue;
}
obj = objectFindNextAtElevation();
}
}
tileWindowRefresh();
for (sid = 0; sid < 15000; sid++) {
if (scriptGetScript(sid, &scr) != -1) {
if (scr->owner != NULL) {
if (scr->owner->pid == 0x500000C) {
scr->owner->sid = -1;
scriptRemove(sid);
}
}
}
}
return 0;
}
} // namespace fallout

11
src/mapper/mp_scrpt.h Normal file
View File

@ -0,0 +1,11 @@
#ifndef FALLOUT_MAPPER_MP_SCRPTR_H_
#define FALLOUT_MAPPER_MP_SCRPTR_H_
namespace fallout {
int map_scr_remove_spatial(int tile, int elevation);
int map_scr_remove_all_spatials();
} // namespace fallout
#endif /* FALLOUT_MAPPER_MP_SCRPTR_H_ */

154
src/mapper/mp_targt.cc Normal file
View File

@ -0,0 +1,154 @@
#include "mapper/mp_targt.h"
#include <string.h>
#include "art.h"
#include "game.h"
#include "map.h"
#include "proto.h"
#include "window_manager_private.h"
namespace fallout {
// 0x53F354
static char default_target_path_base[] = "\\fallout2\\dev\\proto\\";
// 0x559CD0
static char* target_path_base = default_target_path_base;
// 0x559DBC
static bool tgt_overriden = false;
// 0x49B2F0
void target_override_protection()
{
// TODO: Incomplete.
}
// 0x49B2F0
bool target_overriden()
{
return tgt_overriden;
}
// 0x49B34C
void target_make_path(char* path, int pid)
{
if (_cd_path_base[0] != '\0' && _cd_path_base[1] == ':') {
strncpy(path, _cd_path_base, 2);
strcat(path, target_path_base);
} else {
strcpy(path, target_path_base);
}
if (pid != -1) {
strcat(path, artGetObjectTypeName(PID_TYPE(pid)));
}
}
// 0x49B424
int target_init()
{
// TODO: Incomplete.
return 0;
}
// 0x49B434
int target_exit()
{
// TODO: Incomplete.
return 0;
}
// 0x49BD98
int pick_rot()
{
int value;
win_get_num_i(&value,
-1,
5,
false,
"Rotation",
100,
100);
return value;
}
// 0x49BDD0
int target_pick_global_var(int* value_ptr)
{
int value;
int rc;
if (gGameGlobalVarsLength == 0) {
return -1;
}
rc = win_get_num_i(&value,
0,
gGameGlobalVarsLength - 1,
false,
"Global Variable Index #:",
100,
100);
if (rc == -1) {
return -1;
}
*value_ptr = value;
return 0;
}
// 0x49BE20
int target_pick_map_var(int* value_ptr)
{
int value;
int rc;
if (gMapGlobalVarsLength == 0) {
return -1;
}
rc = win_get_num_i(&value,
0,
gMapGlobalVarsLength - 1,
false,
"Map Variable Index #:",
100,
100);
if (rc == -1) {
return -1;
}
*value_ptr = value;
return 0;
}
// 0x49BE70
int target_pick_local_var(int* value_ptr)
{
int value;
int rc;
if (gMapLocalVarsLength == 0) {
return -1;
}
rc = win_get_num_i(&value,
0,
gMapLocalVarsLength - 1,
false,
"Local Variable Index #:",
100,
100);
if (rc == -1) {
return -1;
}
*value_ptr = value;
return 0;
}
} // namespace fallout

18
src/mapper/mp_targt.h Normal file
View File

@ -0,0 +1,18 @@
#ifndef FALLOUT_MAPPER_MP_TARGT_H_
#define FALLOUT_MAPPER_MP_TARGT_H_
namespace fallout {
void target_override_protection();
bool target_overriden();
void target_make_path(char* path, int pid);
int target_init();
int target_exit();
int pick_rot();
int target_pick_global_var(int* value_ptr);
int target_pick_map_var(int* value_ptr);
int target_pick_local_var(int* value_ptr);
} // namespace fallout
#endif /* FALLOUT_MAPPER_MP_TARGT_H_ */

13
src/mapper/mp_text.cc Normal file
View File

@ -0,0 +1,13 @@
#include "mapper/mp_text.h"
namespace fallout {
// 0x49DAC4
int proto_build_all_texts()
{
// TODO: Incomplete.
return 0;
}
} // namespace fallout

10
src/mapper/mp_text.h Normal file
View File

@ -0,0 +1,10 @@
#ifndef FALLOUT_MAPPER_MP_TEXT_H_
#define FALLOUT_MAPPER_MP_TEXT_H_
namespace fallout {
int proto_build_all_texts();
} // namespace fallout
#endif /* FALLOUT_MAPPER_MP_TEXT_H_ */

View File

@ -32,7 +32,6 @@ typedef struct MemoryBlockFooter {
static void* memoryBlockMallocImpl(size_t size);
static void* memoryBlockReallocImpl(void* ptr, size_t size);
static void memoryBlockFreeImpl(void* ptr);
static void memoryBlockPrintStats();
static void* mem_prep_block(void* block, size_t size);
static void memoryBlockValidate(void* block);
@ -176,10 +175,8 @@ static void memoryBlockFreeImpl(void* ptr)
}
}
// NOTE: Not used.
//
// 0x4C5C5C
static void memoryBlockPrintStats()
void mem_check()
{
if (gMallocProc == memoryBlockMallocImpl) {
debugPrint("Current memory allocated: %6d blocks, %9u bytes total\n", gMemoryBlocksCurrentCount, gMemoryBlocksCurrentSize);

View File

@ -9,6 +9,7 @@ char* internal_strdup(const char* string);
void* internal_malloc(size_t size);
void* internal_realloc(void* ptr, size_t size);
void internal_free(void* ptr);
void mem_check();
} // namespace fallout

View File

@ -24,7 +24,6 @@
namespace fallout {
static int _proto_critter_init(Proto* a1, int a2);
static int objectCritterCombatDataRead(CritterCombatData* data, File* stream);
static int objectCritterCombatDataWrite(CritterCombatData* data, File* stream);
static int _proto_update_gen(Object* obj);
@ -39,8 +38,8 @@ static int _proto_load_pid(int pid, Proto** out_proto);
static int _proto_find_free_subnode(int type, Proto** out_ptr);
static void _proto_remove_some_list(int type);
static void _proto_remove_list(int type);
static int _proto_new_id(int a1);
static int _proto_max_id(int a1);
static int _proto_new_id(int type);
static int _proto_max_id(int type);
// 0x50CF3C
static char _aProto_0[] = "proto\\";
@ -187,8 +186,8 @@ static char** _perk_code_strs;
// 0x6648BC
static char** _critter_stats_list;
// NOTE: Inlined.
void _proto_make_path(char* path, int pid)
// 0x49E270
void proto_make_path(char* path, int pid)
{
strcpy(path, _cd_path_base);
strcat(path, _proto_path_base);
@ -211,7 +210,7 @@ int _proto_list_str(int pid, char* proto_path)
}
char path[COMPAT_MAX_PATH];
_proto_make_path(path, pid);
proto_make_path(path, pid);
strcat(path, "\\");
strcat(path, artGetObjectTypeName(PID_TYPE(pid)));
strcat(path, ".lst");
@ -370,32 +369,143 @@ char* protoGetDescription(int pid)
return protoGetMessage(pid, PROTOTYPE_MESSAGE_DESCRIPTION);
}
// 0x49EB2C
int proto_item_init(Proto* proto, int a2)
{
int v1 = a2 & 0xFFFFFF;
proto->item.pid = -1;
proto->item.messageId = 100 * v1;
proto->item.fid = buildFid(OBJ_TYPE_ITEM, v1 - 1, 0, 0, 0);
if (!artExists(proto->item.fid)) {
proto->item.fid = buildFid(OBJ_TYPE_ITEM, 0, 0, 0, 0);
}
proto->item.lightDistance = 0;
proto->item.lightIntensity = 0;
proto->item.flags = 0xA0000008;
proto->item.extendedFlags = 0xA000;
proto->item.sid = -1;
proto->item.type = ITEM_TYPE_MISC;
proto_item_subdata_init(proto, proto->item.type);
proto->item.material = 1;
proto->item.size = 1;
proto->item.weight = 10;
proto->item.cost = 0;
proto->item.inventoryFid = -1;
proto->item.field_80 = '0';
return 0;
}
// 0x49EBFC
int proto_item_subdata_init(Proto* proto, int type)
{
int index;
switch (type) {
case ITEM_TYPE_ARMOR:
proto->item.data.armor.armorClass = 0;
for (index = 0; index < DAMAGE_TYPE_COUNT; index++) {
proto->item.data.armor.damageResistance[index] = 0;
proto->item.data.armor.damageThreshold[index] = 0;
}
proto->item.data.armor.perk = -1;
proto->item.data.armor.maleFid = -1;
proto->item.data.armor.femaleFid = -1;
break;
case ITEM_TYPE_CONTAINER:
proto->item.data.container.openFlags = 0;
proto->item.data.container.maxSize = 250;
proto->item.extendedFlags |= 0x800;
break;
case ITEM_TYPE_DRUG:
proto->item.data.drug.stat[0] = STAT_STRENGTH;
proto->item.data.drug.stat[1] = -1;
proto->item.data.drug.stat[2] = -1;
proto->item.data.drug.amount[0] = 0;
proto->item.data.drug.amount[1] = 0;
proto->item.data.drug.amount[2] = 0;
proto->item.data.drug.duration1 = 0;
proto->item.data.drug.amount1[0] = 0;
proto->item.data.drug.amount1[1] = 0;
proto->item.data.drug.amount1[2] = 0;
proto->item.data.drug.duration2 = 0;
proto->item.data.drug.amount2[0] = 0;
proto->item.data.drug.amount2[1] = 0;
proto->item.data.drug.amount2[2] = 0;
proto->item.data.drug.addictionChance = 0;
proto->item.data.drug.withdrawalEffect = 0;
proto->item.data.drug.withdrawalOnset = 0;
proto->item.extendedFlags |= 0x1000;
break;
case ITEM_TYPE_WEAPON:
proto->item.data.weapon.animationCode = 0;
proto->item.data.weapon.minDamage = 0;
proto->item.data.weapon.maxDamage = 0;
proto->item.data.weapon.damageType = 0;
proto->item.data.weapon.maxRange1 = 0;
proto->item.data.weapon.maxRange2 = 0;
proto->item.data.weapon.projectilePid = -1;
proto->item.data.weapon.minStrength = 0;
proto->item.data.weapon.actionPointCost1 = 0;
proto->item.data.weapon.actionPointCost2 = 0;
proto->item.data.weapon.criticalFailureType = 0;
proto->item.data.weapon.perk = -1;
proto->item.data.weapon.rounds = 0;
proto->item.data.weapon.caliber = 0;
proto->item.data.weapon.ammoTypePid = -1;
proto->item.data.weapon.ammoCapacity = 0;
proto->item.data.weapon.soundCode = 0;
break;
case ITEM_TYPE_AMMO:
proto->item.data.ammo.caliber = 0;
proto->item.data.ammo.quantity = 20;
proto->item.data.ammo.armorClassModifier = 0;
proto->item.data.ammo.damageResistanceModifier = 0;
proto->item.data.ammo.damageMultiplier = 1;
proto->item.data.ammo.damageDivisor = 1;
break;
case ITEM_TYPE_MISC:
proto->item.data.misc.powerTypePid = -1;
proto->item.data.misc.powerType = 20;
break;
case ITEM_TYPE_KEY:
proto->item.data.key.keyCode = -1;
proto->item.extendedFlags |= 0x1000;
break;
}
return 0;
}
// 0x49EDB4
static int _proto_critter_init(Proto* a1, int a2)
int proto_critter_init(Proto* proto, int pid)
{
if (!_protos_been_initialized) {
return -1;
}
int v1 = a2 & 0xFFFFFF;
int num = pid & 0xFFFFFF;
a1->pid = -1;
a1->messageId = 100 * v1;
a1->fid = buildFid(OBJ_TYPE_CRITTER, v1 - 1, 0, 0, 0);
a1->critter.lightDistance = 0;
a1->critter.lightIntensity = 0;
a1->critter.flags = 0x20000000;
a1->critter.extendedFlags = 0x6000;
a1->critter.sid = -1;
a1->critter.data.flags = 0;
a1->critter.data.bodyType = 0;
a1->critter.headFid = -1;
a1->critter.aiPacket = 1;
if (!artExists(a1->fid)) {
a1->fid = buildFid(OBJ_TYPE_CRITTER, 0, 0, 0, 0);
proto->pid = -1;
proto->messageId = 100 * num;
proto->fid = buildFid(OBJ_TYPE_CRITTER, num - 1, 0, 0, 0);
proto->critter.lightDistance = 0;
proto->critter.lightIntensity = 0;
proto->critter.flags = 0x20000000;
proto->critter.extendedFlags = 0x6000;
proto->critter.sid = -1;
proto->critter.data.flags = 0;
proto->critter.data.bodyType = 0;
proto->critter.headFid = -1;
proto->critter.aiPacket = 1;
if (!artExists(proto->fid)) {
proto->fid = buildFid(OBJ_TYPE_CRITTER, 0, 0, 0, 0);
}
CritterProtoData* data = &(a1->critter.data);
CritterProtoData* data = &(proto->critter.data);
data->experience = 60;
data->killType = 0;
data->damageType = 0;
@ -828,6 +938,165 @@ int _proto_dude_init(const char* path)
return 0;
}
// 0x49FBBC
int proto_scenery_init(Proto* proto, int pid)
{
int num = pid & 0xFFFFFF;
proto->scenery.pid = -1;
proto->scenery.messageId = 100 * num;
proto->scenery.fid = buildFid(OBJ_TYPE_SCENERY, num - 1, 0, 0, 0);
if (!artExists(proto->scenery.fid)) {
proto->scenery.fid = buildFid(OBJ_TYPE_SCENERY, 0, 0, 0, 0);
}
proto->scenery.lightDistance = 0;
proto->scenery.lightIntensity = 0;
proto->scenery.flags = 0;
proto->scenery.extendedFlags = 0x2000;
proto->scenery.sid = -1;
proto->scenery.type = SCENERY_TYPE_GENERIC;
proto_scenery_subdata_init(proto, proto->scenery.type);
proto->scenery.field_2C = -1;
proto->scenery.field_34 = '0';
return 0;
}
// 0x49FC74
int proto_scenery_subdata_init(Proto* proto, int type)
{
switch (type) {
case SCENERY_TYPE_DOOR:
proto->scenery.data.door.openFlags = 0;
proto->scenery.extendedFlags |= 0x800;
break;
case SCENERY_TYPE_STAIRS:
proto->scenery.data.stairs.field_0 = -1;
proto->scenery.data.stairs.field_4 = -1;
proto->scenery.extendedFlags |= 0x800;
break;
case SCENERY_TYPE_ELEVATOR:
proto->scenery.data.elevator.type = -1;
proto->scenery.data.elevator.level = -1;
proto->scenery.extendedFlags |= 0x800;
break;
case SCENERY_TYPE_LADDER_UP:
proto->scenery.data.ladder.field_0 = -1;
proto->scenery.extendedFlags |= 0x800;
break;
case SCENERY_TYPE_LADDER_DOWN:
proto->scenery.data.ladder.field_0 = -1;
proto->scenery.extendedFlags |= 0x800;
break;
}
return 0;
}
// 0x49FCFC
int proto_wall_init(Proto* proto, int pid)
{
int num = pid & 0xFFFFFF;
proto->wall.pid = -1;
proto->wall.messageId = 100 * num;
proto->wall.fid = buildFid(OBJ_TYPE_WALL, num - 1, 0, 0, 0);
if (!artExists(proto->wall.fid)) {
proto->wall.fid = buildFid(OBJ_TYPE_WALL, 0, 0, 0, 0);
}
proto->wall.lightDistance = 0;
proto->wall.lightIntensity = 0;
proto->wall.flags = 0;
proto->wall.extendedFlags = 0x2000;
proto->wall.sid = -1;
proto->wall.material = 1;
return 0;
}
// 0x49FD84
int proto_tile_init(Proto* proto, int pid)
{
int num = pid & 0xFFFFFF;
proto->tile.pid = -1;
proto->tile.messageId = 100 * num;
proto->tile.fid = buildFid(OBJ_TYPE_TILE, num - 1, 0, 0, 0);
if (!artExists(proto->tile.fid)) {
proto->tile.fid = buildFid(OBJ_TYPE_TILE, 0, 0, 0, 0);
}
proto->tile.flags = 0;
proto->tile.extendedFlags = 0x2000;
proto->tile.sid = -1;
proto->tile.material = 1;
return 0;
}
// 0x49FDFC
int proto_misc_init(Proto* proto, int pid)
{
int num = pid & 0xFFFFFF;
proto->misc.pid = -1;
proto->misc.messageId = 100 * num;
proto->misc.fid = buildFid(OBJ_TYPE_MISC, num - 1, 0, 0, 0);
if (!artExists(proto->misc.fid)) {
proto->misc.fid = buildFid(OBJ_TYPE_MISC, 0, 0, 0, 0);
}
proto->misc.lightDistance = 0;
proto->misc.lightIntensity = 0;
proto->misc.flags = 0;
proto->misc.extendedFlags = 0;
return 0;
}
// 0x49FE74
int proto_copy_proto(int srcPid, int dstPid)
{
int srcType;
int dstType;
Proto* src;
Proto* dst;
srcType = PID_TYPE(srcPid);
dstType = PID_TYPE(dstPid);
if (srcType != dstType) {
return -1;
}
if (protoGetProto(srcPid, &src) == -1) {
return -1;
}
if (protoGetProto(dstPid, &dst) == -1) {
return -1;
}
memcpy(dst, src, _proto_sizes[srcType]);
dst->pid = dstPid;
return 0;
}
// 0x49FEDC
bool proto_is_subtype(Proto* proto, int subtype)
{
if (subtype == -1) {
return true;
}
switch (PID_TYPE(proto->pid)) {
case OBJ_TYPE_ITEM:
return proto->item.type == subtype;
case OBJ_TYPE_SCENERY:
return proto->scenery.type == subtype;
}
return false;
}
// proto_data_member
// 0x49FFD8
int protoGetDataMember(int pid, int member, ProtoDataMemberValue* value)
@ -1083,7 +1352,7 @@ int protoInit()
compat_mkdir(path);
// TODO: Get rid of cast.
_proto_critter_init((Proto*)&gDudeProto, 0x1000000);
proto_critter_init((Proto*)&gDudeProto, 0x1000000);
gDudeProto.pid = 0x1000000;
gDudeProto.fid = buildFid(OBJ_TYPE_CRITTER, 1, 0, 0, 0);
@ -1202,7 +1471,7 @@ void protoReset()
int i;
// TODO: Get rid of cast.
_proto_critter_init((Proto*)&gDudeProto, 0x1000000);
proto_critter_init((Proto*)&gDudeProto, 0x1000000);
gDudeProto.pid = 0x1000000;
gDudeProto.fid = buildFid(OBJ_TYPE_CRITTER, 1, 0, 0, 0);
@ -1251,7 +1520,7 @@ static int _proto_header_load()
ptr->max_entries_num = 1;
char path[COMPAT_MAX_PATH];
_proto_make_path(path, index << 24);
proto_make_path(path, index << 24);
strcat(path, "\\");
strcat(path, artGetObjectTypeName(index));
strcat(path, ".lst");
@ -1661,7 +1930,7 @@ int _proto_save_pid(int pid)
}
char path[260];
_proto_make_path(path, pid);
proto_make_path(path, pid);
strcat(path, "\\");
_proto_list_str(pid, path + strlen(path));
@ -1682,7 +1951,7 @@ int _proto_save_pid(int pid)
static int _proto_load_pid(int pid, Proto** protoPtr)
{
char path[COMPAT_MAX_PATH];
_proto_make_path(path, pid);
proto_make_path(path, pid);
strcat(path, "\\");
if (_proto_list_str(pid, path + strlen(path)) == -1) {
@ -1761,6 +2030,48 @@ static int _proto_find_free_subnode(int type, Proto** protoPtr)
return 0;
}
// 0x4A1E90
int proto_new(int* pid, int type)
{
Proto* proto;
if (_proto_find_free_subnode(type, &proto) == -1) {
return -1;
}
*pid = _proto_new_id(type) | (type << 24);
switch (type) {
case OBJ_TYPE_ITEM:
proto_item_init(proto, *pid);
proto->item.pid = *pid;
break;
case OBJ_TYPE_CRITTER:
proto_critter_init(proto, *pid);
proto->critter.pid = *pid;
break;
case OBJ_TYPE_SCENERY:
proto_scenery_init(proto, *pid);
proto->scenery.pid = *pid;
break;
case OBJ_TYPE_WALL:
proto_wall_init(proto, *pid);
proto->wall.pid = *pid;
break;
case OBJ_TYPE_TILE:
proto_tile_init(proto, *pid);
proto->tile.pid = *pid;
break;
case OBJ_TYPE_MISC:
proto_misc_init(proto, *pid);
proto->misc.pid = *pid;
break;
default:
return -1;
}
return 0;
}
// Evict top most proto cache block.
//
// 0x4A2040
@ -1850,18 +2161,18 @@ int protoGetProto(int pid, Proto** protoPtr)
}
// 0x4A21DC
static int _proto_new_id(int a1)
static int _proto_new_id(int type)
{
int result = _protoLists[a1].max_entries_num;
_protoLists[a1].max_entries_num = result + 1;
int result = _protoLists[type].max_entries_num;
_protoLists[type].max_entries_num = result + 1;
return result;
}
// 0x4A2214
static int _proto_max_id(int a1)
static int _proto_max_id(int type)
{
return _protoLists[a1].max_entries_num;
return _protoLists[type].max_entries_num;
}
// 0x4A22C0

View File

@ -102,7 +102,7 @@ extern char _cd_path_base[COMPAT_MAX_PATH];
extern MessageList gProtoMessageList;
extern char* _proto_none_str;
void _proto_make_path(char* path, int pid);
void proto_make_path(char* path, int pid);
int _proto_list_str(int pid, char* proto_path);
size_t proto_size(int type);
bool _proto_action_can_use(int pid);
@ -112,19 +112,30 @@ int _proto_action_can_pickup(int pid);
char* protoGetMessage(int pid, int message);
char* protoGetName(int pid);
char* protoGetDescription(int pid);
int proto_item_init(Proto* proto, int pid);
int proto_item_subdata_init(Proto* proto, int type);
int proto_critter_init(Proto* proto, int pid);
void objectDataReset(Object* obj);
int objectDataRead(Object* obj, File* stream);
int objectDataWrite(Object* obj, File* stream);
int _proto_update_init(Object* obj);
int _proto_dude_update_gender();
int _proto_dude_init(const char* path);
int proto_scenery_init(Proto* proto, int pid);
int proto_scenery_subdata_init(Proto* proto, int type);
int proto_wall_init(Proto* proto, int pid);
int proto_tile_init(Proto* proto, int pid);
int proto_misc_init(Proto* proto, int pid);
int proto_copy_proto(int srcPid, int dstPid);
bool proto_is_subtype(Proto* proto, int subtype);
int protoGetDataMember(int pid, int member, ProtoDataMemberValue* value);
int protoInit();
void protoReset();
void protoExit();
int _proto_save_pid(int pid);
int proto_new(int* pid, int type);
void _proto_remove_all();
int protoGetProto(int pid, Proto** out_proto);
int protoGetProto(int pid, Proto** protoPtr);
int _ResetPlayer();
static bool isExitGridPid(int pid)

View File

@ -913,8 +913,8 @@ int scriptsHandleRequests()
}
}
if ((gScriptsRequests & SCRIPT_REQUEST_0x02) != 0) {
gScriptsRequests &= ~SCRIPT_REQUEST_0x02;
if ((gScriptsRequests & SCRIPT_REQUEST_TOWN_MAP) != 0) {
gScriptsRequests &= ~SCRIPT_REQUEST_TOWN_MAP;
wmTownMap();
}
@ -1128,6 +1128,16 @@ void _scripts_request_combat_locked(CombatStartData* a1)
gScriptsRequests |= (SCRIPT_REQUEST_0x0400 | SCRIPT_REQUEST_COMBAT);
}
// 0x4A461C
void scripts_request_townmap()
{
if (isInCombat()) {
_game_user_wants_to_quit = 1;
}
gScriptsRequests |= SCRIPT_REQUEST_TOWN_MAP;
}
// request_world_map()
// 0x4A4644
void scriptsRequestWorldMap()

View File

@ -25,7 +25,7 @@ namespace fallout {
typedef enum ScriptRequests {
SCRIPT_REQUEST_COMBAT = 0x01,
SCRIPT_REQUEST_0x02 = 0x02,
SCRIPT_REQUEST_TOWN_MAP = 0x02,
SCRIPT_REQUEST_WORLD_MAP = 0x04,
SCRIPT_REQUEST_ELEVATOR = 0x08,
SCRIPT_REQUEST_EXPLOSION = 0x10,
@ -175,6 +175,7 @@ int scriptsHandleRequests();
int _scripts_check_state_in_combat();
int scriptsRequestCombat(CombatStartData* combat);
void _scripts_request_combat_locked(CombatStartData* combat);
void scripts_request_townmap();
void scriptsRequestWorldMap();
int scriptsRequestElevator(Object* a1, int a2);
int scriptsRequestExplosion(int tile, int elevation, int minDamage, int maxDamage);

View File

@ -654,6 +654,17 @@ static void tileRefreshGame(Rect* rect, int elevation)
gTileWindowRefreshProc(&rectToUpdate);
}
// 0x4B1634
void tile_toggle_roof(bool refresh)
{
gTileRoofIsVisible = !gTileRoofIsVisible;
if (refresh) {
// NOTE: Uninline.
tileWindowRefresh();
}
}
// 0x4B166C
int tileRoofIsVisible()
{

View File

@ -26,6 +26,7 @@ void tileEnable();
void tileWindowRefreshRect(Rect* rect, int elevation);
void tileWindowRefresh();
int tileSetCenter(int tile, int flags);
void tile_toggle_roof(bool refresh);
int tileRoofIsVisible();
int tileToScreenXY(int tile, int* x, int* y, int elevation);
int tileFromScreenXY(int x, int y, int elevation, bool ignoreBounds = false);

View File

@ -1257,19 +1257,18 @@ Button* buttonGetButton(int btn, Window** windowPtr)
}
// 0x4D7A34
int _GNW_check_menu_bars(int a1)
int _GNW_check_menu_bars(int input)
{
if (!gWindowSystemInitialized) {
return -1;
}
int v1 = a1;
for (int index = gWindowsLength - 1; index >= 1; index--) {
Window* window = gWindows[index];
if (window->menuBar != NULL) {
for (int pulldownIndex = 0; pulldownIndex < window->menuBar->pulldownsLength; pulldownIndex++) {
if (v1 == window->menuBar->pulldowns[pulldownIndex].keyCode) {
v1 = _GNW_process_menu(window->menuBar, pulldownIndex);
if (input == window->menuBar->pulldowns[pulldownIndex].keyCode) {
input = _GNW_process_menu(window->menuBar, pulldownIndex);
break;
}
}
@ -1280,7 +1279,7 @@ int _GNW_check_menu_bars(int a1)
}
}
return v1;
return input;
}
// 0x4D69DC

View File

@ -177,7 +177,7 @@ int windowGetWidth(int win);
int windowGetHeight(int win);
int windowGetRect(int win, Rect* rect);
int _win_check_all_buttons();
int _GNW_check_menu_bars(int a1);
int _GNW_check_menu_bars(int input);
void programWindowSetTitle(const char* title);
bool showMesageBox(const char* str);
int buttonCreate(int win, int x, int y, int width, int height, int mouseEnterEventCode, int mouseExitEventCode, int mouseDownEventCode, int mouseUpEventCode, unsigned char* up, unsigned char* dn, unsigned char* hover, int flags);

View File

@ -20,6 +20,7 @@ namespace fallout {
/// Maximum number of timed messages.
static constexpr int kTimedMsgs = 5;
static int get_num_i(int win, int* value, int max_chars_wcursor, bool clear, bool allow_negative, int x, int y);
static void tm_watch_msgs();
static void tm_kill_msg();
static void tm_kill_out_of_order(int queueIndex);
@ -612,7 +613,7 @@ int _win_get_str(char* dest, int length, const char* title, int x, int y)
windowRefresh(win);
_win_input_str(win,
int rc = _win_input_str(win,
dest,
length,
16,
@ -622,7 +623,7 @@ int _win_get_str(char* dest, int length, const char* title, int x, int y)
windowDestroy(win);
return 0;
return rc;
}
// 0x4DB920
@ -1215,11 +1216,333 @@ int _win_input_str(int win, char* dest, int maxLength, int x, int y, int textCol
return 0;
}
// 0x4DCD68
int win_get_num_i(int* value, int min, int max, bool clear, const char* title, int x, int y)
{
if (!gWindowSystemInitialized) {
return -1;
}
if (max < min) {
return -1;
}
if (*value < min) {
*value = min;
} else if (*value > max) {
*value = max;
}
int original = *value;
int max_chars_wcursor = _calc_max_field_chars_wcursor(min, max);
if (max_chars_wcursor == -1) {
return -1;
}
int v2 = fontGetMonospacedCharacterWidth() * max_chars_wcursor;
int width = fontGetStringWidth(title);
if (width < v2) {
width = v2;
}
width += 16;
if (width < 160) {
width = 160;
}
int height = 5 * fontGetLineHeight() + 16;
int v3 = (width - v2) / 2;
int v4 = fontGetLineHeight();
int v5 = fontGetLineHeight() + 2;
int win = windowCreate(x, y, width, height, 0x100, WINDOW_MODAL | WINDOW_MOVE_ON_TOP);
if (win == -1) {
return -1;
}
windowDrawBorder(win);
windowFill(win, v3, v4 + 14, v2, v5, 0x100 | 1);
windowDrawText(win, title, width - 16, 8, 8, 0x100 | 5);
bufferDrawRectShadowed(windowGetBuffer(win),
width,
v3 - 2,
v4 + 12,
v3 + v2 + 1,
v4 + 14 + v5 - 1,
_colorTable[_GNW_wcolor[2]],
_colorTable[_GNW_wcolor[1]]);
_win_register_text_button(win,
width / 2 - 72,
height - fontGetLineHeight() - 14,
-1,
-1,
-1,
KEY_RETURN,
"Done",
0);
_win_register_text_button(win,
width / 2 + 16,
height - fontGetLineHeight() - 14,
-1,
-1,
-1,
KEY_ESCAPE,
"Cancel",
0);
char* hint = (char*)internal_malloc(80);
if (hint == NULL) {
return -1;
}
sprintf(hint, "Please enter a number between %d and %d.", min, max);
windowRefresh(win);
int rc;
while (1) {
rc = get_num_i(win, value, max_chars_wcursor, clear, min < 0, v3, v4 + 14);
if (*value >= min && *value <= max) {
break;
}
if (rc == -1) {
break;
}
_win_msg(hint, x - 70, y + 100, 0x100 | 6);
*value = original;
}
internal_free(hint);
windowDestroy(win);
return rc;
}
// 0x4DBD04
int process_pull_down(int win, Rect* rect, char** items, int itemsLength, int foregroundColor, int backgroundColor, MenuBar* menuBar, int pulldownIndex)
{
// TODO: Incomplete.
return -1;
if (menuBar != NULL) {
unsigned char* parentWindowBuffer = windowGetWindow(menuBar->win)->buffer;
MenuPulldown* pulldown = &(menuBar->pulldowns[pulldownIndex]);
int x = pulldown->rect.left;
int y = pulldown->rect.top;
int width = pulldown->rect.right - x + 1;
int height = pulldown->rect.bottom - y + 1;
int color1 = menuBar->foregroundColor;
if ((color1 & 0xFF00) != 0) {
int colorIndex = (color1 & 0xFF) - 1;
color1 = (color1 & ~0xFFFF) | _colorTable[_GNW_wcolor[colorIndex]];
}
int color2 = menuBar->backgroundColor;
if ((color2 & 0xFF00) != 0) {
int colorIndex = (color2 & 0xFF) - 1;
color2 = (color2 & ~0xFFFF) | _colorTable[_GNW_wcolor[colorIndex]];
}
_swap_color_buf(parentWindowBuffer + width * y + x,
width,
height,
windowGetWidth(menuBar->win),
color1,
color2);
windowRefreshRect(menuBar->win, &(pulldown->rect));
}
unsigned char* windowBuffer = windowGetWindow(win)->buffer;
int width = rectGetWidth(rect);
int height = rectGetHeight(rect);
int focusedIndex = -1;
int rc;
int mx1;
int my1;
int mx2;
int my2;
int input;
mouseGetPosition(&mx1, &my1);
while (1) {
sharedFpsLimiter.mark();
input = inputGetInput();
if (input != -1) {
break;
}
mouseGetPosition(&mx2, &my2);
if (mx2 < mx1 - 4
|| mx2 > mx1 + 4
|| my2 < my1 - 4
|| my2 > my1 + 4) {
break;
}
renderPresent();
sharedFpsLimiter.throttle();
}
while (1) {
sharedFpsLimiter.mark();
mouseGetPosition(&mx2, &my2);
if (input == -2) {
if (menuBar != NULL) {
if (_mouse_click_in(menuBar->rect.left, menuBar->rect.top, menuBar->rect.right, menuBar->rect.bottom)) {
int index;
for (index = 0; index < menuBar->pulldownsLength; index++) {
MenuPulldown* pulldown = &(menuBar->pulldowns[index]);
if (_mouse_click_in(pulldown->rect.left, pulldown->rect.top, pulldown->rect.right, pulldown->rect.bottom)) {
break;
}
}
if (index < menuBar->pulldownsLength && index != pulldownIndex) {
rc = -2 - index;
break;
}
}
}
}
if ((mouseGetEvent() & MOUSE_EVENT_LEFT_BUTTON_UP) != 0
|| ((mouseGetEvent() & MOUSE_EVENT_LEFT_BUTTON_DOWN) != 0
&& (mouseGetEvent() & MOUSE_EVENT_LEFT_BUTTON_REPEAT) == 0)) {
if (_mouse_click_in(rect->left, rect->top + 8, rect->right, rect->bottom - 9)) {
rc = focusedIndex;
} else {
rc = -1;
}
break;
}
bool done = false;
switch (input) {
case KEY_ESCAPE:
rc = -1;
done = true;
break;
case KEY_RETURN:
rc = focusedIndex;
done = true;
break;
case KEY_ARROW_LEFT:
if (menuBar != NULL && pulldownIndex > 0) {
rc = -2 - (pulldownIndex - 1);
done = true;
}
break;
case KEY_ARROW_RIGHT:
if (menuBar != NULL && pulldownIndex < menuBar->pulldownsLength - 1) {
rc = -2 - (pulldownIndex + 1);
done = true;
}
break;
case KEY_ARROW_UP:
while (focusedIndex > 0) {
focusedIndex--;
if (items[focusedIndex][0] != '\0') {
break;
}
}
input = -3;
break;
case KEY_ARROW_DOWN:
while (focusedIndex < itemsLength - 1) {
focusedIndex++;
if (items[focusedIndex][0] != '\0') {
break;
}
}
input = -3;
break;
default:
if (mx2 != mx1 || my2 != my1) {
if (_mouse_click_in(rect->left, rect->top + 8, rect->right, rect->bottom - 9)) {
input = (my2 - rect->top - 8) / fontGetLineHeight();
if (input != -1) {
focusedIndex = items[input][0] != '\0' ? input : -1;
input = -3;
}
}
mx1 = mx2;
my1 = my2;
}
break;
}
if (done) {
break;
}
if (input == -3) {
windowFill(win, 2, 8, width - 4, height - 16, backgroundColor);
_win_text(win, items, itemsLength, width - 4, 2, 8, foregroundColor);
if (focusedIndex != -1) {
_lighten_buf(windowBuffer + (focusedIndex * fontGetLineHeight() + 8) * width + 2,
width - 4,
fontGetLineHeight(),
width);
}
windowRefresh(win);
}
input = inputGetInput();
renderPresent();
sharedFpsLimiter.throttle();
}
if (menuBar != NULL) {
unsigned char* parentWindowBuffer = windowGetWindow(menuBar->win)->buffer;
MenuPulldown* pulldown = &(menuBar->pulldowns[pulldownIndex]);
int x = pulldown->rect.left;
int y = pulldown->rect.top;
int width = pulldown->rect.right - x + 1;
int height = pulldown->rect.bottom - y + 1;
int color1 = menuBar->foregroundColor;
if ((color1 & 0xFF00) != 0) {
int colorIndex = (color1 & 0xFF) - 1;
color1 = (color1 & ~0xFFFF) | _colorTable[_GNW_wcolor[colorIndex]];
}
int color2 = menuBar->backgroundColor;
if ((color2 & 0xFF00) != 0) {
int colorIndex = (color2 & 0xFF) - 1;
color2 = (color2 & ~0xFFFF) | _colorTable[_GNW_wcolor[colorIndex]];
}
_swap_color_buf(parentWindowBuffer + width * y + x,
width,
height,
windowGetWidth(menuBar->win),
color1,
color2);
windowRefreshRect(menuBar->win, &(pulldown->rect));
renderPresent();
}
windowDestroy(win);
return rc;
}
// 0x4DC930
@ -1285,6 +1608,116 @@ size_t _calc_max_field_chars_wcursor(int value1, int value2)
return std::max(len1, len2) + 1;
}
// 0x4DD0AC
int get_num_i(int win, int* value, int max_chars_wcursor, bool clear, bool allow_negative, int x, int y)
{
bool first_press = false;
Window* window = windowGetWindow(win);
if (window == NULL) {
return -1;
}
int original = *value;
int width = max_chars_wcursor * fontGetMonospacedCharacterWidth();
int height = fontGetLineHeight();
char* string = (char*)internal_malloc(max_chars_wcursor + 1);
if (clear) {
string[0] = '\0';
} else {
snprintf(string,
max_chars_wcursor + 1,
"%d",
*value);
}
int cursorPos = strlen(string);
string[cursorPos] = '_';
string[cursorPos + 1] = '\0';
windowDrawText(win, string, width, x, y, 0x100 | 4);
Rect rect;
rect.left = x;
rect.top = y;
rect.right = x + width;
rect.bottom = y + height;
windowRefreshRect(win, &rect);
bool done = false;
while (cursorPos <= max_chars_wcursor && !done) {
sharedFpsLimiter.mark();
int input = inputGetInput();
if (input == KEY_RETURN) {
done = true;
} else if (input == KEY_BACKSPACE) {
if (cursorPos > 0) {
int stringWidth = fontGetStringWidth(string);
if (first_press) {
string[0] = '_';
string[1] = '\0';
cursorPos = 1;
first_press = false;
} else {
string[cursorPos - 1] = '_';
string[cursorPos] = '\0';
cursorPos--;
}
windowFill(win, x, y, stringWidth, height, 0x100 | 1);
windowDrawText(win, string, width, x, y, 0x100 | 4);
windowRefreshRect(win, &rect);
}
} else if (input == KEY_ESCAPE) {
*value = original;
internal_free(string);
return -1;
} else if (input == KEY_ARROW_LEFT) {
if (cursorPos > 0) {
int stringWidth = fontGetStringWidth(string);
string[cursorPos - 1] = '_';
string[cursorPos] = '\0';
windowFill(win, x, y, stringWidth, height, 0x100 | 1);
windowDrawText(win, string, width, x, y, 0x100 | 4);
windowRefreshRect(win, &rect);
first_press = false;
cursorPos--;
}
} else {
if (cursorPos != max_chars_wcursor - 1) {
if ((input == '-' && allow_negative)
|| (input >= '0' && input <= '9')) {
string[cursorPos] = input;
string[cursorPos + 1] = '_';
string[cursorPos + 2] = '\0';
int stringWidth = fontGetStringWidth(string);
windowFill(win, x, y, stringWidth, height, 0x100 | 1);
windowDrawText(win, string, width, x, y, 0x100 | 4);
windowRefreshRect(win, &rect);
first_press = false;
cursorPos++;
}
}
}
renderPresent();
sharedFpsLimiter.throttle();
}
*value = atoi(string);
internal_free(string);
return 0;
}
// 0x4DD3EC
void _GNW_intr_init()
{

View File

@ -30,6 +30,7 @@ int _win_width_needed(char** fileNameList, int fileNameListLength);
int _win_input_str(int win, char* dest, int maxLength, int x, int y, int textColor, int backgroundColor);
int process_pull_down(int win, Rect* rect, char** items, int itemsLength, int a5, int a6, MenuBar* menuBar, int pulldownIndex);
int _GNW_process_menu(MenuBar* menuBar, int pulldownIndex);
int win_get_num_i(int* value, int min, int max, bool clear, const char* title, int x, int y);
size_t _calc_max_field_chars_wcursor(int value1, int value2);
void _GNW_intr_init();
void _GNW_intr_exit();