2016-08-10 11:10:56 +02:00
|
|
|
/* ScummVM - Graphic Adventure Engine
|
|
|
|
*
|
|
|
|
* ScummVM is the legal property of its developers, whose names
|
|
|
|
* are too numerous to list here. Please refer to the COPYRIGHT
|
|
|
|
* file distributed with this source distribution.
|
|
|
|
*
|
|
|
|
* This program is free software; you can redistribute it and/or
|
|
|
|
* modify it under the terms of the GNU General Public License
|
|
|
|
* as published by the Free Software Foundation; either version 2
|
|
|
|
* of the License, or (at your option) any later version.
|
|
|
|
*
|
|
|
|
* This program is distributed in the hope that it will be useful,
|
|
|
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
|
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
|
|
* GNU General Public License for more details.
|
|
|
|
*
|
|
|
|
* You should have received a copy of the GNU General Public License
|
|
|
|
* along with this program; if not, write to the Free Software
|
|
|
|
* Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
|
|
|
|
*
|
|
|
|
*/
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Based on the Reverse Engineering work of Christophe Fontanel,
|
|
|
|
* maintainer of the Dungeon Master Encyclopaedia (http://dmweb.free.fr/)
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include "console.h"
|
2016-08-10 14:44:54 +02:00
|
|
|
#include "dm.h"
|
|
|
|
#include "champion.h"
|
|
|
|
#include "dungeonman.h"
|
|
|
|
#include "movesens.h"
|
2016-08-10 11:10:56 +02:00
|
|
|
|
|
|
|
|
|
|
|
namespace DM {
|
|
|
|
|
2016-08-10 13:42:46 +02:00
|
|
|
bool cstrEquals(const char* a, const char *b) { return strcmp(a, b) == 0; }
|
|
|
|
|
2016-08-10 13:55:12 +02:00
|
|
|
class SingleUseFlag {
|
|
|
|
bool _flag;
|
|
|
|
public:
|
2016-08-10 14:44:54 +02:00
|
|
|
SingleUseFlag() : _flag(true) {}
|
2016-08-10 13:55:12 +02:00
|
|
|
bool check() {
|
|
|
|
bool currFlagState = _flag;
|
|
|
|
_flag = false;
|
|
|
|
return currFlagState;
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
2016-08-26 22:51:17 +02:00
|
|
|
Console::Console(DM::DMEngine* vm) : _vm(vm) {
|
|
|
|
_debugGodmodeMana = false;
|
|
|
|
_debugGodmodeHP = false;
|
|
|
|
_debugGodmodeStamina = false;
|
2016-08-10 11:10:56 +02:00
|
|
|
|
2016-08-10 13:42:46 +02:00
|
|
|
_debugNoclip = false;
|
|
|
|
|
2016-08-10 15:31:46 +02:00
|
|
|
registerCmd("godmode", WRAP_METHOD(Console, Cmd_godmode));
|
|
|
|
registerCmd("noclip", WRAP_METHOD(Console, Cmd_noclip));
|
|
|
|
registerCmd("pos", WRAP_METHOD(Console, Cmd_pos));
|
|
|
|
registerCmd("map", WRAP_METHOD(Console, Cmd_map));
|
2016-08-26 22:51:17 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
bool Console::Cmd_godmode(int argc, const char** argv) {
|
2016-08-10 13:42:46 +02:00
|
|
|
if (argc != 3)
|
2016-08-26 22:51:17 +02:00
|
|
|
goto argumentError;
|
|
|
|
|
|
|
|
bool setFlagTo;
|
|
|
|
|
2016-08-10 13:42:46 +02:00
|
|
|
if (cstrEquals("on", argv[2])) {
|
2016-08-26 22:51:17 +02:00
|
|
|
setFlagTo = true;
|
2016-08-10 13:42:46 +02:00
|
|
|
} else if (cstrEquals("off", argv[2])) {
|
2016-08-26 22:51:17 +02:00
|
|
|
setFlagTo = false;
|
|
|
|
} else
|
|
|
|
goto argumentError;
|
|
|
|
|
2016-08-10 13:42:46 +02:00
|
|
|
if (cstrEquals("all", argv[1])) {
|
2016-08-26 22:51:17 +02:00
|
|
|
_debugGodmodeHP = _debugGodmodeMana = _debugGodmodeStamina = setFlagTo;
|
2016-08-10 13:42:46 +02:00
|
|
|
} else if (cstrEquals("mana", argv[1])) {
|
2016-08-26 22:51:17 +02:00
|
|
|
_debugGodmodeMana = setFlagTo;
|
2016-08-10 13:42:46 +02:00
|
|
|
} else if (cstrEquals("hp", argv[1])) {
|
2016-08-26 22:51:17 +02:00
|
|
|
_debugGodmodeHP = setFlagTo;
|
2016-08-10 13:42:46 +02:00
|
|
|
} else if (cstrEquals("stamina", argv[1])) {
|
2016-08-26 22:51:17 +02:00
|
|
|
_debugGodmodeStamina = setFlagTo;
|
|
|
|
} else
|
|
|
|
goto argumentError;
|
|
|
|
|
|
|
|
debugPrintf("God mode set for %s to %s\n", argv[1], argv[2]);
|
|
|
|
return true;
|
|
|
|
|
|
|
|
argumentError:
|
|
|
|
debugPrintf("Usage: %s <all/mana/hp/stamina> <on/off>\n", argv[0]);
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2016-08-10 13:42:46 +02:00
|
|
|
bool Console::Cmd_noclip(int argc, const char** argv) {
|
|
|
|
if (argc != 2)
|
|
|
|
goto argumentError;
|
|
|
|
|
|
|
|
if (cstrEquals("on", argv[1])) {
|
|
|
|
_debugNoclip = true;
|
2016-08-10 15:31:12 +02:00
|
|
|
static SingleUseFlag haventWarned;
|
|
|
|
if (haventWarned.check())
|
2016-08-10 14:44:54 +02:00
|
|
|
debugPrintf("Noclip can cause glitches and crashes.\n");
|
2016-08-10 13:42:46 +02:00
|
|
|
} else if (cstrEquals("off", argv[1])) {
|
|
|
|
_debugNoclip = false;
|
|
|
|
} else
|
|
|
|
goto argumentError;
|
|
|
|
|
|
|
|
debugPrintf("Noclip set to %s\n", argv[1]);
|
|
|
|
return true;
|
|
|
|
|
|
|
|
argumentError:
|
|
|
|
debugPrintf("Usage: %s <on/off>\n", argv[0]);
|
|
|
|
return true;
|
|
|
|
}
|
2016-08-10 13:55:12 +02:00
|
|
|
|
2016-08-10 14:44:54 +02:00
|
|
|
bool Console::Cmd_pos(int argc, const char** argv) {
|
|
|
|
if (argc == 2 && cstrEquals("get", argv[1])) {
|
2016-08-10 15:31:12 +02:00
|
|
|
debugPrintf("Position: (%d, %d) Direction: %s\n", _vm->_dungeonMan->_g306_partyMapX,
|
2016-08-10 14:44:54 +02:00
|
|
|
_vm->_dungeonMan->_g307_partyMapY, debugGetDirectionName(_vm->_dungeonMan->_g308_partyDir));
|
|
|
|
} else if (argc == 4 && cstrEquals("set", argv[1])) {
|
|
|
|
int x = atoi(argv[2]);
|
|
|
|
int y = atoi(argv[3]);
|
2016-08-10 15:31:12 +02:00
|
|
|
if ((x == 0 && !cstrEquals("0", argv[2])) || (y == 0 && !cstrEquals("0", argv[3]))) {
|
|
|
|
debugPrintf("Error, supply two numbers to '%s set' command\n", argv[0]);
|
2016-08-10 14:44:54 +02:00
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
Map &currMap = *_vm->_dungeonMan->_g269_currMap;
|
|
|
|
// not >= because dimensions are inslucsive
|
|
|
|
if (x > currMap._width || y > currMap._height) {
|
|
|
|
debugPrintf("Position (%d, %d) is out of bounds, possible values: ([1-%d],[1-%d])\n", x, y,
|
|
|
|
currMap._width, currMap._height);
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2016-08-10 15:31:12 +02:00
|
|
|
static SingleUseFlag haventWarned;
|
|
|
|
if (haventWarned.check())
|
2016-08-10 14:44:54 +02:00
|
|
|
debugPrintf("Setting position directly can cause glitches and crashes.\n");
|
|
|
|
debugPrintf("Position set to (%d, %d)\n", x, y);
|
|
|
|
_vm->_moveSens->f267_getMoveResult(Thing::_party, _vm->_dungeonMan->_g306_partyMapX, _vm->_dungeonMan->_g307_partyMapY, x, y);
|
|
|
|
} else
|
|
|
|
goto argumentError;
|
|
|
|
|
|
|
|
return true;
|
2016-08-10 13:55:12 +02:00
|
|
|
|
2016-08-10 14:44:54 +02:00
|
|
|
argumentError:
|
|
|
|
debugPrintf("Usage: %s get\nUsage: %s set <#> <#>\n", argv[0], argv[0]);
|
|
|
|
return true;
|
|
|
|
}
|
2016-08-10 15:31:12 +02:00
|
|
|
|
|
|
|
bool Console::Cmd_map(int argc, const char** argv) {
|
|
|
|
if (argc == 2 && cstrEquals("get", argv[1])) {
|
|
|
|
debugPrintf("Map index: %d\n", _vm->_dungeonMan->_g309_partyMapIndex);
|
|
|
|
} else if (argc == 3 && cstrEquals("set", argv[1])) {
|
|
|
|
int index = atoi(argv[2]);
|
|
|
|
if (index == 0 && !cstrEquals("0", argv[2])) {
|
|
|
|
debugPrintf("Error, supply a number to '%s set' command\n", argv[0]);
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
// not >= because dimensions are inslucsive
|
|
|
|
if (index < 0 || index >= _vm->_dungeonMan->_g278_dungeonFileHeader._mapCount) {
|
|
|
|
debugPrintf("Map index %d is out of bounds, possible values [0, %d]\n", _vm->_dungeonMan->_g278_dungeonFileHeader._mapCount - 1);
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
static SingleUseFlag haventWarned;
|
|
|
|
if (haventWarned.check())
|
|
|
|
debugPrintf("Setting map directly can cause glitches and crashes.\n");
|
|
|
|
debugPrintf("Map set to %d\n", index);
|
|
|
|
|
|
|
|
_vm->_moveSens->f267_getMoveResult(Thing::_party, _vm->_dungeonMan->_g306_partyMapX, _vm->_dungeonMan->_g307_partyMapY, kM1_MapXNotOnASquare, 0);
|
|
|
|
_vm->_g327_newPartyMapIndex = _vm->_dungeonMan->f154_getLocationAfterLevelChange(
|
|
|
|
_vm->_dungeonMan->_g309_partyMapIndex, index - _vm->_dungeonMan->_g309_partyMapIndex,
|
|
|
|
&_vm->_dungeonMan->_g306_partyMapX, &_vm->_dungeonMan->_g307_partyMapY);
|
|
|
|
if (_vm->_g327_newPartyMapIndex == -1)
|
|
|
|
_vm->_g327_newPartyMapIndex = index;
|
|
|
|
_vm->_dungeonMan->f173_setCurrentMap(_vm->_g327_newPartyMapIndex);
|
|
|
|
_vm->_championMan->f284_setPartyDirection(_vm->_dungeonMan->f155_getStairsExitDirection(_vm->_dungeonMan->_g306_partyMapX, _vm->_dungeonMan->_g307_partyMapY));
|
|
|
|
_vm->_dungeonMan->f173_setCurrentMap(_vm->_dungeonMan->_g309_partyMapIndex);
|
|
|
|
} else
|
|
|
|
goto argumentError;
|
|
|
|
|
|
|
|
return true;
|
|
|
|
|
|
|
|
argumentError:
|
|
|
|
debugPrintf("Usage: %s get\nUsage: %s set <#>\n", argv[0], argv[0]);
|
|
|
|
return true;
|
|
|
|
}
|
2016-08-26 22:51:17 +02:00
|
|
|
}
|