mirror of
https://github.com/scummvm/scummvm.git
synced 2025-04-02 10:52:32 -04:00
276 lines
7.2 KiB
C++
276 lines
7.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 3 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, see <http://www.gnu.org/licenses/>.
|
|
*
|
|
*/
|
|
|
|
#include "darkseed/darkseed.h"
|
|
#include "darkseed/objects.h"
|
|
|
|
#include "darkseed/object_name_tables.h"
|
|
|
|
namespace Darkseed {
|
|
|
|
Objects::Objects() {
|
|
_objectVar.resize(MAX_OBJECTS);
|
|
_objectRunningCode.resize(MAX_OBJECTS);
|
|
_moveObjectXY.resize(MAX_OBJECTS);
|
|
_moveObjectRoom.resize(MAX_OBJECTS); // The original only allocates 42 entries here but writes 199 in the save file!
|
|
_objectNames.resize(MAX_OBJECTS);
|
|
reset();
|
|
}
|
|
|
|
void Objects::reset() {
|
|
for (int i = 0; i < MAX_OBJECTS; i++) {
|
|
_objectVar[i] = 0;
|
|
_objectRunningCode[i] = 0;
|
|
_moveObjectXY[i].x = 0; // TODO verify this is the correct reset state for these XY vars.
|
|
_moveObjectXY[i].y = 0;
|
|
_moveObjectRoom[i] = i < 42 ? 0xff : 0; // Hack for weird behaviour in original engine.
|
|
}
|
|
// Initial object state.
|
|
setVar(52, 1);
|
|
setVar(112, 0);
|
|
setVar(62, 0);
|
|
}
|
|
|
|
void Objects::setVar(uint16 varIdx, int16 newValue) {
|
|
if (varIdx >= MAX_OBJECTS) {
|
|
error("setVar: Object Index out of range! %d", varIdx);
|
|
}
|
|
_objectVar[varIdx] = newValue;
|
|
}
|
|
|
|
int16 Objects::getVar(uint16 varIdx) {
|
|
if (varIdx >= MAX_OBJECTS) {
|
|
error("getVar: Object Index out of range! %d", varIdx);
|
|
}
|
|
return _objectVar[varIdx];
|
|
}
|
|
|
|
Common::Point Objects::getMoveObjectPosition(uint8 objIdx) {
|
|
if (objIdx >= MAX_OBJECTS) {
|
|
error("getMoveObjectPosition: Object Index out of range! %d", objIdx);
|
|
}
|
|
return _moveObjectXY[objIdx];
|
|
}
|
|
|
|
void Objects::setMoveObjectPosition(uint8 objIdx, const Common::Point &newPoint) {
|
|
if (objIdx >= MAX_OBJECTS) {
|
|
error("setMoveObjectPosition: Object Index out of range! %d", objIdx);
|
|
}
|
|
_moveObjectXY[objIdx] = newPoint;
|
|
}
|
|
|
|
void Objects::setMoveObjectX(uint8 objIdx, int16 xPos) {
|
|
if (objIdx >= MAX_OBJECTS) {
|
|
error("setMoveObjectX: Object Index out of range! %d", objIdx);
|
|
}
|
|
_moveObjectXY[objIdx].x = xPos;
|
|
}
|
|
|
|
int16 &Objects::operator[](uint16 varIdx) {
|
|
if (varIdx >= MAX_OBJECTS) {
|
|
error("getVar: Object Index out of range! %d", varIdx);
|
|
}
|
|
return _objectVar[varIdx];
|
|
}
|
|
const int16 &Objects::operator[](uint16 varIdx) const {
|
|
if (varIdx >= MAX_OBJECTS) {
|
|
error("getVar: Object Index out of range! %d", varIdx);
|
|
}
|
|
return _objectVar[varIdx];
|
|
}
|
|
|
|
void Objects::loadObjectNames() {
|
|
auto lang = g_engine->getLanguage();
|
|
if (lang == Common::KO_KOR) {
|
|
loadKoreanObjectNames();
|
|
} else if (lang == Common::ZH_ANY) {
|
|
loadChineseObjectNames();
|
|
} else {
|
|
for (int i = 0; i < MAX_OBJECTS; i++) {
|
|
switch (lang) {
|
|
case Common::FR_FRA: _objectNames[i] = Common::U32String(objectNameTbl_fr[i]); break;
|
|
case Common::DE_DEU: _objectNames[i] = Common::U32String(objectNameTbl_de[i]); break;
|
|
case Common::ES_ESP: _objectNames[i] = Common::U32String(objectNameTbl_es[i]); break;
|
|
default: _objectNames[i] = Common::U32String(objectNameTbl_en[i]); break;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
void Objects::loadKoreanObjectNames() {
|
|
Common::File file;
|
|
if (!file.open("tos.exe")) {
|
|
error("Failed to open TOS.EXE");
|
|
}
|
|
|
|
for (int i = 0; i < MAX_OBJECTS; i++) {
|
|
file.seek(0x22f62 + i * 4);
|
|
uint16 offset = file.readUint16LE();
|
|
file.seek(0x20990 + offset);
|
|
_objectNames[i] = readU32String(file);
|
|
}
|
|
|
|
file.close();
|
|
}
|
|
|
|
void Objects::loadChineseObjectNames() {
|
|
Common::File file;
|
|
if (!file.open("zh_objectnames.dat")) {
|
|
error("Failed to open zh_objectnames.dat");
|
|
}
|
|
|
|
for (int i = 0; i < MAX_OBJECTS; i++) {
|
|
file.seek(i * 21);
|
|
_objectNames[i] = readU32String(file);
|
|
}
|
|
|
|
file.close();
|
|
}
|
|
|
|
Common::U32String Objects::readU32String(Common::SeekableReadStream &readStream) {
|
|
Common::U32String str;
|
|
uint8 byte = readStream.readByte();
|
|
while (byte != 0) {
|
|
if (byte & 0x80) {
|
|
uint8 byte2 = readStream.readByte();
|
|
if (readStream.err()) {
|
|
error("Failed to read byte from stream!");
|
|
}
|
|
str += (int)byte << 8 | byte2;
|
|
} else {
|
|
str += byte;
|
|
}
|
|
byte = readStream.readByte();
|
|
}
|
|
return str;
|
|
}
|
|
|
|
static constexpr uint16 eyeDescriptionsTbl[] = {
|
|
0, 0, 0, 0,
|
|
0, 0, 0, 513,
|
|
0, 421, 0, 521,
|
|
0, 0, 713, 0,
|
|
791, 812, 0, 743,
|
|
0, 0, 661, 166,
|
|
0, 0, 0, 0,
|
|
500, 0, 0, 423,
|
|
425, 427, 418, 667,
|
|
503, 505, 507, 509,
|
|
511, 755, 652, 0,
|
|
728, 0, 0, 43,
|
|
0, 0, 0, 0,
|
|
192, 0, 0, 0,
|
|
0, 893, 845, 0,
|
|
0, 452, 721, 0,
|
|
483, 0, 466, 466,
|
|
466, 0, 0, 705,
|
|
0, 0, 0, 0,
|
|
0, 0, 0, 829,
|
|
552, 0, 0, 0,
|
|
0, 0, 711, 608,
|
|
610, 606, 604, 602,
|
|
600, 598, 596, 594,
|
|
592, 590, 588, 0,
|
|
0, 732, 166, 0,
|
|
0, 0, 0, 0,
|
|
0, 0, 0, 241,
|
|
231, 750, 815, 826,
|
|
838, 794, 797, 806,
|
|
802, 440, 448, 117,
|
|
259, 271, 305, 90,
|
|
161, 136, 376, 398,
|
|
414, 474, 477, 480,
|
|
0, 0, 999, 252,
|
|
0, 0, 170, 182,
|
|
212, 219, 284, 315,
|
|
328, 337, 346, 356,
|
|
515, 526, 533, 0,
|
|
547, 561, 570, 575,
|
|
613, 615, 620, 624,
|
|
636, 638, 641, 643,
|
|
645, 0, 673, 677,
|
|
680, 683, 688, 717,
|
|
726, 746, 0, 759,
|
|
765, 780, 787, 818,
|
|
822, 824, 0, 855,
|
|
862, 0, 880, 887,
|
|
891, 900, 0, 724,
|
|
671, 321, 163
|
|
};
|
|
|
|
int Objects::getEyeDescriptionTosIdx(uint16 objNum) {
|
|
if (objNum >= MAX_OBJECTS) {
|
|
error("getEyeDescriptionTosIdx: Object Index out of range! %d", objNum);
|
|
}
|
|
return eyeDescriptionsTbl[objNum];
|
|
}
|
|
|
|
int Objects::getMoveObjectRoom(uint16 idx) {
|
|
if (idx >= MAX_OBJECTS) {
|
|
error("getMoveObjectRoom: index out of range.");
|
|
}
|
|
return _moveObjectRoom[idx];
|
|
}
|
|
|
|
void Objects::setMoveObjectRoom(uint16 idx, uint8 value) {
|
|
if (idx >= MAX_OBJECTS) {
|
|
error("setMoveObjectRoom: index out of range.");
|
|
}
|
|
_moveObjectRoom[idx] = value;
|
|
}
|
|
|
|
int16 Objects::getObjectRunningCode(int idx) {
|
|
if (idx >= MAX_OBJECTS) {
|
|
error("getObjectRunningCode: index out of range.");
|
|
}
|
|
return _objectRunningCode[idx];
|
|
}
|
|
|
|
void Objects::setObjectRunningCode(int idx, int16 value) {
|
|
if (idx >= MAX_OBJECTS) {
|
|
error("setObjectRunningCode: index out of range.");
|
|
}
|
|
_objectRunningCode[idx] = value;
|
|
}
|
|
|
|
const Common::U32String &Objects::getObjectName(int idx) {
|
|
if (idx < 0 || idx >= MAX_OBJECTS) {
|
|
error("getObjectName: index out of range.");
|
|
}
|
|
|
|
return _objectNames[idx];
|
|
}
|
|
|
|
static inline void syncPoint(Common::Serializer &s, Common::Point &value) {
|
|
s.syncAsSint16LE(value.x);
|
|
s.syncAsSint16LE(value.y);
|
|
}
|
|
|
|
Common::Error Objects::sync(Common::Serializer &s) {
|
|
s.syncArray(_objectVar.data(), _objectVar.size(), Common::Serializer::Sint16LE);
|
|
s.syncArray(_objectRunningCode.data(), _objectRunningCode.size(), Common::Serializer::Sint16LE);
|
|
s.syncArray(_objectRunningCode.data(), _objectRunningCode.size(), Common::Serializer::Sint16LE);
|
|
s.syncArray(_moveObjectXY.data(), _moveObjectXY.size(), syncPoint);
|
|
s.syncArray(_moveObjectRoom.data(), _moveObjectRoom.size(), Common::Serializer::Byte);
|
|
return Common::kNoError;
|
|
}
|
|
|
|
} // End of namespace Darkseed
|