scummvm/engines/m4/mads_logic.cpp
2010-07-29 11:04:54 +00:00

492 lines
15 KiB
C++

/* 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 "m4/m4.h"
#include "m4/mads_logic.h"
#include "m4/scene.h"
namespace M4 {
void MadsGameLogic::initialiseGlobals() {
// Clear the entire globals list
Common::set_to(&_madsVm->globals()->_globals[0], &_madsVm->globals()->_globals[TOTAL_NUM_VARIABLES], 0);
SET_GLOBAL(4, 8);
SET_GLOBAL(33, 1);
SET_GLOBAL(10, 0xFFFF);
SET_GLOBAL(13, 0xFFFF);
SET_GLOBAL(15, 0xFFFF);
SET_GLOBAL(19, 0xFFFF);
SET_GLOBAL(20, 0xFFFF);
SET_GLOBAL(21, 0xFFFF);
SET_GLOBAL(95, 0xFFFF);
// TODO: unknown sub call
// Put the values 0 through 3 in a random ordering in global slots 83 - 86
for (int idx = 0; idx < 4; ) {
int randVal = _madsVm->_random->getRandomNumber(4);
SET_GLOBAL(83 + idx, randVal);
// Check whether the given value has already been used
bool flag = false;
for (int idx2 = 0; idx2 < idx; ++idx2) {
if (randVal == GET_GLOBAL(83 + idx2))
flag = true;
}
if (!flag)
++idx;
}
// Put the values 0 through 3 in a random ordering in global slots 87 - 90
for (int idx = 0; idx < 4; ) {
int randVal = _madsVm->_random->getRandomNumber(3);
SET_GLOBAL(87 + idx, randVal);
// Check whether the given value has already been used
bool flag = false;
for (int idx2 = 0; idx2 < idx; ++idx2) {
if (randVal == GET_GLOBAL(87 + idx2))
flag = true;
}
if (!flag)
++idx;
}
// Miscellaneous global settings
SET_GLOBAL(120, 501);
SET_GLOBAL(121, 0xFFFF);
SET_GLOBAL(110, 0xFFFF);
SET_GLOBAL(119, 1);
SET_GLOBAL(134, 4);
SET_GLOBAL(190, 201);
SET_GLOBAL(191, 301);
SET_GLOBAL(192, 413);
SET_GLOBAL(193, 706);
SET_GLOBAL(194, 801);
SET_GLOBAL(195, 551);
SET_GLOBAL(196, 752);
// Fill out the globals 200 - 209 with unique random number values less than 10000
for (int idx = 0; idx < 10; ) {
int randVal = _madsVm->_random->getRandomNumber(9999);
SET_GLOBAL(200 + idx, randVal);
// Check whether the given value has already been used
bool flag = false;
for (int idx2 = 0; idx2 < idx; ++idx2) {
if (randVal == GET_GLOBAL(87 + idx2))
flag = true;
}
if (!flag)
++idx;
}
switch (_madsVm->globals()->_difficultyLevel) {
case 1:
// Very hard
SET_GLOBAL(35, 0);
// TODO: object set room
SET_GLOBAL(137, 5);
SET_GLOBAL(136, 0);
break;
case 2:
// Hard
SET_GLOBAL(35, 0);
// TODO: object set room
SET_GLOBAL(136, 0xFFFF);
SET_GLOBAL(137, 6);
break;
case 3:
// Easy
SET_GLOBAL(35, 2);
// TODO: object set room
break;
}
_madsVm->_player._direction = 8;
_madsVm->_player._newDirection = 8;
// TODO: unknown processing routine getting called for 'RXM' and 'ROX'
}
/*--------------------------------------------------------------------------*/
const char *MadsSceneLogic::formAnimName(char sepChar, int16 suffixNum) {
return MADSResourceManager::getResourceName(sepChar, _sceneNumber, EXTTYPE_NONE, NULL, suffixNum);
}
void MadsSceneLogic::getSceneSpriteSet() {
char prefix[100];
// Room change sound
_madsVm->_sound->playSound(5);
// Set up sprite set prefix to use
if ((_sceneNumber <= 103) || (_sceneNumber == 111)) {
if (_madsVm->globals()->_globals[0] == SEX_FEMALE)
strcpy(prefix, "ROX");
else
strcpy(prefix, "RXM");
} else if (_sceneNumber <= 110) {
strcpy(prefix, "RXSW");
_madsVm->globals()->_globals[0] = SEX_UNKNOWN;
} else if (_sceneNumber == 112)
strcpy(prefix, "");
_madsVm->globals()->playerSpriteChanged = true;
_madsVm->_player.loadSprites(prefix);
// if ((_sceneNumber == 105) ((_sceneNumber == 109) && (word_84800 != 0)))
// _madsVm->globals()->playerSpriteChanged = true;
_vm->_palette->setEntry(16, 0x38, 0xFF, 0xFF);
_vm->_palette->setEntry(17, 0x38, 0xb4, 0xb4);
}
void MadsSceneLogic::getAnimName() {
const char *newName = MADSResourceManager::getAAName(
((_sceneNumber <= 103) || (_sceneNumber > 111)) ? 0 : 1);
strcpy(_madsVm->scene()->_aaName, newName);
}
IntStorage &MadsSceneLogic::dataMap() {
return _madsVm->globals()->_dataMap;
}
/*--------------------------------------------------------------------------*/
uint16 MadsSceneLogic::loadSpriteSet(uint16 suffixNum, uint16 sepChar) {
assert(sepChar < 256);
const char *resName = formAnimName((char)sepChar, (int16)suffixNum);
return _madsVm->scene()->loadSceneSpriteSet(resName);
}
uint16 MadsSceneLogic::startReversibleSpriteSequence(uint16 srcSpriteIdx, bool flipped, int numTicks, int triggerCountdown, int timeoutTicks, int extraTicks) {
M4Sprite *spriteFrame = _madsVm->scene()->_spriteSlots.getSprite(srcSpriteIdx).getFrame(0);
uint8 depth = _madsVm->_rails->getDepth(Common::Point(spriteFrame->x + (spriteFrame->width() / 2),
spriteFrame->y + (spriteFrame->height() / 2)));
return _madsVm->scene()->_sequenceList.add(srcSpriteIdx, flipped, 1, triggerCountdown, timeoutTicks, extraTicks, numTicks, 0, 0,
true, 100, depth - 1, 1, ANIMTYPE_REVERSIBLE, 0, 0);
}
uint16 MadsSceneLogic::startCycledSpriteSequence(uint16 srcSpriteIdx, bool flipped, int numTicks, int triggerCountdown, int timeoutTicks, int extraTicks) {
M4Sprite *spriteFrame = _madsVm->scene()->_spriteSlots.getSprite(srcSpriteIdx).getFrame(0);
uint8 depth = _madsVm->_rails->getDepth(Common::Point(spriteFrame->x + (spriteFrame->width() / 2),
spriteFrame->y + (spriteFrame->height() / 2)));
return _madsVm->scene()->_sequenceList.add(srcSpriteIdx, flipped, 1, triggerCountdown, timeoutTicks, extraTicks, numTicks, 0, 0,
true, 100, depth - 1, 1, ANIMTYPE_CYCLED, 0, 0);
}
uint16 MadsSceneLogic::startSpriteSequence3(uint16 srcSpriteIdx, bool flipped, int numTicks, int triggerCountdown, int timeoutTicks, int extraTicks) {
M4Sprite *spriteFrame = _madsVm->scene()->_spriteSlots.getSprite(srcSpriteIdx).getFrame(0);
uint8 depth = _madsVm->_rails->getDepth(Common::Point(spriteFrame->x + (spriteFrame->width() / 2),
spriteFrame->y + (spriteFrame->height() / 2)));
return _madsVm->scene()->_sequenceList.add(srcSpriteIdx, flipped, 1, triggerCountdown, timeoutTicks, extraTicks, numTicks, 0, 0,
true, 100, depth - 1, -1, ANIMTYPE_CYCLED, 0, 0);
}
void MadsSceneLogic::activateHotspot(int idx, bool active) {
// TODO:
}
void MadsSceneLogic::lowRoomsEntrySound() {
if (!_madsVm->globals()->_config.musicFlag) {
_madsVm->_sound->playSound(2);
} else {
// Play different sounds for each of the rooms
switch (_madsVm->globals()->sceneNumber) {
case 101:
_madsVm->_sound->playSound(11);
break;
case 102:
_madsVm->_sound->playSound(12);
break;
case 103:
_madsVm->_sound->playSound(3);
_madsVm->_sound->playSound(25);
break;
case 104:
_madsVm->_sound->playSound(10);
break;
case 105:
if ((_madsVm->globals()->previousScene < 104) || (_madsVm->globals()->previousScene > 108))
_madsVm->_sound->playSound(10);
break;
case 106:
_madsVm->_sound->playSound(13);
break;
case 107:
_madsVm->_sound->playSound(3);
break;
case 108:
_madsVm->_sound->playSound(15);
break;
default:
break;
}
}
}
void MadsSceneLogic::getPlayerSpritesPrefix() {
_madsVm->_sound->playSound(5);
char oldName[80];
strcpy(oldName, _madsVm->_player._spritesPrefix);
if ((_madsVm->globals()->_nextSceneId <= 103) || (_madsVm->globals()->_nextSceneId == 111))
strcpy(_madsVm->_player._spritesPrefix, (_madsVm->globals()->_globals[0] == SEX_FEMALE) ? "ROX" : "RXM");
else if (_madsVm->globals()->_nextSceneId <= 110)
strcpy(_madsVm->_player._spritesPrefix, "RXSM");
else if (_madsVm->globals()->_nextSceneId == 112)
strcpy(_madsVm->_player._spritesPrefix, "");
if (strcmp(oldName, _madsVm->_player._spritesPrefix) != 0)
_madsVm->_player._spritesChanged = true;
if ((_madsVm->globals()->_nextSceneId == 105) ||
((_madsVm->globals()->_nextSceneId == 109) && (_madsVm->globals()->_globals[15] != 0))) {
// TODO: unknown flag setting
_madsVm->_player._spritesChanged = true;
}
_madsVm->_palette->setEntry(16, 40, 255, 255);
_madsVm->_palette->setEntry(17, 40, 180, 180);
}
void MadsSceneLogic::getPlayerSpritesPrefix2() {
_madsVm->_sound->playSound(5);
char oldName[80];
strcpy(oldName, _madsVm->_player._spritesPrefix);
if ((_madsVm->globals()->_nextSceneId == 213) || (_madsVm->globals()->_nextSceneId == 216))
strcpy(_madsVm->_player._spritesPrefix, "");
else if (_madsVm->globals()->_globals[0] == SEX_MALE)
strcpy(_madsVm->_player._spritesPrefix, "RXM");
else
strcpy(_madsVm->_player._spritesPrefix, "ROX");
// TODO: unknown flag setting for next scene Id > 212
if (strcmp(oldName, _madsVm->_player._spritesPrefix) != 0)
_madsVm->_player._spritesChanged = true;
/* if ((_madsVm->globals()->_nextSceneId == 203) && (_madsVm->globals()->_nextSceneId == 204) &&
(_madsVm->globals()->_globals[0x22] == 0))
// TODO: unknown flag set
*/
_madsVm->_palette->setEntry(16, 40, 255, 255);
_madsVm->_palette->setEntry(17, 40, 180, 180);
}
/*--------------------------------------------------------------------------*/
/**
* FIXME:
* Currently I'm only working at providing manual implementation of the first Rex Nebular scene.
* It will make more sense to convert the remaining game logic from the games into some
* kind of bytecode scripts
*/
void MadsSceneLogic::selectScene(int sceneNum) {
assert(sceneNum == 101);
_sceneNumber = sceneNum;
Common::set_to(&_spriteIndexes[0], &_spriteIndexes[50], 0);
}
void MadsSceneLogic::setupScene() {
// FIXME: This is the hardcoded logic for Rex scene 101 only
const char *animName = formAnimName('A', -1);
warning("anim - %s", animName);
// sub_1e754(animName, 3);
if ((_sceneNumber >= 101) && (_sceneNumber <= 112))
getPlayerSpritesPrefix();
else
getPlayerSpritesPrefix2();
getAnimName();
}
void MadsSceneLogic::enterScene() {
for (int i = 1; i <= 7; ++i)
_spriteIndexes[i - 1] = loadSpriteSet(i, 'x');
_spriteIndexes[7] = loadSpriteSet(0xFFFF, 'm');
_spriteIndexes[8] = loadSpriteSet(1, 'b');
_spriteIndexes[9] = loadSpriteSet(2, 'b');
_spriteIndexes[10] = loadSpriteSet(0, 'a');
_spriteIndexes[11] = loadSpriteSet(1, 'a');
_spriteIndexes[12] = loadSpriteSet(8, 'x');
_spriteIndexes[13] = loadSpriteSet(0, 'x');
_spriteIndexes[15] = startCycledSpriteSequence(_spriteIndexes[0], false, 5, 0, 0, 25);
_spriteIndexes[16] = startCycledSpriteSequence(_spriteIndexes[1], false, 4, 0, 1, 0);
_spriteIndexes[17] = startCycledSpriteSequence(_spriteIndexes[2], false, 4, 0, 1, 0);
_madsVm->scene()->_sequenceList.addSubEntry(_spriteIndexes[17], SM_FRAME_INDEX, 7, 70);
_spriteIndexes[18] = startReversibleSpriteSequence(_spriteIndexes[3], false, 10, 0, 0, 60);
_spriteIndexes[19] = startCycledSpriteSequence(_spriteIndexes[4], false, 5, 0, 1, 0);
_spriteIndexes[20] = startCycledSpriteSequence(_spriteIndexes[5], false, 10, 0, 2, 0);
_spriteIndexes[21] = startCycledSpriteSequence(_spriteIndexes[6], false, 6, 0, 0, 0);
_spriteIndexes[23] = startCycledSpriteSequence(_spriteIndexes[8], false, 6, 0, 10, 4);
_spriteIndexes[24] = startCycledSpriteSequence(_spriteIndexes[9], false, 6, 0, 32, 47);
activateHotspot(0x137, false); // SHIELD MODULATOR
// shield_panel_opened = 0;
if (_madsVm->globals()->previousScene != -1)
_madsVm->globals()->_globals[10] = 0;
if (_madsVm->globals()->previousScene != -2) {
_madsVm->_player._playerPos = Common::Point(100, 152);
}
if ((_madsVm->globals()->previousScene == 112) ||
((_madsVm->globals()->previousScene != -2) && (_spriteIndexes[29] != 0))) {
// Returning from probe cutscene?
_spriteIndexes[29] = -1;
_madsVm->_player._playerPos = Common::Point(161, 123);
_madsVm->_player._direction = 9;
// TODO: Extra flags setting
_spriteIndexes[25] = startCycledSpriteSequence(_spriteIndexes[10], false, 3, 0, 0, 0);
_madsVm->scene()->_sequenceList.setAnimRange(_spriteIndexes[25], 17, 17);
activateHotspot(0x47, false); // CHAIR
/*timer_unk1 = */_madsVm->scene()->_dynamicHotspots.add(0x47, 0x13F /*SIT_IN*/, -1,
Common::Rect(159, 84, 159+33, 84+36));
//if (_madsVm->globals()->previousScene == 112)
// room101Check();
} else {
_spriteIndexes[26] = startCycledSpriteSequence(_spriteIndexes[11], false, 6, 0, 0, 0);
_madsVm->scene()->_sequenceList.setDepth(_spriteIndexes[26], 4);
}
_madsVm->globals()->loadQuoteSet(0x31, 0x32, 0x37, 0x38, 0x39, -1);
if (_madsVm->globals()->_globals[10]) {
const char *animName = MADSResourceManager::getResourceName('S', 'e', EXTTYPE_AA, NULL, -1);
_madsVm->scene()->loadAnimation(animName, 71);
_madsVm->_player._playerPos = Common::Point(68, 140);
_madsVm->_player._direction = 4;
_madsVm->_player._visible = false;
_madsVm->_player._stepEnabled = false;
dataMap()[0x56FC] = 0;
dataMap()[0x5482] = 0;
dataMap()[0x5484] = 30;
}
_madsVm->globals()->_dataMap[0x5486] = 0;
lowRoomsEntrySound();
}
void MadsSceneLogic::doPreactions() {
warning("Still to do preactions logic");
}
void MadsSceneLogic::doAction() {
warning("Still to do actions logic");
}
void MadsSceneLogic::doSceneStep() {
// TODO: Sound handling
switch (_madsVm->scene()->_abortTimers) {
case 70:
_madsVm->_sound->playSound(9);
break;
case 71:
_madsVm->globals()->_globals[10] = 0;
_madsVm->_player._visible = true;
_madsVm->_player._stepEnabled = true;
_madsVm->_player._priorTimer = _madsVm->_currentTimer - _madsVm->_player._ticksAmount;
break;
case 72:
case 73:
// TODO: Method that should be scripted
break;
default:
break;
}
// Wake up message sequence
Animation *anim = _madsVm->scene()->activeAnimation();
if (anim) {
if ((anim->getCurrentFrame() == 6) && (dataMap()[0x5482] == 0)) {
dataMap()[0x5482]++;
_madsVm->scene()->_kernelMessages.add(Common::Point(63, dataMap()[0x5484]),
0x1110, 0, 0, 240, _madsVm->globals()->getQuote(49));
dataMap()[0x5484] += 14;
}
if ((anim->getCurrentFrame() == 7) && (dataMap()[0x5482] == 1)) {
dataMap()[0x5482]++;
_madsVm->scene()->_kernelMessages.add(Common::Point(63, dataMap()[0x5484]),
0x1110, 0, 0, 240, _madsVm->globals()->getQuote(54));
dataMap()[0x5484] += 14;
}
if ((anim->getCurrentFrame() == 10) && (dataMap()[0x5482] == 2)) {
dataMap()[0x5482]++;
_madsVm->scene()->_kernelMessages.add(Common::Point(63, dataMap()[0x5484]),
0x1110, 0, 0, 240, _madsVm->globals()->getQuote(55));
dataMap()[0x5484] += 14;
}
if ((anim->getCurrentFrame() == 17) && (dataMap()[0x5482] == 3)) {
dataMap()[0x5482]++;
_madsVm->scene()->_kernelMessages.add(Common::Point(63, dataMap()[0x5484]),
0x1110, 0, 0, 240, _madsVm->globals()->getQuote(56));
dataMap()[0x5484] += 14;
}
if ((anim->getCurrentFrame() == 20) && (dataMap()[0x5482] == 4)) {
dataMap()[0x5482]++;
_madsVm->scene()->_kernelMessages.add(Common::Point(63, dataMap()[0x5484]),
0x1110, 0, 0, 240, _madsVm->globals()->getQuote(50));
dataMap()[0x5484] += 14;
}
}
}
}