/* 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. * * $URL$ * $Id$ * */ #include "common/system.h" #include "common/config-manager.h" #include "parallaction/parallaction.h" #include "parallaction/sound.h" #include "parallaction/menu.h" namespace Parallaction { #define MOUSEARROW_WIDTH 16 #define MOUSEARROW_HEIGHT 16 #define MOUSECOMBO_WIDTH 32 // sizes for cursor + selected inventory item #define MOUSECOMBO_HEIGHT 32 int Parallaction_ns::init() { // Detect game if (!detectGame()) { GUIErrorMessage("No valid games were found in the specified directory."); return -1; } _screenWidth = 320; _screenHeight = 200; if (getPlatform() == Common::kPlatformPC) { _disk = new DosDisk_ns(this); } else { if (getFeatures() & GF_DEMO) { strcpy(_location._name, "fognedemo"); } _disk = new AmigaDisk_ns(this); _disk->selectArchive((getFeatures() & GF_DEMO) ? "disk0" : "disk1"); } if (getPlatform() == Common::kPlatformPC) { int midiDriver = MidiDriver::detectMusicDriver(MDT_MIDI | MDT_ADLIB | MDT_PREFER_MIDI); MidiDriver *driver = MidiDriver::createMidi(midiDriver); _soundMan = new DosSoundMan(this, driver); _soundMan->setMusicVolume(ConfMan.getInt("music_volume")); } else { _soundMan = new AmigaSoundMan(this); } initJobs(); initResources(); initFonts(); initCursors(); initOpcodes(); initParsers(); Parallaction::init(); return 0; } Parallaction_ns::~Parallaction_ns() { freeFonts(); _mouseComposedArrow->free(); delete _mouseComposedArrow; delete _commandsNames; delete _instructionNames; delete _locationStmt; } void Parallaction_ns::freeFonts() { delete _dialogueFont; delete _labelFont; delete _menuFont; return; } void Parallaction_ns::renderLabel(Graphics::Surface *cnv, char *text) { if (getPlatform() == Common::kPlatformAmiga) { cnv->create(_labelFont->getStringWidth(text) + 16, 10, 1); _labelFont->setColor(7); _labelFont->drawString((byte*)cnv->pixels + 1, cnv->w, text); _labelFont->drawString((byte*)cnv->pixels + 1 + cnv->w * 2, cnv->w, text); _labelFont->drawString((byte*)cnv->pixels + cnv->w, cnv->w, text); _labelFont->drawString((byte*)cnv->pixels + 2 + cnv->w, cnv->w, text); _labelFont->setColor(1); _labelFont->drawString((byte*)cnv->pixels + 1 + cnv->w, cnv->w, text); } else { cnv->create(_labelFont->getStringWidth(text), _labelFont->height(), 1); _labelFont->drawString((byte*)cnv->pixels, cnv->w, text); } } void Parallaction_ns::initCursors() { _mouseComposedArrow = _disk->loadPointer("pointer"); byte temp[MOUSEARROW_WIDTH*MOUSEARROW_HEIGHT]; memcpy(temp, _mouseArrow, MOUSEARROW_WIDTH*MOUSEARROW_HEIGHT); uint16 k = 0; for (uint16 i = 0; i < 4; i++) { for (uint16 j = 0; j < 64; j++) _mouseArrow[k++] = temp[i + j * 4]; } return; } void Parallaction_ns::setArrowCursor() { debugC(1, kDebugInput, "setting mouse cursor to arrow"); // this stuff is needed to avoid artifacts with labels and selected items when switching cursors hideLabel(kPriority15); _activeItem._id = 0; _system->setMouseCursor(_mouseArrow, MOUSEARROW_WIDTH, MOUSEARROW_HEIGHT, 0, 0, 0); _system->showMouse(true); } void Parallaction_ns::setInventoryCursor(int pos) { if (pos == -1) return; const InventoryItem *item = getInventoryItem(pos); if (item->_index == 0) return; _activeItem._id = item->_id; byte *v8 = (byte*)_mouseComposedArrow->pixels; // FIXME: destination offseting is not clear byte* s = _char._objs->getData(item->_index); byte* d = v8 + 7 + MOUSECOMBO_WIDTH * 7; for (uint i = 0; i < INVENTORYITEM_HEIGHT; i++) { memcpy(d, s, INVENTORYITEM_WIDTH); s += INVENTORYITEM_PITCH; d += MOUSECOMBO_WIDTH; } _system->setMouseCursor(v8, MOUSECOMBO_WIDTH, MOUSECOMBO_HEIGHT, 0, 0, 0); } void Parallaction_ns::callFunction(uint index, void* parm) { assert(index < 25); // magic value 25 is maximum # of callables for Nippon Safes (this->*_callables[index])(parm); } int Parallaction_ns::go() { _menu = new Menu(this); _menu->start(); char *v4 = strchr(_location._name, '.'); if (v4) { *v4 = '\0'; } _globalTable = _disk->loadTable("global"); _engineFlags &= ~kEngineChangeLocation; changeCharacter(_char.getName()); strcpy(_saveData1, _location._name); parseLocation(_location._name); if (_location._startPosition.x != -1000) { _char._ani._left = _location._startPosition.x; _char._ani._top = _location._startPosition.y; _char._ani._frame = _location._startFrame; _location._startPosition.y = -1000; _location._startPosition.x = -1000; }; runGame(); delete _menu; return 0; } /* changeLocation handles transitions between locations, and is able to display slides between one and the other. The input parameter 'location' exists in some flavours: 1 - [S].slide.[L]{.[C]} 2 - [L]{.[C]} where: [S] is the slide to be shown [L] is the location to switch to (immediately in case 2, or right after slide [S] in case 1) [C] is the character to be selected, and is optional The routine tells one form from the other by searching for the '.slide.' NOTE: there exists one script in which [L] is not used in the case 1, but its use is commented out, and would definitely crash the current implementation. */ void Parallaction_ns::changeLocation(char *location) { debugC(1, kDebugExec, "changeLocation(%s)", location); _soundMan->playLocationMusic(location); // WORKAROUND: this hideLabel has been added to avoid crashes caused by // execution of label jobs after a location switch. The other workaround in // Parallaction::runGame should have been rendered useless by this one. hideLabel(kPriority99); _hoverZone = NULL; if (_engineFlags & kEngineBlockInput) { setArrowCursor(); } _animations.remove(&_char._ani); freeLocation(); char buf[100]; strcpy(buf, location); Common::StringList list; char *tok = strtok(location, "."); while (tok) { list.push_back(tok); tok = strtok(NULL, "."); } if (list.size() < 1 || list.size() > 4) error("changeLocation: ill-formed location string '%s'", location); if (list.size() > 1) { if (list[1] == "slide") { showSlide(list[0].c_str()); _gfx->setFont(_menuFont); _gfx->displayCenteredString(14, _slideText[0]); // displays text on screen _gfx->updateScreen(); waitUntilLeftClick(); list.remove_at(0); // removes slide name list.remove_at(0); // removes 'slide' } // list is now only [L].{[C]} (see above comment) if (list.size() == 2) { changeCharacter(list[1].c_str()); } } _animations.push_front(&_char._ani); strcpy(_saveData1, list[0].c_str()); parseLocation(list[0].c_str()); _char._ani._oldPos.x = -1000; _char._ani._oldPos.y = -1000; _char._ani.field_50 = 0; if (_location._startPosition.x != -1000) { _char._ani._left = _location._startPosition.x; _char._ani._top = _location._startPosition.y; _char._ani._frame = _location._startFrame; _location._startPosition.y = -1000; _location._startPosition.x = -1000; } _gfx->copyScreen(Gfx::kBitBack, Gfx::kBitFront); _gfx->copyScreen(Gfx::kBitBack, Gfx::kBit2); _gfx->setBlackPalette(); _gfx->updateScreen(); if (_location._commands.size() > 0) { runCommands(_location._commands); runJobs(); _gfx->swapBuffers(); runJobs(); _gfx->swapBuffers(); } if (_location._comment) { doLocationEnterTransition(); } runJobs(); _gfx->swapBuffers(); _gfx->setPalette(_gfx->_palette); if (_location._aCommands.size() > 0) { runCommands(_location._aCommands); } if (_hasLocationSound) _soundMan->playSfx(_locationSound, 0, true); debugC(1, kDebugExec, "changeLocation() done"); return; } void Parallaction_ns::changeCharacter(const char *name) { debugC(1, kDebugExec, "changeCharacter(%s)", name); _char.setName(name); if (!scumm_stricmp(_char.getFullName(), _characterName1)) { debugC(3, kDebugExec, "changeCharacter: nothing done"); return; } // freeCharacter takes responsibility for checking // character for sanity before memory is freed freeCharacter(); Common::String oldArchive = _disk->selectArchive((getFeatures() & GF_LANG_MULT) ? "disk1" : "disk0"); _char._ani._cnv = _disk->loadFrames(_char.getFullName()); if (_char.getBaseName()[0] != '\0') { if (getPlatform() == Common::kPlatformAmiga && (getFeatures() & GF_LANG_MULT)) _disk->selectArchive("disk0"); _char._head = _disk->loadHead(_char.getBaseName()); _char._talk = _disk->loadTalk(_char.getBaseName()); _char._objs = _disk->loadObjects(_char.getBaseName()); _objectsNames = _disk->loadTable(_char.getBaseName()); _soundMan->playCharacterMusic(_char.getBaseName()); if (!(getFeatures() & GF_DEMO)) parseLocation("common"); } if (!oldArchive.empty()) _disk->selectArchive(oldArchive); strcpy(_characterName1, _char.getFullName()); debugC(3, kDebugExec, "changeCharacter: switch completed"); return; } void Parallaction_ns::initJobs() { static const JobFn jobs[] = { &Parallaction_ns::jobDisplayAnimations, &Parallaction_ns::jobEraseAnimations, &Parallaction_ns::jobDisplayDroppedItem, &Parallaction_ns::jobRemovePickedItem, &Parallaction_ns::jobRunScripts, &Parallaction_ns::jobWalk, &Parallaction_ns::jobDisplayLabel, &Parallaction_ns::jobEraseLabel, &Parallaction_ns::jobWaitRemoveJob, &Parallaction_ns::jobToggleDoor, &Parallaction_ns::jobShowInventory, &Parallaction_ns::jobHideInventory }; _jobsFn = jobs; } JobOpcode* Parallaction_ns::createJobOpcode(uint functionId, Job *job) { return new OpcodeImpl2(this, _jobsFn[functionId], job); } } // namespace Parallaction