2009-09-03 20:59:17 +00:00
|
|
|
/* ScummVM - Graphic Adventure Engine
|
|
|
|
*
|
|
|
|
* ScummVM is the legal property of its developers, whose names
|
|
|
|
* are too numerous to list here. Please refer to the COPYRIGHT
|
|
|
|
* file distributed with this source distribution.
|
|
|
|
*
|
|
|
|
* This program is free software; you can redistribute it and/or
|
|
|
|
* modify it under the terms of the GNU General Public License
|
|
|
|
* as published by the Free Software Foundation; either version 2
|
|
|
|
* of the License, or (at your option) any later version.
|
|
|
|
*
|
|
|
|
* This program is distributed in the hope that it will be useful,
|
|
|
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
|
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
|
|
* GNU General Public License for more details.
|
|
|
|
*
|
|
|
|
* You should have received a copy of the GNU General Public License
|
|
|
|
* along with this program; if not, write to the Free Software
|
|
|
|
* Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
|
|
|
|
*
|
2009-09-03 21:20:13 +00:00
|
|
|
* $URL$
|
|
|
|
* $Id$
|
2009-09-03 20:59:17 +00:00
|
|
|
*/
|
|
|
|
|
|
|
|
#include "common/stream.h"
|
|
|
|
|
2009-09-04 20:08:33 +00:00
|
|
|
#include "teenagent/inventory.h"
|
|
|
|
#include "teenagent/resources.h"
|
|
|
|
#include "teenagent/objects.h"
|
|
|
|
#include "teenagent/teenagent.h"
|
|
|
|
|
|
|
|
namespace TeenAgent {
|
2009-09-03 20:59:17 +00:00
|
|
|
|
2009-09-15 07:41:05 +00:00
|
|
|
void Inventory::init(TeenAgentEngine *engine) {
|
2009-09-03 20:59:17 +00:00
|
|
|
_engine = engine;
|
|
|
|
_active = false;
|
2009-09-15 07:41:05 +00:00
|
|
|
Resources *res = Resources::instance();
|
2009-09-15 08:54:06 +00:00
|
|
|
|
2009-09-15 07:41:05 +00:00
|
|
|
Common::SeekableReadStream *s = res->varia.getStream(3);
|
2009-09-03 20:59:17 +00:00
|
|
|
assert(s != NULL);
|
|
|
|
debug(0, "loading inventory background...");
|
2009-09-15 20:21:18 +00:00
|
|
|
background.load(s, Surface::kTypeOns);
|
2009-09-15 08:54:06 +00:00
|
|
|
|
2009-09-03 20:59:17 +00:00
|
|
|
items = res->varia.getStream(4);
|
|
|
|
assert(items != NULL);
|
2009-09-15 08:54:06 +00:00
|
|
|
|
2009-09-03 20:59:17 +00:00
|
|
|
byte offsets = items->readByte();
|
|
|
|
assert(offsets == 92);
|
2009-10-31 19:03:44 +00:00
|
|
|
for (byte i = 0; i <= offsets; ++i) {
|
2009-09-03 20:59:17 +00:00
|
|
|
offset[i] = items->readUint16LE();
|
|
|
|
}
|
2009-09-26 15:04:09 +00:00
|
|
|
|
2009-10-31 19:03:44 +00:00
|
|
|
for (byte i = 0; i <= 92; ++i) {
|
2009-09-26 15:04:09 +00:00
|
|
|
InventoryObject io;
|
|
|
|
uint16 obj_addr = res->dseg.get_word(0xc4a4 + i * 2);
|
|
|
|
if (obj_addr != 0)
|
|
|
|
io.load(res->dseg.ptr(obj_addr));
|
|
|
|
objects.push_back(io);
|
|
|
|
}
|
|
|
|
|
2009-09-03 20:59:17 +00:00
|
|
|
inventory = res->dseg.ptr(0xc48d);
|
2009-09-15 08:54:06 +00:00
|
|
|
|
|
|
|
for (int y = 0; y < 4; ++y)
|
2009-09-04 20:09:29 +00:00
|
|
|
for (int x = 0; x < 6; ++x) {
|
2009-09-03 20:59:17 +00:00
|
|
|
int i = y * 6 + x;
|
|
|
|
graphics[i].rect.left = 28 + 45 * x - 1;
|
|
|
|
graphics[i].rect.top = 23 + 31 * y - 1;
|
|
|
|
graphics[i].rect.right = graphics[i].rect.left + 40;
|
|
|
|
graphics[i].rect.bottom = graphics[i].rect.top + 26;
|
|
|
|
}
|
2009-09-15 08:54:06 +00:00
|
|
|
|
2009-09-03 20:59:17 +00:00
|
|
|
hovered_obj = selected_obj = NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool Inventory::has(byte item) const {
|
2009-09-04 20:09:29 +00:00
|
|
|
for (int i = 0; i < 24; ++i) {
|
2009-09-15 08:54:06 +00:00
|
|
|
if (inventory[i] == item)
|
2009-09-03 20:59:17 +00:00
|
|
|
return true;
|
|
|
|
}
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
void Inventory::remove(byte item) {
|
|
|
|
debug(0, "removing %02x from inventory", item);
|
|
|
|
int i;
|
2009-09-04 20:09:29 +00:00
|
|
|
for (i = 0; i < 24; ++i) {
|
2009-09-03 20:59:17 +00:00
|
|
|
if (inventory[i] == item) {
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
2009-09-04 20:09:29 +00:00
|
|
|
for (; i < 23; ++i) {
|
2009-09-03 20:59:17 +00:00
|
|
|
inventory[i] = inventory[i + 1];
|
|
|
|
graphics[i].free();
|
|
|
|
}
|
|
|
|
inventory[23] = 0;
|
|
|
|
graphics[23].free();
|
|
|
|
}
|
|
|
|
|
|
|
|
void Inventory::clear() {
|
|
|
|
debug(0, "clearing inventory");
|
2009-09-04 20:09:29 +00:00
|
|
|
for (int i = 0; i < 24; ++i) {
|
2009-09-03 20:59:17 +00:00
|
|
|
inventory[i] = 0;
|
|
|
|
graphics[i].free();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void Inventory::add(byte item) {
|
|
|
|
if (has(item))
|
|
|
|
return;
|
|
|
|
debug(0, "adding %02x to inventory", item);
|
2009-09-04 20:09:29 +00:00
|
|
|
for (int i = 0; i < 24; ++i) {
|
2009-09-03 20:59:17 +00:00
|
|
|
if (inventory[i] == 0) {
|
|
|
|
inventory[i] = item;
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
error("no room for item %02x", item);
|
|
|
|
}
|
|
|
|
|
|
|
|
bool Inventory::processEvent(const Common::Event &event) {
|
2009-09-15 07:41:05 +00:00
|
|
|
Resources *res = Resources::instance();
|
2009-09-15 08:54:06 +00:00
|
|
|
|
|
|
|
switch (event.type) {
|
2009-09-03 20:59:17 +00:00
|
|
|
case Common::EVENT_MOUSEMOVE:
|
|
|
|
mouse = event.mouse;
|
|
|
|
if (!active() && event.mouse.y < 5) {
|
|
|
|
activate(true);
|
|
|
|
return _active;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (event.mouse.x < 17 || event.mouse.x >= 303 || event.mouse.y >= 153) {
|
|
|
|
activate(false);
|
|
|
|
return _active;
|
|
|
|
}
|
2009-09-15 08:54:06 +00:00
|
|
|
|
2009-09-03 20:59:17 +00:00
|
|
|
if (!_active)
|
|
|
|
return false;
|
2009-09-15 08:54:06 +00:00
|
|
|
|
2009-09-03 20:59:17 +00:00
|
|
|
hovered_obj = NULL;
|
2009-09-15 08:54:06 +00:00
|
|
|
|
2009-09-04 20:09:29 +00:00
|
|
|
for (int i = 0; i < 24; ++i) {
|
2009-09-03 20:59:17 +00:00
|
|
|
byte item = inventory[i];
|
|
|
|
if (item == 0)
|
|
|
|
continue;
|
2009-09-15 08:54:06 +00:00
|
|
|
|
2009-09-03 20:59:17 +00:00
|
|
|
graphics[i].hovered = graphics[i].rect.in(mouse);
|
|
|
|
if (graphics[i].hovered)
|
2009-09-26 15:04:09 +00:00
|
|
|
hovered_obj = &objects[item];
|
2009-09-03 20:59:17 +00:00
|
|
|
}
|
|
|
|
return true;
|
|
|
|
|
|
|
|
case Common::EVENT_LBUTTONDOWN: {
|
|
|
|
//check combine
|
2009-09-05 20:58:25 +00:00
|
|
|
if (hovered_obj == NULL)
|
2009-09-03 20:59:17 +00:00
|
|
|
return _active;
|
|
|
|
|
2009-09-05 20:58:25 +00:00
|
|
|
if (selected_obj == NULL) {
|
|
|
|
activate(false);
|
2009-09-26 15:04:09 +00:00
|
|
|
_engine->displayMessage(hovered_obj->description);
|
2009-09-05 20:58:25 +00:00
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2009-09-03 20:59:17 +00:00
|
|
|
int id1 = selected_obj->id;
|
|
|
|
int id2 = hovered_obj->id;
|
|
|
|
|
2009-11-08 22:37:03 +00:00
|
|
|
debug(0, "combine(%u, %u)!", id1, id2);
|
2009-09-15 07:41:05 +00:00
|
|
|
byte *table = res->dseg.ptr(0xC335);
|
2009-09-04 20:09:29 +00:00
|
|
|
while (table[0] != 0 && table[1] != 0) {
|
2009-09-03 20:59:17 +00:00
|
|
|
if (
|
2009-09-15 08:54:06 +00:00
|
|
|
(id1 == table[0] && id2 == table[1]) ||
|
|
|
|
(id2 == table[0] && id1 == table[1])
|
2009-09-03 20:59:17 +00:00
|
|
|
) {
|
|
|
|
remove(id1);
|
|
|
|
remove(id2);
|
2009-11-08 22:37:03 +00:00
|
|
|
debug(0, "adding object %u", table[2]);
|
2009-09-03 20:59:17 +00:00
|
|
|
add(table[2]);
|
|
|
|
uint16 msg = READ_LE_UINT16(table + 3);
|
|
|
|
_engine->playSoundNow(69);
|
2009-11-08 22:37:03 +00:00
|
|
|
_engine->displayMessage(msg);
|
2009-09-03 20:59:17 +00:00
|
|
|
activate(false);
|
|
|
|
resetSelectedObject();
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
table += 5;
|
|
|
|
}
|
|
|
|
_engine->displayMessage(0xc3e2);
|
|
|
|
activate(false);
|
|
|
|
resetSelectedObject();
|
|
|
|
return true;
|
2009-09-15 08:54:06 +00:00
|
|
|
}
|
|
|
|
|
2009-09-03 20:59:17 +00:00
|
|
|
case Common::EVENT_RBUTTONDOWN:
|
|
|
|
if (!_active)
|
|
|
|
return false;
|
2009-09-15 08:54:06 +00:00
|
|
|
|
2009-09-03 20:59:17 +00:00
|
|
|
if (hovered_obj != NULL) {
|
|
|
|
byte id = hovered_obj->id;
|
|
|
|
debug(0, "rclick object %u", id);
|
|
|
|
uint i = 0;
|
2009-09-15 07:41:05 +00:00
|
|
|
for (byte *table = res->dseg.ptr(0xBB6F + 3); //original offset + 3 bytes.
|
2009-09-15 08:54:06 +00:00
|
|
|
table[0] != 0 && i < 7; table += 3, ++i) {
|
2009-09-03 20:59:17 +00:00
|
|
|
if (table[0] == id) {
|
|
|
|
resetSelectedObject();
|
|
|
|
activate(false);
|
|
|
|
if (_engine->processCallback(READ_LE_UINT16(table + 1)))
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2009-09-15 08:54:06 +00:00
|
|
|
|
2009-09-03 20:59:17 +00:00
|
|
|
selected_obj = hovered_obj;
|
|
|
|
if (selected_obj)
|
2009-09-26 15:04:09 +00:00
|
|
|
debug(0, "selected object %s", selected_obj->name.c_str());
|
2009-09-03 20:59:17 +00:00
|
|
|
return true;
|
|
|
|
|
|
|
|
case Common::EVENT_KEYDOWN:
|
|
|
|
if (_active && event.kbd.keycode == Common::KEYCODE_ESCAPE) {
|
|
|
|
activate(false);
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
return false;
|
|
|
|
|
|
|
|
case Common::EVENT_LBUTTONUP:
|
|
|
|
case Common::EVENT_RBUTTONUP:
|
|
|
|
return _active;
|
2009-09-15 08:54:06 +00:00
|
|
|
|
|
|
|
default:
|
2009-09-03 20:59:17 +00:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void Inventory::Item::free() {
|
|
|
|
animation.free();
|
|
|
|
surface.free();
|
|
|
|
}
|
|
|
|
|
2009-09-15 07:41:05 +00:00
|
|
|
void Inventory::Item::render(Inventory *inventory, InventoryObject *obj, Graphics::Surface *dst) {
|
|
|
|
Resources *res = Resources::instance();
|
2009-09-15 08:54:06 +00:00
|
|
|
|
|
|
|
rect.render(dst, hovered ? 233 : 234);
|
2009-09-03 20:59:17 +00:00
|
|
|
if (obj->animated) {
|
|
|
|
if (animation.empty()) {
|
|
|
|
debug(0, "loading item %d from offset %x", obj->id, inventory->offset[obj->id - 1]);
|
|
|
|
inventory->items->seek(inventory->offset[obj->id - 1]);
|
2009-09-15 20:21:18 +00:00
|
|
|
animation.load(inventory->items, Animation::kTypeInventory);
|
2009-09-03 20:59:17 +00:00
|
|
|
}
|
2009-09-07 20:38:02 +00:00
|
|
|
if (hovered) {
|
2009-09-15 07:41:05 +00:00
|
|
|
Surface *s = animation.currentFrame();
|
2009-09-07 20:38:02 +00:00
|
|
|
if (animation.currentIndex() == 0)
|
|
|
|
s = animation.currentFrame();
|
|
|
|
if (s != NULL)
|
|
|
|
s->render(dst, rect.left + 1, rect.top + 1);
|
|
|
|
} else {
|
2009-09-15 07:41:05 +00:00
|
|
|
Surface *s = animation.firstFrame();
|
2009-09-07 20:38:02 +00:00
|
|
|
if (s != NULL)
|
|
|
|
s->render(dst, rect.left + 1, rect.top + 1);
|
|
|
|
}
|
2009-09-03 20:59:17 +00:00
|
|
|
} else {
|
|
|
|
if (surface.empty()) {
|
|
|
|
debug(0, "loading item %d from offset %x", obj->id, inventory->offset[obj->id - 1]);
|
|
|
|
inventory->items->seek(inventory->offset[obj->id - 1]);
|
2009-09-15 20:21:18 +00:00
|
|
|
surface.load(inventory->items, Surface::kTypeOns);
|
2009-09-03 20:59:17 +00:00
|
|
|
}
|
|
|
|
surface.render(dst, rect.left + 1, rect.top + 1);
|
|
|
|
}
|
|
|
|
|
|
|
|
Common::String name;
|
|
|
|
if (inventory->selected_obj && inventory->selected_obj != inventory->hovered_obj) {
|
|
|
|
name = inventory->selected_obj->name;
|
|
|
|
name += " & ";
|
|
|
|
}
|
|
|
|
name += obj->name;
|
2009-09-15 08:54:06 +00:00
|
|
|
|
2009-09-03 20:59:17 +00:00
|
|
|
if (hovered) {
|
2009-11-08 19:11:57 +00:00
|
|
|
int w = res->font7.render(NULL, 0, 0, name, 0xd1, true);
|
|
|
|
res->font7.render(dst, (320 - w) / 2, 180, name, 0xd1, true);
|
2009-09-03 20:59:17 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2009-09-15 07:41:05 +00:00
|
|
|
void Inventory::render(Graphics::Surface *surface) {
|
2009-09-03 20:59:17 +00:00
|
|
|
if (!_active)
|
|
|
|
return;
|
|
|
|
|
|
|
|
background.render(surface);
|
2009-09-15 08:54:06 +00:00
|
|
|
|
2009-09-03 20:59:17 +00:00
|
|
|
for (int y = 0; y < 4; y++) {
|
|
|
|
for (int x = 0; x < 6; x++) {
|
|
|
|
int idx = x + 6 * y;
|
|
|
|
byte item = inventory[idx];
|
|
|
|
if (item == 0)
|
|
|
|
continue;
|
2009-09-15 08:54:06 +00:00
|
|
|
|
2009-09-03 20:59:17 +00:00
|
|
|
//debug(0, "%d,%d -> %u", x0, y0, item);
|
2009-09-15 08:54:06 +00:00
|
|
|
|
2009-09-26 15:04:09 +00:00
|
|
|
InventoryObject *obj = &objects[item];
|
2009-09-03 20:59:17 +00:00
|
|
|
graphics[idx].render(this, obj, surface);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2009-09-04 20:08:33 +00:00
|
|
|
|
|
|
|
} // End of namespace TeenAgent
|