mirror of
https://github.com/libretro/scummvm.git
synced 2025-01-10 11:51:52 +00:00
227 lines
6.2 KiB
C++
227 lines
6.2 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.
|
|
*
|
|
*/
|
|
|
|
#include "common/scummsys.h"
|
|
#include "mads/mads.h"
|
|
#include "mads/inventory.h"
|
|
|
|
namespace MADS {
|
|
|
|
void InventoryObject::synchronize(Common::Serializer &s) {
|
|
s.syncAsUint16LE(_descId);
|
|
s.syncAsUint16LE(_roomNumber);
|
|
s.syncAsByte(_article);
|
|
s.syncAsByte(_vocabCount);
|
|
s.syncAsByte(_qualitiesCount);
|
|
s.skip(1);
|
|
|
|
for (int i = 0; i < MAX_VOCAB; ++i) {
|
|
s.syncAsUint16LE(_vocabList[i]._vocabId);
|
|
s.syncAsByte(_vocabList[i]._verbType);
|
|
s.syncAsByte(_vocabList[i]._prepType);
|
|
}
|
|
|
|
for (int i = 0; i < MAX_QUALITIES; ++i)
|
|
s.syncAsByte(_qualityId[i]);
|
|
for (int i = 0; i < MAX_QUALITIES; ++i)
|
|
s.syncAsSint32LE(_qualityValue[i]);
|
|
}
|
|
|
|
bool InventoryObject::hasQuality(int qualityId) const {
|
|
for (int i = 0; i < _qualitiesCount; ++i) {
|
|
if (_qualityId[i] == qualityId)
|
|
return true;
|
|
}
|
|
|
|
return false;
|
|
}
|
|
|
|
void InventoryObject::setQuality(int qualityId, int qualityValue) {
|
|
for (int i = 0; i < _qualitiesCount; ++i) {
|
|
if (_qualityId[i] == qualityId) {
|
|
_qualityValue[i] = qualityValue;
|
|
}
|
|
}
|
|
}
|
|
|
|
int InventoryObject::getQuality(int qualityId) const {
|
|
for (int i = 0; i < _qualitiesCount; ++i) {
|
|
if (_qualityId[i] == qualityId) {
|
|
return _qualityValue[i];
|
|
}
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
/*------------------------------------------------------------------------*/
|
|
|
|
void InventoryObjects::load() {
|
|
File f("*OBJECTS.DAT");
|
|
int count = f.readUint16LE();
|
|
Common::Serializer s(&f, nullptr);
|
|
|
|
// Load the objects data
|
|
reserve(count);
|
|
for (int i = 0; i < count; ++i) {
|
|
InventoryObject obj;
|
|
obj.synchronize(s);
|
|
push_back(obj);
|
|
|
|
// If it's for the player's inventory, add the index to the inventory list
|
|
if (obj._roomNumber == PLAYER_INVENTORY) {
|
|
_inventoryList.push_back(i);
|
|
assert(_inventoryList.size() <= 32);
|
|
}
|
|
}
|
|
}
|
|
|
|
void InventoryObjects::synchronize(Common::Serializer &s) {
|
|
int count = size();
|
|
s.syncAsUint16LE(count);
|
|
|
|
if (s.isSaving()) {
|
|
// Store the data for each object in the inventory lsit
|
|
for (int idx = 0; idx < count; ++idx)
|
|
(*this)[idx].synchronize(s);
|
|
|
|
// Synchronize the player's inventory
|
|
_inventoryList.synchronize(s);
|
|
} else {
|
|
clear();
|
|
|
|
// Read in each object
|
|
reserve(count);
|
|
for (int i = 0; i < count; ++i) {
|
|
InventoryObject obj;
|
|
obj.synchronize(s);
|
|
push_back(obj);
|
|
}
|
|
|
|
// Synchronize the player's inventory
|
|
_inventoryList.synchronize(s);
|
|
}
|
|
}
|
|
|
|
void InventoryObjects::setRoom(int objectId, int sceneNumber) {
|
|
InventoryObject &obj = (*this)[objectId];
|
|
|
|
if (obj._roomNumber == PLAYER_INVENTORY)
|
|
removeFromInventory(objectId, 1);
|
|
|
|
if (sceneNumber == PLAYER_INVENTORY)
|
|
addToInventory(objectId);
|
|
else
|
|
obj._roomNumber = sceneNumber;
|
|
}
|
|
|
|
bool InventoryObjects::isInRoom(int objectId) const {
|
|
return objectId >= 0 && (*this)[objectId]._roomNumber == _vm->_game->_scene._currentSceneId;
|
|
}
|
|
|
|
bool InventoryObjects::isInInventory(int objectId) const {
|
|
return objectId >= 0 && (*this)[objectId]._roomNumber == PLAYER_INVENTORY;
|
|
}
|
|
|
|
void InventoryObjects::addToInventory(int objectId) {
|
|
assert(_inventoryList.size() < 32);
|
|
UserInterface &userInterface = _vm->_game->_scene._userInterface;
|
|
|
|
if (!isInInventory(objectId)) {
|
|
_inventoryList.push_back(objectId);
|
|
userInterface._selectedInvIndex = _inventoryList.size() - 1;
|
|
userInterface._inventoryTopIndex = CLIP(userInterface._inventoryTopIndex,
|
|
0, userInterface._selectedInvIndex);
|
|
|
|
if ((userInterface._inventoryTopIndex + 5) <= (int)_inventoryList.size())
|
|
userInterface._inventoryTopIndex = _inventoryList.size() - 5;
|
|
userInterface._inventoryChanged = true;
|
|
|
|
(*this)[objectId]._roomNumber = PLAYER_INVENTORY;
|
|
|
|
if (_vm->_game->_kernelMode == KERNEL_ACTIVE_CODE &&
|
|
_vm->_game->_screenObjects._inputMode == kInputBuildingSentences) {
|
|
userInterface.categoryChanged();
|
|
userInterface.selectObject(userInterface._selectedInvIndex);
|
|
}
|
|
}
|
|
}
|
|
|
|
void InventoryObjects::removeFromInventory(int objectId, int newScene) {
|
|
Scene &scene = _vm->_game->_scene;
|
|
UserInterface &userInterface = scene._userInterface;
|
|
|
|
// Scan the inventory list for the object
|
|
int invIndex = -1;
|
|
for (int idx = 0; idx < (int)_inventoryList.size() && invIndex == -1; ++idx) {
|
|
if (_inventoryList[idx] == objectId)
|
|
invIndex = idx;
|
|
}
|
|
|
|
// If the object isn't in the player's inventory, stop here
|
|
if (invIndex < 0)
|
|
return;
|
|
|
|
int selectedIndex = userInterface._selectedInvIndex;
|
|
bool noSelection = selectedIndex < 0;
|
|
|
|
if (_vm->_game->_kernelMode == KERNEL_ACTIVE_CODE &&
|
|
_vm->_game->_screenObjects._inputMode == kInputBuildingSentences)
|
|
userInterface.selectObject(-1);
|
|
|
|
// Remove the item from the inventory list
|
|
_inventoryList.remove_at(invIndex);
|
|
|
|
if (invIndex > userInterface._inventoryTopIndex) {
|
|
userInterface._inventoryTopIndex = MAX(userInterface._inventoryTopIndex, 0);
|
|
}
|
|
|
|
userInterface._inventoryChanged = true;
|
|
(*this)[objectId]._roomNumber = newScene;
|
|
|
|
int newIndex = selectedIndex;
|
|
if (!noSelection) {
|
|
if (newIndex >= invIndex)
|
|
--newIndex;
|
|
if (newIndex < 0 && size() > 0)
|
|
newIndex = 0;
|
|
}
|
|
|
|
if (_vm->_game->_kernelMode == KERNEL_ACTIVE_CODE &&
|
|
_vm->_game->_screenObjects._inputMode == kInputBuildingSentences) {
|
|
userInterface.categoryChanged();
|
|
userInterface.selectObject(newIndex);
|
|
}
|
|
}
|
|
|
|
int InventoryObjects::getIdFromDesc(int descId) {
|
|
for (int i = 0; i < (int)size(); ++i) {
|
|
InventoryObject &obj = (*this)[i];
|
|
if (obj._descId == descId)
|
|
return i;
|
|
}
|
|
|
|
return -1;
|
|
}
|
|
|
|
} // End of namespace MADS
|