scummvm/engines/cine/various.cpp
Jordi Vilalta Prat d6c34bdc4b Fixed the spaces before tabs.
svn-id: r30667
2008-01-28 00:14:17 +00:00

2672 lines
60 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 "common/endian.h"
#include "common/events.h"
#include "common/savefile.h"
#include "cine/cine.h"
#include "cine/main_loop.h"
#include "cine/object.h"
#include "cine/sound.h"
#include "cine/bg_list.h"
#include "cine/various.h"
namespace Cine {
bool disableSystemMenu = false;
bool inMenu;
int16 commandVar3[4];
int16 commandVar1;
int16 commandVar2;
Message messageTable[NUM_MAX_MESSAGE];
uint16 var2;
uint16 var3;
uint16 var4;
uint16 var5;
int16 buildObjectListCommand(int16 param);
int16 canUseOnObject = 0;
void drawString(const char *string, byte param) {
}
void waitPlayerInput(void) {
}
void setTextWindow(uint16 param1, uint16 param2, uint16 param3, uint16 param4) {
}
uint16 errorVar;
byte menuVar;
bool fadeRequired;
uint16 allowPlayerInput;
uint16 checkForPendingDataLoadSwitch;
uint16 isDrawCommandEnabled;
uint16 waitForPlayerClick;
uint16 menuCommandLen;
bool _paletteNeedUpdate;
uint16 _messageLen;
byte _danKeysPressed;
int16 playerCommand;
char commandBuffer[80];
char currentPrcName[20];
char currentRelName[20];
char currentObjectName[20];
char currentMsgName[20];
char newPrcName[20];
char newRelName[20];
char newObjectName[20];
char newMsgName[20];
char currentBgName[8][15];
char currentCtName[15];
char currentPartName[15];
char currentDatName[30];
int16 saveVar2;
byte isInPause = 0;
uint16 defaultMenuBoxColor;
byte inputVar1 = 0;
uint16 inputVar2 = 0, inputVar3 = 0;
SelectedObjStruct currentSelectedObject;
static CommandeType currentSaveName[10];
int16 currentDisk;
static const int16 choiceResultTable[] = { 1, 1, 1, 2, 1, 1, 1 };
static const int16 subObjectUseTable[] = { 3, 3, 3, 3, 3, 0, 0 };
static const int16 canUseOnItemTable[] = { 1, 0, 0, 1, 1, 0, 0 };
CommandeType objectListCommand[20];
int16 objListTab[20];
uint16 exitEngine;
uint16 defaultMenuBoxColor2;
uint16 zoneData[NUM_MAX_ZONE];
void stopMusicAfterFadeOut(void) {
// if (g_sfxPlayer->_fadeOutCounter != 0 && g_sfxPlayer->_fadeOutCounter < 100) {
// g_sfxPlayer->stop();
// }
}
int16 stopObjectScript(int16 entryIdx) {
prcLinkedListStruct *currentHead = &objScriptList;
prcLinkedListStruct *tempHead = currentHead;
currentHead = tempHead->next;
while (currentHead) {
if (currentHead->scriptIdx == entryIdx) {
currentHead->scriptIdx = -1;
return 0;
}
currentHead = currentHead->next;
}
return -1;
}
void runObjectScript(int16 entryIdx) {
uint16 i;
prcLinkedListStruct *pNewElement;
prcLinkedListStruct *currentHead = &objScriptList;
prcLinkedListStruct *tempHead = currentHead;
currentHead = tempHead->next;
while (currentHead) {
tempHead = currentHead;
assert(tempHead);
currentHead = tempHead->next;
}
pNewElement = new prcLinkedListStruct;
assert(pNewElement);
pNewElement->next = tempHead->next;
tempHead->next = pNewElement;
// copy the stack into the script instance
for (i = 0; i < SCRIPT_STACK_SIZE; i++) {
pNewElement->stack[i] = 0;
}
pNewElement->compareResult = 0;
pNewElement->scriptPosition = 0;
pNewElement->scriptPtr = (byte *)relTable[entryIdx].data;
pNewElement->scriptIdx = entryIdx;
if (g_cine->getGameType() == Cine::GType_OS) {
pNewElement->localVars[0] = relTable[entryIdx].runCount;
++relTable[entryIdx].runCount;
}
computeScriptStack(pNewElement->scriptPtr, pNewElement->stack, relTable[entryIdx].size);
}
void addPlayerCommandMessage(int16 cmd) {
overlayHeadElement *currentHeadPtr = overlayHead.next;
overlayHeadElement *tempHead = &overlayHead;
overlayHeadElement *pNewElement;
while (currentHeadPtr) {
tempHead = currentHeadPtr;
currentHeadPtr = tempHead->next;
}
pNewElement = new overlayHeadElement;
assert(pNewElement);
pNewElement->next = tempHead->next;
tempHead->next = pNewElement;
pNewElement->objIdx = cmd;
pNewElement->type = 3;
if (!currentHeadPtr) {
currentHeadPtr = &overlayHead;
}
pNewElement->previous = currentHeadPtr->previous;
currentHeadPtr->previous = pNewElement;
}
int16 getRelEntryForObject(uint16 param1, uint16 param2, SelectedObjStruct *pSelectedObject) {
int16 i;
int16 di = -1;
for (i = 0; i < NUM_MAX_REL; i++) {
if (relTable[i].data && relTable[i].obj1Param1 == param1 && relTable[i].obj1Param2 == pSelectedObject->idx) {
if (param2 == 1) {
di = i;
} else if (param2 == 2) {
if (relTable[i].obj2Param == pSelectedObject->param) {
di = i;
}
}
}
if (di != -1)
break;
}
return di;
}
int16 getObjectUnderCursor(uint16 x, uint16 y) {
overlayHeadElement *currentHead = overlayHead.previous;
int16 objX, objY, frame, part, threshold, height, xdif, ydif;
while (currentHead) {
if (currentHead->type < 2) {
if (objectTable[currentHead->objIdx].name[0]) {
objX = objectTable[currentHead->objIdx].x;
objY = objectTable[currentHead->objIdx].y;
frame = ABS((int16)(objectTable[currentHead->objIdx].frame));
part = objectTable[currentHead->objIdx].part;
if (currentHead->type == 0) {
threshold = animDataTable[frame].var1;
} else {
threshold = animDataTable[frame].width / 2;
}
height = animDataTable[frame].height;
xdif = x - objX;
ydif = y - objY;
if ((xdif >= 0) && ((threshold << 4) > xdif) && (ydif > 0) && (ydif < height)) {
if (animDataTable[frame].ptr1) {
if (g_cine->getGameType() == Cine::GType_OS)
return currentHead->objIdx;
if (currentHead->type == 0) { // use generated mask
if (gfxGetBit(x - objX, y - objY, animDataTable[frame].ptr2, animDataTable[frame].width)) {
return currentHead->objIdx;
}
} else if (currentHead->type == 1) { // is mask
if (gfxGetBit(x - objX, y - objY, animDataTable[frame].ptr1, animDataTable[frame].width * 4)) {
return currentHead->objIdx;
}
}
}
}
}
}
currentHead = currentHead->previous;
}
return -1;
}
bool CineEngine::loadSaveDirectory(void) {
Common::InSaveFile *fHandle;
char tmp[80];
snprintf(tmp, 80, "%s.dir", _targetName.c_str());
fHandle = g_saveFileMan->openForLoading(tmp);
if (!fHandle) {
return false;
}
fHandle->read(currentSaveName, 10 * 20);
delete fHandle;
return true;
}
void loadScriptFromSave(Common::InSaveFile *fHandle, bool isGlobal) {
int16 i;
prcLinkedListStruct *newElement;
prcLinkedListStruct *currentHead = &globalScriptsHead;
prcLinkedListStruct *tempHead = currentHead;
currentHead = tempHead->next;
while (currentHead) {
tempHead = currentHead;
currentHead = tempHead->next;
}
newElement = new prcLinkedListStruct;
newElement->next = tempHead->next;
tempHead->next = newElement;
for (i = 0; i < SCRIPT_STACK_SIZE; i++)
newElement->stack[i] = fHandle->readUint16BE();
newElement->localVars.load(*fHandle);
newElement->compareResult = fHandle->readUint16BE();
newElement->scriptPosition = fHandle->readUint16BE();
newElement->scriptIdx = fHandle->readUint16BE();
if (isGlobal)
newElement->scriptPtr = scriptTable[newElement->scriptIdx].ptr;
else
newElement->scriptPtr = (byte *)relTable[newElement->scriptIdx].data;
}
void loadOverlayFromSave(Common::InSaveFile *fHandle) {
overlayHeadElement *newElement;
overlayHeadElement *currentHead = &overlayHead;
overlayHeadElement *tempHead = currentHead;
currentHead = tempHead->next;
while (currentHead) {
tempHead = currentHead;
currentHead = tempHead->next;
}
newElement = new overlayHeadElement;
fHandle->readUint32BE();
fHandle->readUint32BE();
newElement->objIdx = fHandle->readUint16BE();
newElement->type = fHandle->readUint16BE();
newElement->x = fHandle->readSint16BE();
newElement->y = fHandle->readSint16BE();
newElement->width = fHandle->readSint16BE();
newElement->color = fHandle->readSint16BE();
newElement->next = tempHead->next;
tempHead->next = newElement;
if (!currentHead)
currentHead = &overlayHead;
newElement->previous = currentHead->previous;
currentHead->previous = newElement;
}
void setupScriptList(bool isGlobal) {
prcLinkedListStruct *currentHead;
if (isGlobal)
currentHead = globalScriptsHead.next;
else
currentHead = objScriptList.next;
while (currentHead) {
if (isGlobal)
currentHead->scriptPtr = scriptTable[currentHead->scriptIdx].ptr;
else
currentHead->scriptPtr = (byte *)relTable[currentHead->scriptIdx].data;
currentHead = currentHead->next;
}
}
bool CineEngine::makeLoad(char *saveName) {
int16 i;
int16 size;
Common::InSaveFile *fHandle;
fHandle = g_saveFileMan->openForLoading(saveName);
if (!fHandle) {
drawString(otherMessages[0], 0);
waitPlayerInput();
// restoreScreen();
checkDataDisk(-1);
return false;
}
g_sound->stopMusic();
freeAnimDataTable();
unloadAllMasks();
// if (g_cine->getGameType() == Cine::GType_OS) {
// freeUnkList();
// }
freePrcLinkedList();
releaseObjectScripts();
freeBgIncrustList();
closePart();
for (i = 0; i < NUM_MAX_REL; i++) {
if (relTable[i].data) {
free(relTable[i].data);
relTable[i].data = NULL;
relTable[i].size = 0;
relTable[i].obj1Param1 = 0;
relTable[i].obj1Param2 = 0;
relTable[i].obj2Param = 0;
}
}
for (i = 0; i < NUM_MAX_SCRIPT; i++) {
if (scriptTable[i].ptr) {
free(scriptTable[i].ptr);
scriptTable[i].ptr = NULL;
scriptTable[i].size = 0;
}
}
for (i = 0; i < NUM_MAX_MESSAGE; i++) {
messageTable[i].len = 0;
if (messageTable[i].ptr) {
free(messageTable[i].ptr);
messageTable[i].ptr = NULL;
}
}
for (i = 0; i < NUM_MAX_OBJECT; i++) {
objectTable[i].part = 0;
objectTable[i].name[0] = 0;
objectTable[i].frame = 0;
objectTable[i].mask = 0;
objectTable[i].costume = 0;
}
globalVars.reset();
var2 = var3 = var4 = var5 = 0;
strcpy(newPrcName, "");
strcpy(newRelName, "");
strcpy(newObjectName, "");
strcpy(newMsgName, "");
strcpy(currentBgName[0], "");
strcpy(currentCtName, "");
allowPlayerInput = 0;
waitForPlayerClick = 0;
playerCommand = -1;
isDrawCommandEnabled = 0;
strcpy(commandBuffer, "");
globalVars[VAR_MOUSE_X_POS] = 0;
globalVars[VAR_MOUSE_Y_POS] = 0;
fadeRequired = false;
for (i = 0; i < 16; i++) {
c_palette[i] = 0;
}
checkForPendingDataLoadSwitch = 0;
currentDisk = fHandle->readUint16BE();
fHandle->read(currentPartName, 13);
fHandle->read(currentDatName, 13);
saveVar2 = fHandle->readSint16BE();
fHandle->read(currentPrcName, 13);
fHandle->read(currentRelName, 13);
fHandle->read(currentMsgName, 13);
fHandle->read(currentBgName[0], 13);
fHandle->read(currentCtName, 13);
fHandle->readUint16BE();
fHandle->readUint16BE();
for (i = 0; i < 255; i++) {
objectTable[i].x = fHandle->readSint16BE();
objectTable[i].y = fHandle->readSint16BE();
objectTable[i].mask = fHandle->readUint16BE();
objectTable[i].frame = fHandle->readSint16BE();
objectTable[i].costume = fHandle->readSint16BE();
fHandle->read(objectTable[i].name, 20);
objectTable[i].part = fHandle->readUint16BE();
}
for (i = 0; i < 16; i++) {
c_palette[i] = fHandle->readUint16BE();
}
for (i = 0; i < 16; i++) {
tempPalette[i] = fHandle->readUint16BE();
}
globalVars.load(*fHandle, NUM_MAX_VAR - 1);
for (i = 0; i < 16; i++) {
zoneData[i] = fHandle->readUint16BE();
}
for (i = 0; i < 4; i++) {
commandVar3[i] = fHandle->readUint16BE();
}
fHandle->read(commandBuffer, 0x50);
defaultMenuBoxColor = fHandle->readUint16BE();
bgVar0 = fHandle->readUint16BE();
allowPlayerInput = fHandle->readUint16BE();
playerCommand = fHandle->readSint16BE();
commandVar1 = fHandle->readSint16BE();
isDrawCommandEnabled = fHandle->readUint16BE();
var5 = fHandle->readUint16BE();
var4 = fHandle->readUint16BE();
var3 = fHandle->readUint16BE();
var2 = fHandle->readUint16BE();
commandVar2 = fHandle->readSint16BE();
defaultMenuBoxColor2 = fHandle->readUint16BE();
fHandle->readUint16BE();
fHandle->readUint16BE();
for (i = 0; i < NUM_MAX_ANIMDATA; i++) {
animDataTable[i].width = fHandle->readUint16BE();
animDataTable[i].var1 = fHandle->readUint16BE();
animDataTable[i].bpp = fHandle->readUint16BE();
animDataTable[i].height = fHandle->readUint16BE();
animDataTable[i].ptr1 = NULL;
animDataTable[i].ptr2 = NULL;
animDataTable[i].fileIdx = fHandle->readSint16BE();
animDataTable[i].frameIdx = fHandle->readSint16BE();
fHandle->read(animDataTable[i].name, 10);
animDataTable[i].refresh = (fHandle->readByte() != 0);
}
// TODO: handle screen params (really required ?)
fHandle->readUint16BE();
fHandle->readUint16BE();
fHandle->readUint16BE();
fHandle->readUint16BE();
fHandle->readUint16BE();
fHandle->readUint16BE();
size = fHandle->readSint16BE();
for (i = 0; i < size; i++) {
loadScriptFromSave(fHandle, true);
}
size = fHandle->readSint16BE();
for (i = 0; i < size; i++) {
loadScriptFromSave(fHandle, false);
}
size = fHandle->readSint16BE();
for (i = 0; i < size; i++) {
loadOverlayFromSave(fHandle);
}
size = fHandle->readSint16BE();
for (i = 0; i < size; i++) {
loadBgIncrustFromSave(fHandle);
}
delete fHandle;
checkDataDisk(currentDisk);
if (strlen(currentPartName)) {
loadPart(currentPartName);
}
if (strlen(currentPrcName)) {
loadPrc(currentPrcName);
setupScriptList(true);
}
if (strlen(currentRelName)) {
loadRel(currentRelName);
setupScriptList(false);
}
if (strlen(currentMsgName)) {
loadMsg(currentMsgName);
}
if (strlen(currentBgName[0])) {
loadBg(currentBgName[0]);
}
if (strlen(currentCtName)) {
loadCt(currentCtName);
}
loadResourcesFromSave();
reincrustAllBg();
setMouseCursor(MOUSE_CURSOR_NORMAL);
if (strlen(currentDatName)) {
/* i = saveVar2;
saveVar2 = 0;
loadMusic();
if (i) {
playMusic();
}*/
}
return true;
}
void makeSave(char *saveFileName) {
int16 i;
Common::OutSaveFile *fHandle;
fHandle = g_saveFileMan->openForSaving(saveFileName);
if (!fHandle) {
drawString(otherMessages[1], 0);
waitPlayerInput();
// restoreScreen();
checkDataDisk(-1);
return;
}
fHandle->writeUint16BE(currentDisk);
fHandle->write(currentPartName, 13);
fHandle->write(currentDatName, 13);
fHandle->writeUint16BE(saveVar2);
fHandle->write(currentPrcName, 13);
fHandle->write(currentRelName, 13);
fHandle->write(currentMsgName, 13);
fHandle->write(currentBgName[0], 13);
fHandle->write(currentCtName, 13);
fHandle->writeUint16BE(0xFF);
fHandle->writeUint16BE(0x20);
for (i = 0; i < 255; i++) {
fHandle->writeUint16BE(objectTable[i].x);
fHandle->writeUint16BE(objectTable[i].y);
fHandle->writeUint16BE(objectTable[i].mask);
fHandle->writeUint16BE(objectTable[i].frame);
fHandle->writeUint16BE(objectTable[i].costume);
fHandle->write(objectTable[i].name, 20);
fHandle->writeUint16BE(objectTable[i].part);
}
for (i = 0; i < 16; i++) {
fHandle->writeUint16BE(c_palette[i]);
}
for (i = 0; i < 16; i++) {
fHandle->writeUint16BE(tempPalette[i]);
}
globalVars.save(*fHandle, NUM_MAX_VAR - 1);
for (i = 0; i < 16; i++) {
fHandle->writeUint16BE(zoneData[i]);
}
for (i = 0; i < 4; i++) {
fHandle->writeUint16BE(commandVar3[i]);
}
fHandle->write(commandBuffer, 0x50);
fHandle->writeUint16BE(defaultMenuBoxColor);
fHandle->writeUint16BE(bgVar0);
fHandle->writeUint16BE(allowPlayerInput);
fHandle->writeUint16BE(playerCommand);
fHandle->writeUint16BE(commandVar1);
fHandle->writeUint16BE(isDrawCommandEnabled);
fHandle->writeUint16BE(var5);
fHandle->writeUint16BE(var4);
fHandle->writeUint16BE(var3);
fHandle->writeUint16BE(var2);
fHandle->writeUint16BE(commandVar2);
fHandle->writeUint16BE(defaultMenuBoxColor2);
fHandle->writeUint16BE(0xFF);
fHandle->writeUint16BE(0x1E);
for (i = 0; i < NUM_MAX_ANIMDATA; i++) {
fHandle->writeUint16BE(animDataTable[i].width);
fHandle->writeUint16BE(animDataTable[i].var1);
fHandle->writeUint16BE(animDataTable[i].bpp);
fHandle->writeUint16BE(animDataTable[i].height);
fHandle->writeSint16BE(animDataTable[i].fileIdx);
fHandle->writeSint16BE(animDataTable[i].frameIdx);
fHandle->write(animDataTable[i].name, 10);
// Horrifyingly, cinE used to dump the entire struct to the
// save file, including the data pointers. While these pointers
// would be invalid after loading, the loadResourcesFromSave()
// function would still test if ptr1 was non-NULL, presumably
// to see if the object was present in the room.
fHandle->writeByte(animDataTable[i].ptr1 ? 1 : 0);
}
fHandle->writeUint16BE(0); // Screen params, unhandled
fHandle->writeUint16BE(0);
fHandle->writeUint16BE(0);
fHandle->writeUint16BE(0);
fHandle->writeUint16BE(0);
fHandle->writeUint16BE(0);
{
int16 numScript = 0;
prcLinkedListStruct *currentHead = globalScriptsHead.next;
while (currentHead) {
numScript++;
currentHead = currentHead->next;
}
fHandle->writeUint16BE(numScript);
// actual save
currentHead = globalScriptsHead.next;
while (currentHead) {
for (i = 0; i < SCRIPT_STACK_SIZE; i++) {
fHandle->writeUint16BE(currentHead->stack[i]);
}
currentHead->localVars.save(*fHandle);
fHandle->writeUint16BE(currentHead->compareResult);
fHandle->writeUint16BE(currentHead->scriptPosition);
fHandle->writeUint16BE(currentHead->scriptIdx);
currentHead = currentHead->next;
}
}
{
int16 numScript = 0;
prcLinkedListStruct *currentHead = objScriptList.next;
while (currentHead) {
numScript++;
currentHead = currentHead->next;
}
fHandle->writeUint16BE(numScript);
// actual save
currentHead = objScriptList.next;
while (currentHead) {
for (i = 0; i < SCRIPT_STACK_SIZE; i++) {
fHandle->writeUint16BE(currentHead->stack[i]);
}
currentHead->localVars.save(*fHandle);
fHandle->writeUint16BE(currentHead->compareResult);
fHandle->writeUint16BE(currentHead->scriptPosition);
fHandle->writeUint16BE(currentHead->scriptIdx);
currentHead = currentHead->next;
}
}
{
int16 numScript = 0;
overlayHeadElement *currentHead = overlayHead.next;
while (currentHead) {
numScript++;
currentHead = currentHead->next;
}
fHandle->writeUint16BE(numScript);
// actual save
currentHead = overlayHead.next;
while (currentHead) {
fHandle->writeUint32BE(0);
fHandle->writeUint32BE(0);
fHandle->writeUint16BE(currentHead->objIdx);
fHandle->writeUint16BE(currentHead->type);
fHandle->writeSint16BE(currentHead->x);
fHandle->writeSint16BE(currentHead->y);
fHandle->writeSint16BE(currentHead->width);
fHandle->writeSint16BE(currentHead->color);
currentHead = currentHead->next;
}
}
int numBgIncrustList = 0;
BGIncrustList *bgIncrustPtr = bgIncrustList;
while (bgIncrustPtr) {
numBgIncrustList++;
bgIncrustPtr = bgIncrustPtr->next;
}
fHandle->writeUint16BE(numBgIncrustList);
bgIncrustPtr = bgIncrustList;
while (bgIncrustPtr) {
fHandle->writeUint32BE(0); // next
fHandle->writeUint32BE(0); // unkPtr
fHandle->writeUint16BE(bgIncrustPtr->objIdx);
fHandle->writeUint16BE(bgIncrustPtr->param);
fHandle->writeUint16BE(bgIncrustPtr->x);
fHandle->writeUint16BE(bgIncrustPtr->y);
fHandle->writeUint16BE(bgIncrustPtr->frame);
fHandle->writeUint16BE(bgIncrustPtr->part);
bgIncrustPtr = bgIncrustPtr->next;
}
delete fHandle;
setMouseCursor(MOUSE_CURSOR_NORMAL);
}
void CineEngine::makeSystemMenu(void) {
int16 numEntry, systemCommand;
int16 mouseX, mouseY, mouseButton;
int16 selectedSave;
if (!disableSystemMenu) {
inMenu = true;
do {
manageEvents();
getMouseData(mouseUpdateStatus, (uint16 *)&mouseButton, (uint16 *)&mouseX, (uint16 *)&mouseY);
} while (mouseButton);
numEntry = 6;
if (!allowPlayerInput) {
numEntry--;
}
systemCommand = makeMenuChoice(systemMenu, numEntry, mouseX, mouseY, 140);
switch (systemCommand) {
case 0:
{
drawString(otherMessages[2], 0);
waitPlayerInput();
break;
}
case 1:
{
getMouseData(mouseUpdateStatus, (uint16 *)&mouseButton, (uint16 *)&mouseX, (uint16 *)&mouseY);
if (!makeMenuChoice(confirmMenu, 2, mouseX, mouseY + 8, 100)) {
//reinitEngine();
}
break;
}
case 2:
{
getMouseData(mouseUpdateStatus, (uint16 *)&mouseButton, (uint16 *)&mouseX, (uint16 *)&mouseY);
if (!makeMenuChoice(confirmMenu, 2, mouseX, mouseY + 8, 100)) {
exitEngine = 1;
}
break;
}
case 3: // Select save drive... change ?
{
break;
}
case 4: // load game
{
if (loadSaveDirectory()) {
// int16 selectedSave;
getMouseData(mouseUpdateStatus, (uint16 *)&mouseButton, (uint16 *)&mouseX, (uint16 *)&mouseY);
selectedSave = makeMenuChoice(currentSaveName, 10, mouseX, mouseY + 8, 180);
if (selectedSave >= 0) {
char saveNameBuffer[256];
sprintf(saveNameBuffer, "%s.%1d", _targetName.c_str(), selectedSave);
getMouseData(mouseUpdateStatus, (uint16 *)&mouseButton, (uint16 *)&mouseX, (uint16 *)&mouseY);
if (!makeMenuChoice(confirmMenu, 2, mouseX, mouseY + 8, 100)) {
char loadString[256];
sprintf(loadString, otherMessages[3], currentSaveName[selectedSave]);
drawString(loadString, 0);
makeLoad(saveNameBuffer);
} else {
drawString(otherMessages[4], 0);
waitPlayerInput();
checkDataDisk(-1);
}
} else {
drawString(otherMessages[4], 0);
waitPlayerInput();
checkDataDisk(-1);
}
} else {
drawString(otherMessages[5], 0);
waitPlayerInput();
checkDataDisk(-1);
}
break;
}
case 5:
{
loadSaveDirectory();
selectedSave = makeMenuChoice(currentSaveName, 10, mouseX, mouseY + 8, 180);
if (selectedSave >= 0) {
char saveFileName[256];
char saveName[20];
saveName[0] = 0;
if (!makeTextEntryMenu(otherMessages[6], saveName, 20, 120))
break;
strncpy(currentSaveName[selectedSave], saveName, 20);
sprintf(saveFileName, "%s.%1d", _targetName.c_str(), selectedSave);
getMouseData(mouseUpdateStatus, (uint16 *)&mouseButton, (uint16 *)&mouseX, (uint16 *)&mouseY);
if (!makeMenuChoice(confirmMenu, 2, mouseX, mouseY + 8, 100)) {
char saveString[256], tmp[80];
snprintf(tmp, 80, "%s.dir", _targetName.c_str());
Common::OutSaveFile *fHandle = g_saveFileMan->openForSaving(tmp);
if (!fHandle) {
warning("Unable to open file %s for saving", tmp);
break;
}
fHandle->write(currentSaveName, 200);
delete fHandle;
sprintf(saveString, otherMessages[3], currentSaveName[selectedSave]);
drawString(saveString, 0);
makeSave(saveFileName);
checkDataDisk(-1);
} else {
drawString(otherMessages[4], 0);
waitPlayerInput();
checkDataDisk(-1);
}
}
break;
}
}
inMenu = false;
}
}
int drawChar(byte character, int16 x, int16 y) {
if (character == ' ') {
x += 5;
} else {
byte characterWidth = fontParamTable[character].characterWidth;
if (characterWidth) {
byte characterIdx = fontParamTable[character].characterIdx;
drawSpriteRaw(textTable[characterIdx][0], textTable[characterIdx][1], 2, 8, page1Raw, x, y);
x += characterWidth + 1;
}
}
return x;
}
void drawMessageBox(int16 x, int16 y, int16 width, int16 currentY, int16 offset, int16 color, byte* page) {
gfxDrawLine(x + offset, y + offset, x + width - offset, y + offset, color, page); // top
gfxDrawLine(x + offset, currentY + 4 - offset, x + width - offset, currentY + 4 - offset, color, page); // bottom
gfxDrawLine(x + offset, y + offset, x + offset, currentY + 4 - offset, color, page); // left
gfxDrawLine(x + width - offset, y + offset, x + width - offset, currentY + 4 - offset, color, page); // right
}
void drawDoubleMessageBox(int16 x, int16 y, int16 width, int16 currentY, int16 color, byte* page) {
drawMessageBox(x, y, width, currentY, 1, 0, page);
drawMessageBox(x, y, width, currentY, 0, color, page);
}
void makeTextEntry(const CommandeType commandList[], uint16 height, uint16 X, uint16 Y, uint16 width) {
byte color = 2;
byte color2 = defaultMenuBoxColor2;
int16 paramY = (height * 9) + 10;
int16 currentX, currentY;
int16 i;
uint16 j;
byte currentChar;
if (X + width > 319) {
X = 319 - width;
}
if (Y + paramY > 199) {
Y = 199 - paramY;
}
hideMouse();
blitRawScreen(page1Raw);
gfxDrawPlainBoxRaw(X, Y, X + width, Y + 4, color2, page1Raw);
currentX = X + 4;
currentY = Y + 4;
for (i = 0; i < height; i++) {
gfxDrawPlainBoxRaw(X, currentY, X + width, currentY + 9, color2, page1Raw);
currentX = X + 4;
for (j = 0; j < strlen(commandList[i]); j++) {
currentChar = commandList[i][j];
currentX = drawChar(currentChar, currentX, currentY);
}
currentY += 9;
}
gfxDrawPlainBoxRaw(X, currentY, X + width, currentY + 4, color2, page1Raw); // bottom part
drawDoubleMessageBox(X, Y, width, currentY, color, page1Raw);
blitRawScreen(page1Raw);
}
void processInventory(int16 x, int16 y) {
int16 listSize = buildObjectListCommand(-2);
uint16 button;
if (!listSize)
return;
makeTextEntry(objectListCommand, listSize, x, y, 140);
do {
manageEvents();
getMouseData(mouseUpdateStatus, &button, &dummyU16, &dummyU16);
} while (!button);
}
int16 buildObjectListCommand(int16 param) {
int16 i = 0, j = 0;
for (i = 0; i < 20; i++) {
objectListCommand[i][0] = 0;
}
for (i = 0; i < 255; i++) {
if (objectTable[i].name[0] && objectTable[i].costume == param) {
strcpy(objectListCommand[j], objectTable[i].name);
objListTab[j] = i;
j++;
}
}
return j;
}
int16 selectSubObject(int16 x, int16 y, int16 param) {
int16 listSize = buildObjectListCommand(param);
int16 selectedObject;
bool osExtras = g_cine->getGameType() == Cine::GType_OS;
if (!listSize) {
return -2;
}
selectedObject = makeMenuChoice(objectListCommand, listSize, x, y, 140, osExtras);
if (selectedObject == -1)
return -1;
if (osExtras) {
if (selectedObject >= 8000) {
return objListTab[selectedObject - 8000] + 8000;
}
}
return objListTab[selectedObject];
}
void makeCommandLine(void) {
uint16 x, y;
commandVar1 = 0;
commandVar2 = -10;
if (playerCommand != -1) {
strcpy(commandBuffer, defaultActionCommand[playerCommand]);
} else {
strcpy(commandBuffer, "");
}
if ((playerCommand != -1) && (choiceResultTable[playerCommand] == 2)) { // need object selection ?
int16 si;
getMouseData(mouseUpdateStatus, &dummyU16, &x, &y);
if (g_cine->getGameType() == Cine::GType_FW) {
si = selectSubObject(x, y + 8, -2);
} else {
si = selectSubObject(x, y + 8, -subObjectUseTable[playerCommand]);
}
if (si < 0) {
playerCommand = -1;
strcpy(commandBuffer, "");
} else {
if (g_cine->getGameType() == Cine::GType_OS) {
if (si >= 8000) {
si -= 8000;
canUseOnObject = canUseOnItemTable[playerCommand];
} else {
canUseOnObject = 0;
}
}
commandVar3[0] = si;
commandVar1 = 1;
strcat(commandBuffer, " ");
strcat(commandBuffer, objectTable[commandVar3[0]].name);
strcat(commandBuffer, " ");
strcat(commandBuffer, commandPrepositionOn);
}
} else {
if (playerCommand == 2) {
getMouseData(mouseUpdateStatus, &dummyU16, &x, &y);
processInventory(x, y + 8);
playerCommand = -1;
commandVar1 = 0;
strcpy(commandBuffer, "");
}
}
if (g_cine->getGameType() == Cine::GType_OS) {
if (playerCommand != -1 && canUseOnObject != 0) { // call use on sub object
int16 si;
getMouseData(mouseUpdateStatus, &dummyU16, &x, &y);
si = selectSubObject(x, y + 8, -subObjectUseTable[playerCommand]);
if (si) {
if (si >= 8000) {
si -= 8000;
}
commandVar3[commandVar1] = si;
commandVar1++;
// TODO: add command message draw
}
isDrawCommandEnabled = 1;
if (playerCommand != -1 && choiceResultTable[playerCommand] == commandVar1) {
SelectedObjStruct obj;
obj.idx = commandVar3[0];
obj.param = commandVar3[1];
int16 di = getRelEntryForObject(playerCommand, commandVar1, &obj);
if (di != -1) {
runObjectScript(di);
}
}
}
}
if (!disableSystemMenu) {
isDrawCommandEnabled = 1;
}
}
uint16 needMouseSave = 0;
uint16 menuVar4 = 0;
uint16 menuVar5 = 0;
int16 makeMenuChoice(const CommandeType commandList[], uint16 height, uint16 X, uint16 Y,
uint16 width, bool recheckValue) {
byte color = 2;
byte color2 = defaultMenuBoxColor2;
int16 paramY;
int16 currentX, currentY;
int16 i;
uint16 button;
int16 var_A;
int16 di;
uint16 j;
int16 mouseX, mouseY;
int16 var_16;
int16 var_14;
int16 currentSelection, oldSelection;
int16 var_4;
byte currentChar;
if (disableSystemMenu)
return -1;
paramY = (height * 9) + 10;
if (X + width > 319) {
X = 319 - width;
}
if (Y + paramY > 199) {
Y = 199 - paramY;
}
hideMouse();
blitRawScreen(page1Raw);
gfxDrawPlainBoxRaw(X, Y, X + width, Y + 4, color2, page1Raw);
currentX = X + 4;
currentY = Y + 4;
for (i = 0; i < height; i++) {
gfxDrawPlainBoxRaw(X, currentY, X + width, currentY + 9, color2, page1Raw);
currentX = X + 4;
for (j = 0; j < strlen(commandList[i]); j++) {
currentChar = commandList[i][j];
currentX = drawChar(currentChar, currentX, currentY);
}
currentY += 9;
}
gfxDrawPlainBoxRaw(X, currentY, X + width, currentY + 4, color2, page1Raw); // bottom part
drawDoubleMessageBox(X, Y, width, currentY, color, page1Raw);
blitRawScreen(page1Raw);
do {
manageEvents();
getMouseData(mouseUpdateStatus, &button, &dummyU16, &dummyU16);
} while (button);
var_A = 0;
currentSelection = 0;
di = currentSelection * 9 + Y + 4;
gfxDrawPlainBoxRaw(X + 2, di - 1, X + width - 2, di + 7, 0, page1Raw); // draw black box behind selection
currentX = X + 4;
for (j = 0; j < strlen(commandList[currentSelection]); j++) {
currentChar = commandList[currentSelection][j];
currentX = drawChar(currentChar, currentX, di);
}
blitRawScreen(page1Raw);
manageEvents();
getMouseData(mouseUpdateStatus, &button, (uint16 *)&mouseX, (uint16 *)&mouseY);
var_16 = mouseX;
var_14 = mouseY;
menuVar = 0;
do {
manageEvents();
getMouseData(mouseUpdateStatus, &button, (uint16 *)&mouseX, (uint16 *)&mouseY);
if (button) {
var_A = 1;
}
oldSelection = currentSelection;
if (needMouseSave) {
for (j = 0; j < 3; j++) {
mainLoopSub6();
}
if (menuVar4 && currentSelection > 0) { // go up
currentSelection--;
}
if (menuVar5) { // go down
if (height - 1 > currentSelection) {
currentSelection++;
}
}
} else {
if (mouseX > X && mouseX < X + width && mouseY > Y && mouseY < Y + height * 9) {
currentSelection = (mouseY - (Y + 4)) / 9;
if (currentSelection < 0)
currentSelection = 0;
if (currentSelection >= height)
currentSelection = height - 1;
}
}
if (currentSelection != oldSelection) { // old != new
if (needMouseSave) {
hideMouse();
}
di = oldSelection * 9 + Y + 4;
gfxDrawPlainBoxRaw(X + 2, di - 1, X + width - 2, di + 7, color2, page1Raw); // restore color
currentX = X + 4;
for (j = 0; j < strlen(commandList[oldSelection]); j++) {
currentChar = commandList[oldSelection][j];
currentX = drawChar(currentChar, currentX, di);
}
di = currentSelection * 9 + Y + 4;
gfxDrawPlainBoxRaw(X + 2, di - 1, X + width - 2, di + 7, 0, page1Raw); // black new
currentX = X + 4;
for (j = 0; j < strlen(commandList[currentSelection]); j++) {
currentChar = commandList[currentSelection][j];
currentX = drawChar(currentChar, currentX, di);
}
blitRawScreen(page1Raw);
// if (needMouseSave) {
// gfxRedrawMouseCursor();
// }
}
} while (!var_A);
assert(!needMouseSave);
var_4 = button;
menuVar = 0;
do {
manageEvents();
getMouseData(mouseUpdateStatus, &button, &dummyU16, &dummyU16);
} while (button);
if (var_4 == 2) { // recheck
if (!recheckValue)
return -1;
else
return currentSelection + 8000;
}
return currentSelection;
}
void drawMenuBox(char *command, int16 x, int16 y) {
byte j;
byte lColor = 2;
hideMouse();
gfxDrawPlainBoxRaw(x, y, x + 300, y + 10, 0, page2Raw);
drawMessageBox(x, y, 300, y + 6, -1, lColor, page2Raw);
x += 2;
y += 2;
for (j = 0; j < strlen(command); j++) {
byte currentChar = command[j];
if (currentChar == ' ') {
x += 5;
} else {
byte characterWidth = fontParamTable[currentChar].characterWidth;
if (characterWidth) {
byte characterIdx = fontParamTable[currentChar].characterIdx;
drawSpriteRaw(textTable[characterIdx][0], textTable[characterIdx][1], 2, 8, page2Raw, x, y);
x += characterWidth + 1;
}
}
}
// gfxRedrawMouseCursor();
}
void makeActionMenu(void) {
uint16 mouseButton;
uint16 mouseX;
uint16 mouseY;
inMenu = true;
getMouseData(mouseUpdateStatus, &mouseButton, &mouseX, &mouseY);
if (g_cine->getGameType() == Cine::GType_OS) {
playerCommand = makeMenuChoice(defaultActionCommand, 6, mouseX, mouseY, 70, true);
if (playerCommand >= 8000) {
playerCommand -= 8000;
canUseOnObject = 1;
}
} else {
playerCommand = makeMenuChoice(defaultActionCommand, 6, mouseX, mouseY, 70);
}
inMenu = false;
}
uint16 executePlayerInput(void) {
uint16 var_5E;
uint16 var_2;
uint16 mouseX, mouseY, mouseButton;
uint16 currentEntry = 0;
uint16 di = 0;
canUseOnObject = 0;
if (isInPause) {
drawString(otherMessages[2], 0);
waitPlayerInput();
isInPause = 0;
}
if (allowPlayerInput) {
if (isDrawCommandEnabled) {
drawMenuBox(commandBuffer, 10, defaultMenuBoxColor);
isDrawCommandEnabled = 0;
}
getMouseData(mouseUpdateStatus, &mouseButton, &mouseX, &mouseY);
while (mouseButton && currentEntry < 200) {
if (mouseButton & 1) {
di |= 1;
}
if (mouseButton & 2) {
di |= 2;
}
getMouseData(mouseUpdateStatus, &mouseButton, &mouseX, &mouseY);
currentEntry++;
}
if (di) {
mouseButton = di;
}
if (playerCommand != -1) {
if (mouseButton & 1) {
if (mouseButton & 2) {
g_cine->makeSystemMenu();
} else {
int16 si;
do {
manageEvents();
getMouseData(mouseUpdateStatus, &mouseButton, &dummyU16, &dummyU16);
} while (mouseButton);
si = getObjectUnderCursor(mouseX,
mouseY);
if (si != -1) {
commandVar3[commandVar1] = si;
commandVar1++;
strcat(commandBuffer, " ");
strcat(commandBuffer, objectTable[si].name);
isDrawCommandEnabled = 1;
if (choiceResultTable[playerCommand] == commandVar1) {
int16 relEntry;
drawMenuBox(commandBuffer, 10, defaultMenuBoxColor);
SelectedObjStruct obj;
obj.idx = commandVar3[0];
obj.param = commandVar3[1];
relEntry = getRelEntryForObject(playerCommand, commandVar1, &obj);
if (relEntry != -1) {
runObjectScript(relEntry);
} else {
addPlayerCommandMessage(playerCommand);
}
playerCommand = -1;
commandVar1 = 0;
strcpy(commandBuffer, "");
}
} else {
globalVars[VAR_MOUSE_X_POS] = mouseX;
globalVars[VAR_MOUSE_Y_POS] = mouseY;
}
}
} else {
if (mouseButton & 2) {
if (mouseButton & 1) {
g_cine->makeSystemMenu();
}
makeActionMenu();
makeCommandLine();
} else {
int16 objIdx;
objIdx = getObjectUnderCursor(mouseX, mouseY);
if (commandVar2 != objIdx) {
if (objIdx != -1) {
char command[256];
strcpy(command, commandBuffer);
strcat(command, " ");
strcat(command, objectTable[objIdx].name);
drawMenuBox(command, 10, defaultMenuBoxColor);
} else {
isDrawCommandEnabled = 1;
}
}
commandVar2 = objIdx;
}
}
} else {
if (mouseButton & 2) {
if (!(mouseButton & 1)) {
if (g_cine->getGameType() == Cine::GType_OS) {
playerCommand = makeMenuChoice(defaultActionCommand, 6, mouseX, mouseY, 70, true);
if (playerCommand >= 8000) {
playerCommand -= 8000;
canUseOnObject = 1;
}
} else {
playerCommand = makeMenuChoice(defaultActionCommand, 6, mouseX, mouseY, 70);
}
makeCommandLine();
} else {
g_cine->makeSystemMenu();
}
} else {
if (mouseButton & 1) {
if (!(mouseButton & 2)) {
int16 objIdx;
int16 relEntry;
globalVars[VAR_MOUSE_X_POS] = mouseX;
if (!mouseX) {
globalVars[VAR_MOUSE_X_POS]++;
}
globalVars[VAR_MOUSE_Y_POS] = mouseY;
objIdx = getObjectUnderCursor(mouseX, mouseY);
if (objIdx != -1) {
currentSelectedObject.idx = objIdx;
currentSelectedObject.param = -1;
relEntry = getRelEntryForObject(6, 1, &currentSelectedObject);
if (relEntry != -1) {
runObjectScript(relEntry);
}
}
} else {
g_cine->makeSystemMenu();
}
}
}
}
} else {
di = 0;
getMouseData(mouseUpdateStatus, &mouseButton, &mouseX, &mouseY);
while (mouseButton) {
if (mouseButton & 1) {
di |= 1;
}
if (mouseButton & 2) {
di |= 2;
}
manageEvents();
getMouseData(mouseUpdateStatus, &mouseButton, &mouseX, &mouseY);
}
if (di) {
mouseButton = di;
}
if ((mouseButton & 1) && (mouseButton & 2)) {
g_cine->makeSystemMenu();
}
}
var_2 = menuVar & 0x7F;
var_5E = var_2;
if (menuVar & 0x80) {
var_5E = 0;
var_2 = 0;
}
if (inputVar1 && allowPlayerInput) { // use keyboard
inputVar1 = 0;
switch (globalVars[VAR_MOUSE_X_MODE]) {
case 1:
mouseX = objectTable[1].x + 12;
break;
case 2:
mouseX = objectTable[1].x + 7;
break;
default:
mouseX = globalVars[VAR_MOUSE_X_POS];
break;
}
switch (globalVars[VAR_MOUSE_Y_MODE]) {
case 1:
mouseY = objectTable[1].y + 34;
break;
case 2:
mouseY = objectTable[1].y + 28;
break;
default:
mouseY = globalVars[VAR_MOUSE_Y_POS];
break;
}
if (var_5E == bgVar0) {
var_5E = 0;
globalVars[VAR_MOUSE_X_POS] = mouseX;
globalVars[VAR_MOUSE_Y_POS] = mouseY;
} else {
if (inputVar2) {
if (inputVar2 == 2) {
globalVars[VAR_MOUSE_X_POS] = 1;
} else {
globalVars[VAR_MOUSE_X_POS] = 320;
}
} else {
globalVars[VAR_MOUSE_X_POS] = mouseX;
}
if (inputVar3) {
if (inputVar3 == 2) {
globalVars[VAR_MOUSE_Y_POS] = 1;
} else {
globalVars[VAR_MOUSE_Y_POS] = 200;
}
} else {
globalVars[VAR_MOUSE_Y_POS] = mouseY;
}
}
bgVar0 = var_5E;
} else { // don't use keyboard for move -> shortcuts to commands
getMouseData(mouseUpdateStatus, &mouseButton, &mouseX, &mouseY);
switch (var_2 - 59) {
case 0:
case 1:
case 2:
case 3:
case 4:
case 5:
if (allowPlayerInput) {
playerCommand = var_2 - 59;
makeCommandLine();
}
break;
case 6:
case 7:
case 8:
case 23:
break;
case 9:
case 24:
g_cine->makeSystemMenu();
break;
default:
// printf("Unhandled case %d in last part of executePlayerInput\n",var2-59);
break;
}
}
return var_5E;
}
void drawSprite(overlayHeadElement *currentOverlay, byte *spritePtr,
byte *maskPtr, uint16 width, uint16 height, byte *page, int16 x, int16 y) {
byte *ptr = NULL;
byte *msk = NULL;
byte i = 0;
uint16 si = 0;
overlayHeadElement *pCurrentOverlay = currentOverlay;
int16 maskX, maskY, maskWidth, maskHeight;
uint16 maskSpriteIdx;
if (g_cine->getGameType() == Cine::GType_OS) {
drawSpriteRaw2(spritePtr, objectTable[currentOverlay->objIdx].part, width, height, page, x, y);
return;
}
while (pCurrentOverlay) {
if (pCurrentOverlay->type == 5) {
if (!si) {
ptr = (byte *)malloc(width * 8 * height);
msk = (byte *)malloc(width * 8 * height);
si = 1;
}
maskX = objectTable[pCurrentOverlay->objIdx].x;
maskY = objectTable[pCurrentOverlay->objIdx].y;
maskSpriteIdx = objectTable[pCurrentOverlay->objIdx].frame;
maskWidth = animDataTable[maskSpriteIdx].width / 2;
maskHeight = animDataTable[maskSpriteIdx].height;
gfxUpdateSpriteMask(spritePtr, maskPtr, width, height, animDataTable[maskSpriteIdx].ptr1, maskWidth, maskHeight, ptr, msk, x, y, maskX, maskY, i++);
#ifdef DEBUG_SPRITE_MASK
gfxFillSprite(animDataTable[maskSpriteIdx].ptr1, maskWidth, maskHeight, page, maskX, maskY, 1);
#endif
}
pCurrentOverlay = pCurrentOverlay->next;
}
if (si) {
gfxDrawMaskedSprite(ptr, msk, width, height, page, x, y);
free(ptr);
free(msk);
} else {
gfxDrawMaskedSprite(spritePtr, maskPtr, width, height, page, x, y);
}
}
int16 additionalBgVScroll = 0;
void backupOverlayPage(void) {
byte *scrollBg;
byte *bgPage = additionalBgTable[currentAdditionalBgIdx];
if (bgPage) {
if (!additionalBgVScroll) {
memcpy(page1Raw, bgPage, 320 * 200);
} else {
scrollBg = additionalBgTable[currentAdditionalBgIdx2];
for (int16 i = additionalBgVScroll; i < 200 + additionalBgVScroll; i++) {
if (i > 200) {
memcpy(page1Raw + (i - additionalBgVScroll) * 320, scrollBg + (i - 200) * 320, 320);
} else {
memcpy(page1Raw + (i - additionalBgVScroll) * 320, bgPage + (i) * 320, 320);
}
}
}
}
}
void drawMessage(const char *messagePtr, int16 x, int16 y, int16 width, int16 color) {
byte color2 = 2;
byte endOfMessageReached = 0;
int16 localX, localY, localWidth;
uint16 messageLength = 0, numWords = 0, messageWidth = 0;
uint16 lineResult, fullLineWidth;
uint16 interWordSize, interWordSizeRemain;
const char *endOfMessagePtr;
byte currentChar, characterWidth;
gfxDrawPlainBoxRaw(x, y, x + width, y + 4, color, page1Raw);
localX = x + 4;
localY = y + 4;
localWidth = width - 8;
do {
messageLength = 0;
while (messagePtr[messageLength] == ' ') {
messageLength++;
}
messagePtr += messageLength;
messageLength = computeMessageLength((const byte *)messagePtr, localWidth, &numWords, &messageWidth, &lineResult);
endOfMessagePtr = messagePtr + messageLength;
if (lineResult) {
fullLineWidth = localWidth - messageWidth;
if (numWords) {
interWordSize = fullLineWidth / numWords;
interWordSizeRemain = fullLineWidth % numWords;
} else {
interWordSize = 5;
interWordSizeRemain = 0;
}
} else {
interWordSize = 5;
interWordSizeRemain = 0;
}
gfxDrawPlainBoxRaw(x, localY, x + width, localY + 9, color, page1Raw);
do {
currentChar = *(messagePtr++);
if (currentChar == 0) {
endOfMessageReached = 1;
} else if (currentChar == ' ') {
localX += interWordSizeRemain + interWordSize;
if (interWordSizeRemain)
interWordSizeRemain = 0;
} else {
characterWidth = fontParamTable[currentChar].characterWidth;
if (characterWidth) {
byte characterIdx = fontParamTable[currentChar].characterIdx;
drawSpriteRaw(textTable[characterIdx][0], textTable[characterIdx][1], 2, 8, page1Raw, localX, localY);
localX += characterWidth + 1;
}
}
} while ((messagePtr < endOfMessagePtr) && !endOfMessageReached);
localX = x + 4;
localY += 9;
} while (!endOfMessageReached);
gfxDrawPlainBoxRaw(x, localY, x + width, localY + 4, color, page1Raw);
drawDoubleMessageBox(x, y, width, localY, color2, page1Raw);
}
void drawDialogueMessage(byte msgIdx, int16 x, int16 y, int16 width, int16 color) {
const char *messagePtr = (const char *)messageTable[msgIdx].ptr;
if (!messagePtr) {
freeOverlay(msgIdx, 2);
return;
}
_messageLen += strlen(messagePtr);
drawMessage(messagePtr, x, y, width, color);
freeOverlay(msgIdx, 2);
}
void drawFailureMessage(byte cmd) {
byte msgIdx = cmd * 4 + g_cine->_rnd.getRandomNumber(3);
const char *messagePtr = failureMessages[msgIdx];
int len = strlen(messagePtr);
_messageLen += len;
int16 width = 6 * len + 20;
if (width > 300)
width = 300;
int16 x = (320 - width) / 2;
int16 y = 80;
int16 color = 4;
drawMessage(messagePtr, x, y, width, color);
freeOverlay(cmd, 3);
}
void drawOverlays(void) {
uint16 partVar1, partVar2;
AnimData *pPart;
overlayHeadElement *currentOverlay, *nextOverlay;
int16 x, y;
objectStruct *objPtr;
byte messageIdx;
int16 part;
backupOverlayPage();
_messageLen = 0;
currentOverlay = (&overlayHead)->next;
while (currentOverlay) {
nextOverlay = currentOverlay->next;
switch (currentOverlay->type) {
case 0: // sprite
{
assert(currentOverlay->objIdx <= NUM_MAX_OBJECT);
objPtr = &objectTable[currentOverlay->objIdx];
x = objPtr->x;
y = objPtr->y;
if (objPtr->frame >= 0) {
if (g_cine->getGameType() == Cine::GType_OS) {
pPart = &animDataTable[objPtr->frame];
partVar1 = pPart->var1;
partVar2 = pPart->height;
if (pPart->ptr1) {
// NOTE: is the mask supposed to be in ptr1? Shouldn't that be ptr2, like below?
drawSprite(currentOverlay, pPart->ptr1, pPart->ptr1, partVar1, partVar2, page1Raw, x, y);
}
} else {
part = objPtr->part;
assert(part >= 0 && part <= NUM_MAX_ANIMDATA);
pPart = &animDataTable[objPtr->frame];
partVar1 = pPart->var1;
partVar2 = pPart->height;
if (pPart->ptr1) {
drawSprite(currentOverlay, pPart->ptr1, pPart->ptr2, partVar1, partVar2, page1Raw, x, y);
}
}
}
break;
}
case 2: // text
{
// gfxWaitVSync();
// hideMouse();
messageIdx = currentOverlay->objIdx;
x = currentOverlay->x;
y = currentOverlay->y;
partVar1 = currentOverlay->width;
partVar2 = currentOverlay->color;
blitRawScreen(page1Raw);
drawDialogueMessage(messageIdx, x, y, partVar1, partVar2);
// blitScreen(page0, NULL);
// gfxRedrawMouseCursor();
waitForPlayerClick = 1;
break;
}
case 3:
{
// gfxWaitSync()
// hideMouse();
blitRawScreen(page1Raw);
drawFailureMessage(currentOverlay->objIdx);
// blitScreen(page0, NULL);
// gfxRedrawMouseCursor();
waitForPlayerClick = 1;
break;
}
case 4:
{
assert(currentOverlay->objIdx <= NUM_MAX_OBJECT);
objPtr = &objectTable[currentOverlay->objIdx];
x = objPtr->x;
y = objPtr->y;
if (objPtr->frame >= 0) {
part = objPtr->part;
assert(part >= 0 && part <= NUM_MAX_ANIMDATA);
pPart = &animDataTable[objPtr->frame];
partVar1 = pPart->width / 2;
partVar2 = pPart->height;
if (pPart->ptr1) {
gfxFillSprite(pPart->ptr1, partVar1, partVar2, page1Raw, x, y);
}
}
break;
}
case 20:
{
assert(currentOverlay->objIdx <= NUM_MAX_OBJECT);
objPtr = &objectTable[currentOverlay->objIdx];
x = objPtr->x;
y = objPtr->y;
var5 = currentOverlay->x;
if (objPtr->frame >= 0) {
if (var5 <= 8) {
if (additionalBgTable[var5]) {
if (animDataTable[objPtr->frame].bpp == 1) {
int16 x2;
int16 y2;
x2 = animDataTable[objPtr->frame].width / 2;
y2 = animDataTable[objPtr->frame].height;
if (animDataTable[objPtr->frame].ptr1) {
// drawSpriteRaw(animDataTable[objPtr->frame].ptr1, animDataTable[objPtr->frame].ptr1, x2, y2,
// additionalBgTable[currentAdditionalBgIdx], x, y);
}
}
}
}
}
break;
}
}
currentOverlay = nextOverlay;
}
}
uint16 processKeyboard(uint16 param) {
return 0;
}
void mainLoopSub6(void) {
}
void checkForPendingDataLoad(void) {
if (newPrcName[0] != 0) {
freePrcLinkedList();
resetglobalScriptsHead();
loadPrc(newPrcName);
strcpy(currentPrcName, newPrcName);
strcpy(newPrcName, "");
addScriptToList0(1);
}
if (newRelName[0] != 0) {
releaseObjectScripts();
resetObjectScriptHead();
loadRel(newRelName);
strcpy(currentRelName, newRelName);
strcpy(newRelName, "");
}
if (newObjectName[0] != 0) {
unloadAllMasks();
resetMessageHead();
loadObject(newObjectName);
strcpy(currentObjectName, newObjectName);
strcpy(newObjectName, "");
}
if (newMsgName[0] != 0) {
loadMsg(newMsgName);
strcpy(currentMsgName, newMsgName);
strcpy(newMsgName, "");
}
}
void hideMouse(void) {
}
void removeExtention(char *dest, const char *source) {
strcpy(dest, source);
byte *ptr = (byte *) strchr(dest, '.');
if (ptr) {
*ptr = 0;
}
}
void addMessage(byte param1, int16 param2, int16 param3, int16 param4, int16 param5) {
overlayHeadElement *currentHead = &overlayHead;
overlayHeadElement *tempHead = currentHead;
overlayHeadElement *newElement;
currentHead = tempHead->next;
while (currentHead) {
tempHead = currentHead;
currentHead = tempHead->next;
}
newElement = new overlayHeadElement;
newElement->next = tempHead->next;
tempHead->next = newElement;
newElement->objIdx = param1;
newElement->type = 2;
newElement->x = param2;
newElement->y = param3;
newElement->width = param4;
newElement->color = param5;
if (!currentHead)
currentHead = &overlayHead;
newElement->previous = currentHead->previous;
currentHead->previous = newElement;
}
SeqListElement seqList;
void addSeqListElement(int16 param0, int16 param1, int16 param2, int16 param3, int16 param4, int16 param5, int16 param6, int16 param7, int16 param8) {
SeqListElement *currentHead = &seqList;
SeqListElement *tempHead = currentHead;
SeqListElement *newElement;
currentHead = tempHead->next;
while (currentHead && currentHead->varE < param7) {
tempHead = currentHead;
currentHead = tempHead->next;
}
newElement = new SeqListElement;
newElement->next = tempHead->next;
tempHead->next = newElement;
newElement->var6 = param0;
newElement->var4 = param1;
newElement->var8 = param2;
newElement->varA = param3;
newElement->varC = param4;
newElement->var14 = 0;
newElement->var16 = 0;
newElement->var18 = param5;
newElement->var1A = param6;
newElement->varE = param7;
newElement->var10 = param8;
newElement->var12 = param8;
newElement->var1C = 0;
newElement->var1E = 0;
}
void resetSeqList() {
seqList.next = NULL;
}
void computeMove1(SeqListElement *element, int16 x, int16 y, int16 param1,
int16 param2, int16 x2, int16 y2) {
element->var16 = 0;
element->var14 = 0;
if (y2) {
if (y - param2 > y2) {
element->var16 = 2;
}
if (y + param2 < y2) {
element->var16 = 1;
}
}
if (x2) {
if (x - param1 > x2) {
element->var14 = 2;
}
if (x + param1 < x2) {
element->var14 = 1;
}
}
}
uint16 computeMove2(SeqListElement *element) {
int16 returnVar = 0;
if (element->var16 == 1) {
returnVar = 4;
} else if (element->var16 == 2) {
returnVar = 3;
}
if (element->var14 == 1) {
returnVar = 1;
} else if (element->var14 == 2) {
returnVar = 2;
}
return returnVar;
}
// sort all the gfx stuff...
void resetGfxEntityEntry(uint16 objIdx) {
#if 0
overlayHeadElement* tempHead = &overlayHead;
byte* var_16 = NULL;
uint16 var_10 = 0;
uint16 var_12 = 0;
overlayHeadElement* currentHead = tempHead->next;
byte* var_1A = NULL;
overlayHeadElement* var1E = &overlayHead;
while (currentHead) {
tempHead2 = currentHead->next;
if (currentHead->objIdx == objIdx && currentHead->type!=2 && currentHead->type!=3 && currentHead->type!=0x14) {
tempHead->next = tempHead2;
if (tempHead2) {
tempHead2->previous = currentHead->previous;
} else {
seqVar0 = currentHead->previous;
}
var_22 = var_16;
if (!var_22) {
// todo: goto?
}
var_22->previous = currentHead;
} else {
}
if (currentHead->type == 0x14) {
} else {
}
if (currentHead->type == 0x2 || currentHead->type == 0x3) {
si = 10000;
} else {
si = objectTable[currentHead->objIdx];
}
if (objectTable[objIdx]>si) {
var1E = currentHead;
}
tempHead = tempHead->next;
}
if (var_1A) {
currentHead = var_16;
var_22 = var_1E->next;
var_1E->next = currentHead;
var_1A->next = var_22;
if (var_1E != &gfxEntityHead) {
currentHead->previous = var_1E;
}
if (!var_22) {
seqVar0 = var_1A;
} else {
var_22->previous = var_1A;
}
}
#endif
}
uint16 addAni(uint16 param1, uint16 param2, byte *ptr, SeqListElement *element, uint16 param3, int16 *param4) {
byte *currentPtr = ptr;
byte *ptrData;
byte *ptr2;
int16 di;
assert(ptr);
assert(element);
assert(param4);
dummyU16 = READ_BE_UINT16((currentPtr + param1 * 2) + 8);
ptrData = ptr + dummyU16;
assert(*ptrData);
di = (objectTable[param2].costume + 1) % (*ptrData);
ptr2 = (ptrData + (di * 8)) + 1;
if ((checkCollision(param2, ptr2[0], ptr2[1], ptr2[2], ptr[0]) & 1)) {
return 0;
}
objectTable[param2].x += (int8)ptr2[4];
objectTable[param2].y += (int8)ptr2[5];
objectTable[param2].mask += (int8)ptr2[6];
if (objectTable[param2].frame) {
resetGfxEntityEntry(param2);
}
objectTable[param2].frame = ptr2[7] + element->var8;
if (param3 || !element->var14) {
objectTable[param2].costume = di;
} else {
*param4 = di;
}
return 1;
}
void processSeqListElement(SeqListElement *element) {
int16 x = objectTable[element->var6].x;
int16 y = objectTable[element->var6].y;
byte *ptr1 = animDataTable[element->varA].ptr1;
int16 var_10;
int16 var_4;
int16 var_2;
if (element->var12 < element->var10) {
element->var12++;
return;
}
element->var12 = 0;
if (ptr1) {
uint16 param1 = ptr1[1];
uint16 param2 = ptr1[2];
if (element->varC != 255) {
// FIXME: Why is this here? Fingolfin gets lots of these
// in his copy of Operation Stealth (value 0 or 236) under
// Mac OS X. Maybe it's a endian issue? At least the graphics
// in the copy protection screen are partially messed up.
warning("processSeqListElement: varC = %d", element->varC);
}
if (globalVars[VAR_MOUSE_X_POS] || globalVars[VAR_MOUSE_Y_POS]) {
computeMove1(element, ptr1[4] + x, ptr1[5] + y, param1, param2, globalVars[VAR_MOUSE_X_POS], globalVars[VAR_MOUSE_Y_POS]);
} else {
element->var16 = 0;
element->var14 = 0;
}
var_10 = computeMove2(element);
if (var_10) {
element->var1C = var_10;
element->var1E = var_10;
}
var_4 = -1;
if ((element->var16 == 1
&& !addAni(3, element->var6, ptr1, element, 0, &var_4)) || (element->var16 == 2 && !addAni(2, element->var6, ptr1, element, 0,
&var_4))) {
if (element->varC == 255) {
globalVars[VAR_MOUSE_Y_POS] = 0;
}
}
if ((element->var14 == 1
&& !addAni(0, element->var6, ptr1, element, 1, &var_2))) {
if (element->varC == 255) {
globalVars[VAR_MOUSE_X_POS] = 0;
if (var_4 != -1) {
objectTable[element->var6].costume = var_4;
}
}
}
if ((element->var14 == 2 && !addAni(1, element->var6, ptr1, element, 1, &var_2))) {
if (element->varC == 255) {
globalVars[VAR_MOUSE_X_POS] = 0;
if (var_4 != -1) {
objectTable[element->var6].costume = var_4;
}
}
}
if (element->var16 + element->var14) {
if (element->var1C) {
if (element->var1E) {
objectTable[element->var6].costume = 0;
element->var1E = 0;
}
addAni(element->var1C + 3, element->var6, ptr1, element, 1, (int16 *) & var2);
}
}
}
}
void processSeqList(void) {
SeqListElement *currentHead = &seqList;
SeqListElement *tempHead = currentHead;
currentHead = tempHead->next;
while (currentHead) {
if (currentHead->var4 != -1) {
processSeqListElement(currentHead);
}
tempHead = currentHead;
currentHead = tempHead->next;
}
}
bool makeTextEntryMenu(const char *messagePtr, char *inputString, int stringMaxLength, int y) {
int16 color = 2;
byte color2 = defaultMenuBoxColor2;
byte endOfMessageReached = 0;
int16 localX, localY, localWidth;
int margins = 16;
int len = strlen(messagePtr);
int16 width = 6 * len + 20;
uint16 messageLength = 0, numWords = 0, messageWidth = 0;
uint16 lineResult, fullLineWidth;
uint16 interWordSize, interWordSizeRemain;
const char *endOfMessagePtr;
byte currentChar, characterWidth;
width = CLIP((int)width, 180, 250);
int16 x = (320 - width) / 2;
gfxDrawPlainBoxRaw(x - margins, y, x + width + margins, y + 4, color2, page1Raw);
localX = x + 4;
localY = y + 4;
localWidth = width;
getKeyData(); // clear input key
do {
messageLength = 0;
while (messagePtr[messageLength] == ' ') {
messageLength++;
}
messagePtr += messageLength;
messageLength = computeMessageLength((const byte *)messagePtr, localWidth, &numWords, &messageWidth, &lineResult);
endOfMessagePtr = messagePtr + messageLength;
if (lineResult) {
fullLineWidth = localWidth - messageWidth;
if (numWords) {
interWordSize = fullLineWidth / numWords;
interWordSizeRemain = fullLineWidth % numWords;
} else {
interWordSize = 5;
interWordSizeRemain = 0;
}
} else {
interWordSize = 5;
interWordSizeRemain = 0;
}
gfxDrawPlainBoxRaw(x - margins, localY, x + width + margins, localY + 9, color2, page1Raw);
do {
currentChar = *(messagePtr++);
if (currentChar == 0) {
endOfMessageReached = 1;
} else if (currentChar == ' ') {
localX += interWordSizeRemain + interWordSize;
if (interWordSizeRemain)
interWordSizeRemain = 0;
} else {
characterWidth = fontParamTable[currentChar].characterWidth;
if (characterWidth) {
byte characterIdx = fontParamTable[currentChar].characterIdx;
drawSpriteRaw(textTable[characterIdx][0], textTable[characterIdx][1], 2, 8, page1Raw, localX, localY);
localX += characterWidth + 1;
}
}
} while ((messagePtr < endOfMessagePtr) && !endOfMessageReached);
localX = x + 4;
localY += 9;
} while (!endOfMessageReached);
// Input string
gfxDrawPlainBoxRaw(x - margins, localY, x + width + margins, localY + 9, color2, page1Raw);
localY += 9;
x -= margins;
width += margins * 2;
gfxDrawPlainBoxRaw(x, localY, x + width, localY + 4, color2, page1Raw);
drawDoubleMessageBox(x, y, width, localY, color, page1Raw);
x += margins;
width -= margins * 2;
localY -= 9;
int quit = 0;
bool redraw = true;
CommandeType tempString;
int inputLength = strlen(inputString);
int inputPos = inputLength + 1;
while (!quit) {
if (redraw) {
gfxDrawPlainBoxRaw(x, localY - 1, x + width, localY + 8, 0, page1Raw);
int currentX = x + 4;
for (uint j = 0; j < strlen(inputString); j++) {
currentChar = inputString[j];
currentX = drawChar(currentChar, currentX, localY);
// draw cursor here
if (inputPos == (int)(j + 2))
gfxDrawLine(currentX, localY - 1, currentX, localY + 8, color, page1Raw);
}
if (strlen(inputString) == 0 || inputPos == 1) // cursor wasn't yet drawn
gfxDrawLine(x + 4, localY - 1, x + 4, localY + 8, color, page1Raw);
blitRawScreen(page1Raw);
redraw = false;
}
char ch[2];
memset(tempString, 0, stringMaxLength);
ch[1] = 0;
manageEvents();
int keycode = getKeyData();
uint16 mouseButton, mouseX, mouseY;
getMouseData(0, &mouseButton, &mouseX, &mouseY);
if (mouseButton & 2)
quit = 2;
else if (mouseButton & 1)
quit = 1;
switch (keycode) {
case Common::KEYCODE_BACKSPACE:
if (inputPos <= 1) {
break;
}
inputPos--;
redraw = true;
case Common::KEYCODE_DELETE:
if (inputPos <= inputLength) {
if (inputPos != 1) {
strncpy(tempString, inputString, inputPos - 1);
}
if (inputPos != inputLength) {
strncat(tempString, &inputString[inputPos], inputLength - inputPos);
}
strcpy(inputString, tempString);
inputLength = strlen(inputString);
redraw = true;
}
break;
case Common::KEYCODE_LEFT:
if (inputPos > 1) {
inputPos--;
redraw = true;
}
break;
case Common::KEYCODE_RIGHT:
if (inputPos <= inputLength) {
inputPos++;
redraw = true;
}
break;
default:
if (((keycode >= 'a') && (keycode <='z')) ||
((keycode >= '0') && (keycode <='9')) ||
((keycode >= 'A') && (keycode <='Z')) ||
(keycode == ' ')) {
if (inputLength < stringMaxLength - 1) {
ch[0] = keycode;
if (inputPos != 1) {
strncpy(tempString, inputString, inputPos - 1);
strcat(tempString, ch);
}
if ((inputLength == 0) || (inputPos == 1)) {
strcpy(tempString, ch);
}
if ((inputLength != 0) && (inputPos != inputLength)) {
strncat(tempString, &inputString[inputPos - 1], inputLength - inputPos + 1);
}
strcpy(inputString, tempString);
inputLength = strlen(inputString);
inputPos++;
redraw = true;
}
}
break;
}
}
if (quit == 2)
return false;
return true;
}
} // End of namespace Cine