scummvm/engines/adl/adl.cpp

459 lines
10 KiB
C++
Raw Normal View History

2016-02-26 22:32:06 +00: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.
*
*/
2016-02-27 12:35:49 +00:00
#include "common/scummsys.h"
2016-02-26 22:32:06 +00:00
#include "common/config-manager.h"
#include "common/debug.h"
#include "common/error.h"
#include "common/file.h"
#include "common/system.h"
#include "common/events.h"
#include "common/stream.h"
#include "engines/util.h"
2016-02-27 12:35:49 +00:00
#include "graphics/palette.h"
2016-02-26 22:32:06 +00:00
#include "adl/adl.h"
#include "adl/display.h"
#include "adl/parser.h"
namespace Adl {
AdlEngine::AdlEngine(OSystem *syst, const AdlGameDescription *gd) :
Engine(syst),
_gameDescription(gd),
2016-02-27 12:54:47 +00:00
_display(nullptr),
_parser(nullptr) {
2016-02-26 22:32:06 +00:00
}
AdlEngine::~AdlEngine() {
2016-02-27 12:54:47 +00:00
delete _parser;
2016-02-26 22:32:06 +00:00
delete _display;
}
Common::Error AdlEngine::run() {
initGraphics(560, 384, true);
byte palette[6 * 3] = {
0x00, 0x00, 0x00,
0xff, 0xff, 0xff,
0xc7, 0x34, 0xff,
0x38, 0xcb, 0x00,
0x00, 0x00, 0xff, // FIXME
0xff, 0xa5, 0x00 // FIXME
};
g_system->getPaletteManager()->setPalette(palette, 0, 6);
_display = new Display();
_parser = new Parser(*this, *_display);
runGame();
return Common::kNoError;
}
Common::String AdlEngine::readString(Common::ReadStream &stream, byte until) {
Common::String str;
while (1) {
byte b = stream.readByte();
if (stream.eos() || stream.err() || b == until)
break;
str += b;
};
return str;
}
void AdlEngine::printStrings(Common::SeekableReadStream &stream, int count) {
while (1) {
Common::String str = readString(stream);
_display->printString(str);
if (--count == 0)
break;
stream.seek(3, SEEK_CUR);
};
}
Common::String AdlEngine::getEngineString(int str) {
return _strings[str];
}
void AdlEngine::wordWrap(Common::String &str) {
uint end = 39;
while (1) {
if (str.size() <= end)
return;
while (str[end] != APPLECHAR(' '))
--end;
str.setChar(APPLECHAR('\r'), end);
end += 40;
}
}
void AdlEngine::printMessage(uint idx, bool wait) {
Common::String msg = _messages[idx - 1];
wordWrap(msg);
_display->printString(msg);
if (wait)
_display->delay(14 * 166018 / 1000);
}
void AdlEngine::printEngineMessage(EngineMessage msg) {
printMessage(getEngineMessage(msg));
}
void AdlEngine::readCommands(Common::ReadStream &stream, Commands &commands) {
while (1) {
Command command;
command.room = stream.readByte();
if (command.room == 0xff)
return;
command.verb = stream.readByte();
command.noun = stream.readByte();
byte scriptSize = stream.readByte() - 6;
command.numCond = stream.readByte();
command.numAct = stream.readByte();
for (uint i = 0; i < scriptSize; ++i)
command.script.push_back(stream.readByte());
if (stream.eos() || stream.err())
error("Failed to read commands");
commands.push_back(command);
}
}
void AdlEngine::takeItem(byte noun) {
Common::Array<Item>::iterator item;
for (item = _state.items.begin(); item != _state.items.end(); ++item) {
if (item->noun != noun || item->room != _state.room)
continue;
if (item->state == IDI_ITEM_DOESNT_MOVE) {
printEngineMessage(IDI_MSG_ITEM_DOESNT_MOVE);
return;
}
if (item->state == IDI_ITEM_MOVED) {
item->room = IDI_NONE;
return;
}
Common::Array<byte>::const_iterator pic;
for (pic = item->roomPictures.begin(); item->roomPictures.end(); ++pic) {
if (*pic == _state.rooms[_state.room].curPicture) {
item->room = IDI_NONE;
item->state = IDI_ITEM_MOVED;
return;
}
}
}
printEngineMessage(IDI_MSG_ITEM_NOT_HERE);
}
void AdlEngine::dropItem(byte noun) {
Common::Array<Item>::iterator item;
for (item = _state.items.begin(); item != _state.items.end(); ++item) {
if (item->noun != noun || item->room != IDI_NONE)
continue;
item->room = _state.room;
item->state = IDI_ITEM_MOVED;
return;
}
printEngineMessage(IDI_MSG_DONT_UNDERSTAND);
}
#define ARG(N) (command.script[offset + (N)])
void AdlEngine::doActions(const Command &command, byte noun, byte offset) {
for (uint i = 0; i < command.numAct; ++i) {
switch (ARG(0)) {
2016-02-28 14:56:17 +00:00
case IDO_ACT_VAR_ADD:
_state.vars[ARG(2)] += ARG(1);
offset += 3;
break;
2016-02-28 14:56:17 +00:00
case IDO_ACT_VAR_SUB:
_state.vars[ARG(2)] -= ARG(1);
offset += 3;
break;
2016-02-28 14:56:17 +00:00
case IDO_ACT_VAR_SET:
_state.vars[ARG(1)] = ARG(2);
offset += 3;
break;
2016-02-28 14:56:17 +00:00
case IDO_ACT_LIST_ITEMS: {
Common::Array<Item>::const_iterator item;
for (item = _state.items.begin(); item != _state.items.end(); ++item)
if (item->room == IDI_NONE)
printMessage(item->description);
++offset;
break;
}
2016-02-28 14:56:17 +00:00
case IDO_ACT_MOVE_ITEM:
_state.items[ARG(1) - 1].room = ARG(2);
offset += 3;
break;
2016-02-28 14:56:17 +00:00
case IDO_ACT_SET_ROOM:
_state.rooms[_state.room].curPicture = _state.rooms[_state.room].picture;
_state.room = ARG(1);
offset += 2;
break;
2016-02-28 14:56:17 +00:00
case IDO_ACT_SET_CUR_PIC:
_state.rooms[_state.room].curPicture = ARG(1);
offset += 2;
break;
2016-02-28 14:56:17 +00:00
case IDO_ACT_SET_PIC:
_state.rooms[_state.room].picture = _state.rooms[_state.room].curPicture = ARG(1);
offset += 2;
break;
2016-02-28 14:56:17 +00:00
case IDO_ACT_PRINT_MSG:
printMessage(ARG(1));
offset += 2;
break;
2016-02-28 14:56:17 +00:00
case IDO_ACT_SET_LIGHT:
_state.isDark = false;
++offset;
break;
2016-02-28 14:56:17 +00:00
case IDO_ACT_SET_DARK:
_state.isDark = true;
++offset;
break;
2016-02-28 14:56:17 +00:00
case IDO_ACT_SAVE:
warning("Save game not implemented");
++offset;
break;
2016-02-28 14:56:17 +00:00
case IDO_ACT_LOAD:
warning("Load game not implemented");
++offset;
break;
2016-02-28 14:56:17 +00:00
case IDO_ACT_RESTART: {
_display->printString(_strings[IDI_STR_PLAY_AGAIN]);
Common::String input = _display->inputString();
if (input.size() == 0 || input[0] != APPLECHAR('N')) {
warning("Restart game not implemented");
return;
}
// Fall-through
}
2016-02-28 14:56:17 +00:00
case IDO_ACT_QUIT:
printEngineMessage(IDI_MSG_THANKS_FOR_PLAYING);
quitGame();
return;
case IDO_ACT_PLACE_ITEM:
_state.items[ARG(1) - 1].room = ARG(2);
_state.items[ARG(1) - 1].position.x = ARG(3);
_state.items[ARG(1) - 1].position.y = ARG(4);
offset += 5;
break;
case IDO_ACT_SET_ITEM_PIC:
_state.items[ARG(2) - 1].picture = ARG(1);
offset += 3;
break;
2016-02-28 14:56:17 +00:00
case IDO_ACT_RESET_PIC:
_state.rooms[_state.room].curPicture = _state.rooms[_state.room].picture;
++offset;
break;
2016-02-28 14:56:17 +00:00
case IDO_ACT_GO_NORTH:
case IDO_ACT_GO_SOUTH:
case IDO_ACT_GO_EAST:
case IDO_ACT_GO_WEST:
case IDO_ACT_GO_UP:
case IDO_ACT_GO_DOWN: {
byte room = _state.rooms[_state.room].connections[ARG(0) - 0x15];
if (room == 0) {
printEngineMessage(IDI_MSG_CANT_GO_THERE);
return;
}
_state.rooms[_state.room].curPicture = _state.rooms[_state.room].picture;
_state.room = room;
return;
}
2016-02-28 14:56:17 +00:00
case IDO_ACT_TAKE_ITEM:
takeItem(noun);
++offset;
break;
2016-02-28 14:56:17 +00:00
case IDO_ACT_DROP_ITEM:
dropItem(noun);
++offset;
break;
2016-02-28 14:56:17 +00:00
case IDO_ACT_SET_ROOM_PIC:
_state.rooms[ARG(1)].picture = _state.rooms[ARG(1)].curPicture = ARG(2);
offset += 3;
break;
default:
error("Invalid action opcode %02x", ARG(0));
}
}
}
bool AdlEngine::checkCommand(const Command &command, byte verb, byte noun) {
if (command.room != IDI_NONE && command.room != _state.room)
return false;
2016-02-28 13:24:04 +00:00
if (command.verb != IDI_NONE && command.verb != verb)
return false;
2016-02-28 13:24:04 +00:00
if (command.noun != IDI_NONE && command.noun != noun)
return false;
uint offset = 0;
for (uint i = 0; i < command.numCond; ++i) {
switch (command.script[offset]) {
2016-02-28 14:56:17 +00:00
case IDO_CND_ITEM_IN_ROOM:
if (_state.items[ARG(1) - 1].room != ARG(2))
return false;
offset += 3;
break;
case IDO_CND_MOVES_GE:
if (ARG(1) > _state.moves)
return false;
offset += 2;
break;
2016-02-28 14:56:17 +00:00
case IDO_CND_VAR_EQ:
if (_state.vars[ARG(1)] != ARG(2))
return false;
offset += 3;
break;
2016-02-28 14:56:17 +00:00
case IDO_CND_CUR_PIC_EQ:
if (_state.rooms[_state.room].curPicture != ARG(1))
return false;
offset += 2;
break;
2016-02-28 14:56:17 +00:00
case IDO_CND_ITEM_PIC_EQ:
if (_state.items[ARG(1) - 1].picture != ARG(2))
return false;
offset += 3;
break;
default:
error("Invalid condition opcode %02x", command.script[offset]);
}
}
doActions(command, noun, offset);
return true;
}
#undef ARG
bool AdlEngine::doOneCommand(const Commands &commands, byte verb, byte noun) {
Commands::const_iterator cmd;
for (cmd = commands.begin(); cmd != commands.end(); ++cmd)
if (checkCommand(*cmd, verb, noun))
return true;
return false;
}
void AdlEngine::doAllCommands(const Commands &commands, byte verb, byte noun) {
Commands::const_iterator cmd;
for (cmd = commands.begin(); cmd != commands.end(); ++cmd)
checkCommand(*cmd, verb, noun);
}
void AdlEngine::clearScreen() {
_display->setMode(Display::kModeMixed);
_display->clear(0x00);
}
void AdlEngine::drawItems() {
Common::Array<Item>::const_iterator item;
uint dropped = 0;
for (item = _state.items.begin(); item != _state.items.end(); ++item) {
if (item->room != _state.room)
continue;
if (item->state == IDI_ITEM_MOVED) {
if (_state.rooms[_state.room].picture == _state.rooms[_state.room].curPicture) {
const Common::Point &p = _itemOffsets[dropped];
2016-02-28 16:21:09 +00:00
if (item->isLineArt)
_display->drawLineArt(_lineArt[item->picture - 1], p);
else
drawPic(item->picture, p);
++dropped;
}
continue;
}
Common::Array<byte>::const_iterator pic;
for (pic = item->roomPictures.begin(); pic != item->roomPictures.end(); ++pic) {
if (*pic == _state.rooms[_state.room].curPicture) {
2016-02-28 16:21:09 +00:00
if (item->isLineArt)
_display->drawLineArt(_lineArt[item->picture - 1], item->position);
else
drawPic(item->picture, item->position);
continue;
}
}
}
}
void AdlEngine::showRoom() {
if (!_state.isDark) {
drawPic(_state.rooms[_state.room].curPicture);
drawItems();
}
_display->decodeFrameBuffer();
printMessage(_state.rooms[_state.room].description, false);
}
2016-02-26 22:32:06 +00:00
AdlEngine *AdlEngine::create(GameType type, OSystem *syst, const AdlGameDescription *gd) {
switch(type) {
2016-02-27 09:46:37 +00:00
case kGameTypeHires1:
return HiRes1Engine__create(syst, gd);
2016-02-26 22:32:06 +00:00
default:
error("Unknown GameType");
}
}
} // End of namespace Adl