scummvm/scumm/verbs.cpp

445 lines
11 KiB
C++
Raw Normal View History

2001-10-09 14:30:12 +00:00
/* ScummVM - Scumm Interpreter
* Copyright (C) 2001 Ludvig Strigeus
* Copyright (C) 2001-2003 The ScummVM project
2001-10-09 14:30:12 +00:00
*
* 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., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA.
*
* $Header$
2001-10-09 14:30:12 +00:00
*
*/
#include "stdafx.h"
#include "charset.h"
#include "object.h"
#include "resource.h"
#include "scumm.h"
2002-11-29 15:13:49 +00:00
#include "verbs.h"
2001-10-09 14:30:12 +00:00
void Scumm::checkV2Inventory(int x, int y) {
int object = 0;
if ((y < virtscr[2].topline + 34) || !(_mouseButStat & MBS_LEFT_CLICK))
return;
if (x > 145 && x < 160) { // Inventory Arrows
if (y < virtscr[2].topline + 38) // Up arrow
_inventoryOffset-=2;
else if (y > virtscr[2].topline + 47) // Down arrow
_inventoryOffset+=2;
if (_inventoryOffset < 0)
_inventoryOffset = 0;
if (_inventoryOffset > (getInventoryCount(_scummVars[VAR_EGO])-2))
_inventoryOffset = (getInventoryCount(_scummVars[VAR_EGO])-2);
redrawV2Inventory();
}
object = ((y - virtscr[2].topline - 34) / 8) * 2;
if (x > 150)
object++;
object = findInventory(_scummVars[VAR_EGO], object+1+_inventoryOffset);
if (object > 0) {
2003-05-21 23:54:39 +00:00
runInputScript(3, object, 0);
}
}
2003-05-21 09:29:24 +00:00
void Scumm::redrawV2Inventory() {
int i, items = 0;
2003-05-21 09:29:24 +00:00
bool alternate = false;
int max_inv = getInventoryCount(_scummVars[VAR_EGO]);
ScummVM::Rect inventoryBox;
2003-05-21 09:29:24 +00:00
// Clear on all invocations, so hiding works properly
inventoryBox.top = virtscr[2].topline + 32;
inventoryBox.bottom = virtscr[2].topline + virtscr[2].height;
inventoryBox.left = 0;
inventoryBox.right = 319;
restoreBG(inventoryBox);
2003-05-21 09:29:24 +00:00
if (!(_userState & 64)) // Don't draw inventory unless active
return;
2003-05-21 09:29:24 +00:00
_string[1].charset = 1;
_string[1].color = 5;
2003-05-21 09:29:24 +00:00
items = 0;
for (i = _inventoryOffset + 1; i <= max_inv; i++) {
int obj = findInventory(_scummVars[VAR_EGO], i);
if ((obj == 0) || (items == 4)) break;
_string[1].ypos = virtscr[2].topline + 34 + (8*(items / 2));
if (alternate)
_string[1].xpos = 200;
else
_string[1].xpos = 0;
2003-05-21 09:29:24 +00:00
_messagePtr = getObjOrActorName(obj);
assert(_messagePtr);
drawString(1);
items++;
2003-05-21 09:29:24 +00:00
alternate = !alternate;
2003-05-21 09:29:24 +00:00
}
if (_inventoryOffset > 0) { // Draw Up Arrow
2003-05-21 09:29:24 +00:00
_string[1].xpos = 145;
_string[1].ypos = virtscr[2].topline + 32;
2003-05-21 23:54:39 +00:00
_messagePtr = (const byte *)"\1\2";
2003-05-21 09:29:24 +00:00
drawString(1);
}
if (items == 4) { // Draw Down Arrow
_string[1].xpos = 145;
_string[1].ypos = virtscr[2].topline + 47;
2003-05-21 23:54:39 +00:00
_messagePtr = (const byte *)"\3\4";
2003-05-21 09:29:24 +00:00
drawString(1);
}
}
2003-03-06 17:58:13 +00:00
void Scumm::redrawVerbs() {
2001-10-09 14:30:12 +00:00
int i;
int verb = (_cursor.state > 0 ? checkMouseOver(_mouse.x, _mouse.y) : 0);
for (i = _maxVerbs-1; i >= 0; i--) {
if (i == verb && _verbs[verb].hicolor)
drawVerb(i, 1);
else
drawVerb(i, 0);
}
_verbMouseOver = verb;
2001-10-09 14:30:12 +00:00
}
2003-03-06 17:58:13 +00:00
void Scumm::checkExecVerbs() {
int i, over;
2001-10-09 14:30:12 +00:00
VerbSlot *vs;
if (_userPut <= 0 || _mouseButStat == 0)
2001-10-09 14:30:12 +00:00
return;
if (_mouseButStat < MBS_MAX_KEY) {
2001-10-09 14:30:12 +00:00
/* Check keypresses */
2001-10-16 10:01:48 +00:00
vs = &_verbs[1];
for (i = 1; i < _maxVerbs; i++, vs++) {
if (vs->verbid && vs->saveid == 0 && vs->curmode == 1) {
2001-10-09 14:30:12 +00:00
if (_mouseButStat == vs->key) {
runInputScript(1, vs->verbid, 1);
return;
}
}
}
runInputScript(4, _mouseButStat, 1);
} else if (_mouseButStat & MBS_MOUSE_MASK) {
2003-05-21 23:54:39 +00:00
VirtScreen *zone = findVirtScreen(_mouse.y);
byte code = _mouseButStat & MBS_LEFT_CLICK ? 1 : 2;
2003-05-21 23:54:39 +00:00
if (zone->number == 0) {
over = checkMouseOver(_mouse.x, _mouse.y);
2001-10-09 14:30:12 +00:00
if (over != 0) {
runInputScript(1, _verbs[over].verbid, code);
2001-10-09 14:30:12 +00:00
return;
}
runInputScript(2, 0, code);
2003-05-21 23:54:39 +00:00
} else if (_features & GF_AFTER_V2 && zone->number == 2 && _mouse.y > zone->topline + 32) {
checkV2Inventory(_mouse.x, _mouse.y);
2001-10-09 14:30:12 +00:00
} else {
over = checkMouseOver(_mouse.x, _mouse.y);
2002-07-07 20:25:23 +00:00
// FIXME For the future: Indy3 and under inv scrolling
/*
2002-07-07 20:25:23 +00:00
if (over >= 31 && over <= 36)
over += _inventoryOffset;
*/
runInputScript(1, over != 0 ? _verbs[over].verbid : 0, code);
2001-10-09 14:30:12 +00:00
}
}
}
2003-03-06 17:58:13 +00:00
void Scumm::verbMouseOver(int verb) {
if (_verbMouseOver == verb)
2001-10-09 14:30:12 +00:00
return;
2002-11-29 18:27:35 +00:00
if (_verbs[_verbMouseOver].type != kImageVerbType) {
2001-10-09 14:30:12 +00:00
drawVerb(_verbMouseOver, 0);
_verbMouseOver = verb;
}
2002-11-29 18:27:35 +00:00
if (_verbs[verb].type != kImageVerbType && _verbs[verb].hicolor) {
drawVerb(verb, 1);
2001-10-09 14:30:12 +00:00
_verbMouseOver = verb;
}
}
2003-03-06 17:58:13 +00:00
int Scumm::checkMouseOver(int x, int y) {
2001-10-09 14:30:12 +00:00
VerbSlot *vs;
int i = _maxVerbs - 1;
2001-10-09 14:30:12 +00:00
2001-10-16 10:01:48 +00:00
vs = &_verbs[i];
2001-10-09 14:30:12 +00:00
do {
2002-07-07 20:25:23 +00:00
if (vs->curmode != 1 || !vs->verbid || vs->saveid || y < vs->y || y >= vs->bottom)
continue;
2001-10-09 14:30:12 +00:00
if (vs->center) {
if (x < -(vs->right - vs->x - vs->x) || x >= vs->right)
continue;
} else {
2001-10-09 14:30:12 +00:00
if (x < vs->x || x >= vs->right)
continue;
}
2001-10-09 14:30:12 +00:00
return i;
} while (--vs, --i);
2001-10-09 14:30:12 +00:00
return 0;
}
2003-03-06 17:58:13 +00:00
void Scumm::drawVerb(int verb, int mode) {
2001-10-09 14:30:12 +00:00
VerbSlot *vs;
bool tmp;
2001-10-09 14:30:12 +00:00
2002-12-13 00:52:14 +00:00
if (!verb)
2001-10-09 14:30:12 +00:00
return;
2002-12-13 00:52:14 +00:00
vs = &_verbs[verb];
2001-10-09 14:30:12 +00:00
if (!vs->saveid && vs->curmode && vs->verbid) {
2002-11-29 18:27:35 +00:00
if (vs->type == kImageVerbType) {
2002-12-13 00:52:14 +00:00
drawVerbBitmap(verb, vs->x, vs->y);
2001-10-09 14:30:12 +00:00
return;
}
restoreVerbBG(verb);
2001-10-09 14:30:12 +00:00
_string[4].charset = vs->charset_nr;
_string[4].xpos = vs->x;
_string[4].ypos = vs->y;
_string[4].right = _screenWidth - 1;
_string[4].center = vs->center;
if (vs->curmode == 2)
_string[4].color = vs->dimcolor;
else if (mode && vs->hicolor)
_string[4].color = vs->hicolor;
else
_string[4].color = vs->color;
// FIXME For the future: Indy3 and under inv scrolling
/*
2002-12-13 00:52:14 +00:00
if (verb >= 31 && verb <= 36)
verb += _inventoryOffset;
2002-07-07 20:25:23 +00:00
*/
2002-12-13 00:52:14 +00:00
_messagePtr = getResourceAddress(rtVerb, verb);
if (!_messagePtr)
return;
2001-10-09 14:30:12 +00:00
assert(_messagePtr);
2003-01-12 21:12:28 +00:00
if ((_features & GF_AFTER_V8) && (_messagePtr[0] == '/')) {
char pointer[20];
int i, j;
2003-01-12 21:12:28 +00:00
translateText(_messagePtr, _transText);
2003-01-12 21:12:28 +00:00
for (i = 0, j = 0; (_messagePtr[i] != '/' || j == 0) && j < 19; i++) {
if (_messagePtr[i] != '/')
pointer[j++] = _messagePtr[i];
}
pointer[j] = 0;
_messagePtr = _transText;
}
tmp = _charset->_center;
_charset->_center = 0;
2001-10-09 14:30:12 +00:00
drawString(4);
_charset->_center = tmp;
vs->right = _charset->_str.right;
vs->bottom = _charset->_str.bottom;
2003-05-15 22:36:43 +00:00
vs->old = _charset->_str;
_charset->_str.left = _charset->_str.right;
2001-10-09 14:30:12 +00:00
} else {
2002-12-13 00:52:14 +00:00
restoreVerbBG(verb);
2001-10-09 14:30:12 +00:00
}
}
2003-03-06 17:58:13 +00:00
void Scumm::restoreVerbBG(int verb) {
2001-10-09 14:30:12 +00:00
VerbSlot *vs;
2001-10-16 10:01:48 +00:00
vs = &_verbs[verb];
2001-10-09 14:30:12 +00:00
2003-05-15 22:36:43 +00:00
if (vs->old.left != -1) {
restoreBG(vs->old, vs->bkcolor);
vs->old.left = -1;
2001-10-09 14:30:12 +00:00
}
}
2003-03-06 17:58:13 +00:00
void Scumm::drawVerbBitmap(int verb, int x, int y) {
2001-10-09 14:30:12 +00:00
VirtScreen *vs;
VerbSlot *vst;
byte twobufs, *imptr = 0;
2001-10-09 14:30:12 +00:00
int ydiff, xstrip;
int imgw, imgh;
int i, tmp;
2001-10-26 17:34:50 +00:00
byte *obim;
ImageHeader *imhd;
2002-02-15 17:17:35 +00:00
uint32 size;
2001-10-09 14:30:12 +00:00
if ((vs = findVirtScreen(y)) == NULL)
2001-10-09 14:30:12 +00:00
return;
2001-10-26 17:34:50 +00:00
gdi.disableZBuffer();
2001-10-09 14:30:12 +00:00
twobufs = vs->alloctwobuffers;
vs->alloctwobuffers = 0;
xstrip = x >> 3;
2001-10-09 14:30:12 +00:00
ydiff = y - vs->topline;
obim = getResourceAddress(rtVerb, verb);
assert(obim);
if (_features & GF_OLD_BUNDLE) {
imgw = obim[0];
imgh = obim[1] >> 3;
imptr = obim + 2;
} else if (_features & GF_SMALL_HEADER) {
size = READ_LE_UINT32(obim);
imgw = (*(obim + size + 11));
imgh = (*(obim + size + 17)) >> 3;
imptr = (obim + 8);
} else {
imhd = (ImageHeader *)findResourceData(MKID('IMHD'), obim);
if (_features & GF_AFTER_V7) {
imgw = READ_LE_UINT16(&imhd->v7.width) >> 3;
imgh = READ_LE_UINT16(&imhd->v7.height) >> 3;
} else {
imgw = READ_LE_UINT16(&imhd->old.width) >> 3;
imgh = READ_LE_UINT16(&imhd->old.height) >> 3;
}
if (_features & GF_AFTER_V8) {
warning("drawVerbBitmap(%d, %d, %d)", verb, x, y);
imptr = findResource(MKID('IMAG'), obim);
assert(imptr);
imptr = findResource(MKID('WRAP'), imptr);
assert(imptr);
imptr = findResource(MKID('OFFS'), imptr);
assert(imptr);
// Get the address of the second SMAP (corresponds to IM01)
imptr += READ_LE_UINT32(imptr + 12);
} else
imptr = findResource(MKID('IM01'), obim);
if (!imptr)
2002-12-13 00:52:14 +00:00
error("No image for verb %d", verb);
}
assert(imptr);
for (i = 0; i < imgw; i++) {
2001-10-26 17:34:50 +00:00
tmp = xstrip + i;
if (tmp < gdi._numStrips)
gdi.drawBitmap(imptr, vs, tmp, ydiff, imgw * 8, imgh * 8, i, 1, Gdi::dbAllowMaskOr);
2001-10-09 14:30:12 +00:00
}
2002-12-13 00:52:14 +00:00
vst = &_verbs[verb];
vst->right = vst->x + imgw * 8;
vst->bottom = vst->y + imgh * 8;
2003-05-15 22:36:43 +00:00
vst->old.left = vst->x;
vst->old.right = vst->right;
vst->old.top = vst->y;
vst->old.bottom = vst->bottom;
2001-10-26 17:34:50 +00:00
gdi.enableZBuffer();
2001-10-09 14:30:12 +00:00
vs->alloctwobuffers = twobufs;
}
2003-03-06 17:58:13 +00:00
int Scumm::getVerbSlot(int id, int mode) {
2001-10-09 14:30:12 +00:00
int i;
for (i = 1; i < _maxVerbs; i++) {
2001-10-16 10:01:48 +00:00
if (_verbs[i].verbid == id && _verbs[i].saveid == mode) {
2001-10-09 14:30:12 +00:00
return i;
}
}
return 0;
}
2003-03-06 17:58:13 +00:00
void Scumm::killVerb(int slot) {
2001-10-09 14:30:12 +00:00
VerbSlot *vs;
if (slot == 0)
2001-10-09 14:30:12 +00:00
return;
2001-10-16 10:01:48 +00:00
vs = &_verbs[slot];
2001-10-09 14:30:12 +00:00
vs->verbid = 0;
vs->curmode = 0;
nukeResource(rtVerb, slot);
2001-10-09 14:30:12 +00:00
if (vs->saveid == 0) {
2001-10-09 14:30:12 +00:00
drawVerb(slot, 0);
verbMouseOver(0);
}
vs->saveid = 0;
}
2003-03-06 17:58:13 +00:00
void Scumm::setVerbObject(uint room, uint object, uint verb) {
byte *obimptr;
byte *obcdptr;
2002-02-15 17:17:35 +00:00
uint32 size, size2;
FindObjectInRoom foir;
int i;
2001-10-09 14:30:12 +00:00
if (whereIsObject(object) == WIO_FLOBJECT)
2001-10-09 14:30:12 +00:00
error("Can't grab verb image from flobject");
if (_features & GF_OLD_BUNDLE) {
for (i = (_numLocalObjects-1); i > 0; i--) {
if (_objs[i].obj_nr == object) {
findObjectInRoom(&foir, foImageHeader, object, room);
size = READ_LE_UINT16(foir.obim);
byte *ptr = createResource(rtVerb, verb, size + 2);
obcdptr = getResourceAddress(rtRoom, room) + getOBCDOffs(object);
ptr[0] = *(obcdptr + 9); // Width
ptr[1] = *(obcdptr + 15); // Height
memcpy(ptr + 2, foir.obim, size);
return;
}
}
} else if (_features & GF_SMALL_HEADER) {
2003-01-12 08:37:59 +00:00
for (i = (_numLocalObjects-1); i > 0; i--) {
if (_objs[i].obj_nr == object) {
// FIXME - the only thing we need from the OBCD is the image size!
// So we could use almost the same code (save for offsets)
// as in the GF_OLD_BUNDLE code. But of course that would break save games
// unless we insert special conversion code... <sigh>
findObjectInRoom(&foir, foImageHeader, object, room);
size = READ_LE_UINT32(foir.obim);
2002-02-15 17:17:35 +00:00
obcdptr = getResourceAddress(rtRoom, room) + getOBCDOffs(object);
size2 = READ_LE_UINT32(obcdptr);
createResource(rtVerb, verb, size + size2);
obimptr = getResourceAddress(rtRoom, room) - foir.roomptr + foir.obim;
2002-02-15 17:17:35 +00:00
obcdptr = getResourceAddress(rtRoom, room) + getOBCDOffs(object);
memcpy(getResourceAddress(rtVerb, verb), obimptr, size);
memcpy(getResourceAddress(rtVerb, verb) + size, obcdptr, size2);
return;
}
}
} else {
findObjectInRoom(&foir, foImageHeader, object, room);
size = READ_BE_UINT32_UNALIGNED(foir.obim + 4);
createResource(rtVerb, verb, size);
obimptr = getResourceAddress(rtRoom, room) - foir.roomptr + foir.obim;
memcpy(getResourceAddress(rtVerb, verb), obimptr, size);
}
2001-10-09 14:30:12 +00:00
}