scummvm/engines/voyeur/files.cpp
2013-12-08 22:57:11 -05:00

1442 lines
36 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 "voyeur/files.h"
#include "voyeur/graphics.h"
#include "voyeur/voyeur.h"
namespace Voyeur {
#define BOLT_GROUP_SIZE 16
BoltFilesState::BoltFilesState() {
_curLibPtr = NULL;
_curGroupPtr = NULL;
_curMemberPtr = NULL;
_curMemInfoPtr = NULL;
_fromGroupFlag = 0;
_xorMask = 0;
_encrypt = false;
_curFilePosition = 0;
_bufferEnd = 0;
_bufferBegin = 0;
_bytesLeft = 0;
_bufSize = 0;
_bufStart = NULL;
_bufPos = NULL;
_historyIndex = 0;
_runLength = 0;
_decompState = 0;
_runType = 0;
_runValue = 0;
_runOffset = 0;
Common::fill(&_historyBuffer[0], &_historyBuffer[0x200], 0);
_curFd = NULL;
_boltPageFrame = NULL;
}
#define NEXT_BYTE if (--_bytesLeft < 0) nextBlock()
byte *BoltFilesState::decompress(byte *buf, int size, int mode) {
if (!buf) {
buf = new byte[size];
Common::fill(buf, buf + size, 0);
}
byte *bufP = buf;
if (mode & 8) {
_decompState = 1;
_runType = 0;
_runLength = size;
}
while (size > 0) {
if (!_decompState) {
NEXT_BYTE;
byte nextByte = *_bufPos++;
switch (nextByte & 0xC0) {
case 0:
_runType = 0;
_runLength = 30 - (nextByte & 0x1f) + 1;
break;
case 0x40:
_runType = 1;
_runLength = 35 - (nextByte & 0x1f);
NEXT_BYTE;
_runOffset = *_bufPos++ + ((nextByte & 0x20) << 3);
break;
case 0x80:
_runType = 1;
_runLength = (nextByte & 0x20) ? ((32 - (nextByte & 0x1f)) << 2) + 2 :
(32 - (nextByte & 0x1f)) << 2;
NEXT_BYTE;
_runOffset = *_bufPos++ << 1;
break;
default:
_runType = 2;
if (nextByte & 0x20) {
_runLength = 0;
} else {
NEXT_BYTE;
_runLength = ((32 - (nextByte & 0x1f)) + (*_bufPos++ << 5)) << 2;
NEXT_BYTE;
_bufPos++;
NEXT_BYTE;
_runValue = *_bufPos++;
}
break;
}
_runOffset = _historyIndex - _runOffset;
}
int runOffset = _runOffset & 0x1ff;
int len;
if (_runLength <= size) {
len = _runLength;
_decompState = 0;
} else {
_decompState = 1;
len = size;
_runLength -= size;
if (_runType == 1)
_runOffset += len;
}
// Reduce the remaining size
size -= len;
// Handle the run lengths
switch (_runType) {
case 0:
while (len-- > 0) {
NEXT_BYTE;
byte v = *_bufPos++;
_historyBuffer[_historyIndex] = v;
*bufP++ = v;
_historyIndex = (_historyIndex + 1) & 0x1ff;
}
break;
case 1:
while (len-- > 0) {
_historyBuffer[_historyIndex] = _historyBuffer[runOffset];
*bufP++ = _historyBuffer[runOffset];
_historyIndex = (_historyIndex + 1) & 0x1ff;
runOffset = (runOffset + 1) & 0x1ff;
}
break;
default:
while (len-- > 0) {
_historyBuffer[_historyIndex] = _runValue;
*bufP++ = _runValue;
_historyIndex = (_historyIndex + 1) & 0x1ff;
}
break;
}
}
return buf;
}
#undef NEXT_BYTE
void BoltFilesState::nextBlock() {
if (_curFilePosition != _bufferEnd)
_curFd->seek(_bufferEnd);
_bufferBegin = _bufferEnd;
int bytesRead = _curFd->read(_bufStart, _bufSize);
_bufferEnd = _curFilePosition = _bufferBegin + bytesRead;
_bytesLeft = bytesRead - 1;
_bufPos = _bufStart;
}
/*------------------------------------------------------------------------*/
FilesManager::FilesManager() {
_decompressSize = 0x7000;
}
bool FilesManager::openBoltLib(const Common::String &filename, BoltFile *&boltFile) {
if (boltFile != NULL) {
_boltFilesState._curLibPtr = boltFile;
return true;
}
// Create the bolt file interface object and load the index
if (filename == "bvoy.blt")
boltFile = _boltFilesState._curLibPtr = new BVoyBoltFile(_boltFilesState);
else if (filename == "stampblt.blt")
boltFile = _boltFilesState._curLibPtr = new StampBoltFile(_boltFilesState);
else
error("Unknown bolt file specified");
return true;
}
byte *FilesManager::fload(const Common::String &filename, int *size) {
Common::File f;
int filesize;
byte *data = NULL;
if (f.open(filename)) {
// Read in the file
filesize = f.size();
data = new byte[filesize];
f.read(data, filesize);
} else {
filesize = 0;
}
if (size)
*size = filesize;
return data;
}
/*------------------------------------------------------------------------*/
BoltFile::BoltFile(const Common::String &filename, BoltFilesState &state): _state(state) {
_state._curFd = &_file;
if (!_file.open(filename))
error("Could not open %s", filename.c_str());
_state._curFilePosition = 0;
// Read in the file header
byte header[16];
_file.read(&header[0], 16);
if (strncmp((const char *)&header[0], "BOLT", 4) != 0)
error("Tried to load non-bolt file");
int totalGroups = header[11] ? header[11] : 0x100;
for (int i = 0; i < totalGroups; ++i)
_groups.push_back(BoltGroup(_state._curFd));
}
BoltFile::~BoltFile() {
_file.close();
if (_state._curFd == &_file)
_state._curFd = NULL;
}
bool BoltFile::getBoltGroup(uint32 id) {
++_state._fromGroupFlag;
_state._curLibPtr = this;
_state._curGroupPtr = &_groups[(id >> 8) & 0xff];
if (!_state._curGroupPtr->_loaded) {
// Load the group index
_state._curGroupPtr->load();
}
if (_state._curGroupPtr->_callInitGro)
initGro();
if ((id >> 16) != 0) {
id &= 0xff00;
for (int idx = 0; idx < _state._curGroupPtr->_count; ++idx, ++id) {
byte *member = getBoltMember(id);
assert(member);
}
} else if (!_state._curGroupPtr->_processed) {
_state._curGroupPtr->_processed = true;
_state._curGroupPtr->load();
}
resolveAll();
--_state._fromGroupFlag;
return true;
}
void BoltFile::freeBoltGroup(uint32 id) {
_state._curLibPtr = this;
_state._curGroupPtr = &_groups[(id >> 8) & 0xff];
// Unload the group
_state._curGroupPtr->unload();
}
void BoltFile::freeBoltMember(uint32 id) {
// TODO: Determine whether this is needed
warning("TODO: BoltFile::freeBoltMember");
}
BoltEntry &BoltFile::getBoltEntryFromLong(uint32 id) {
BoltGroup &group = _groups[id >> 24];
assert(group._loaded);
BoltEntry &entry = group._entries[(id >> 16) & 0xff];
assert(!entry.hasResource() || (id & 0xffff) == 0);
return entry;
}
BoltEntry &BoltFile::boltEntry(uint16 id) {
BoltGroup &group = _groups[id >> 8];
assert(group._loaded);
BoltEntry &entry = group._entries[id & 0xff];
assert(entry.hasResource());
return entry;
}
PictureResource *BoltFile::getPictureResource(uint32 id) {
if ((int32)id == -1)
return NULL;
if (id & 0xffff)
id <<= 16;
return getBoltEntryFromLong(id)._picResource;
}
CMapResource *BoltFile::getCMapResource(uint32 id) {
if ((int32)id == -1)
return NULL;
if (id & 0xffff)
id <<= 16;
return getBoltEntryFromLong(id)._cMapResource;
}
byte *BoltFile::memberAddr(uint32 id) {
BoltGroup &group = _groups[id >> 8];
if (!group._loaded)
return NULL;
// If an entry already has a processed representation, we shouldn't
// still be accessing the raw data
BoltEntry &entry = group._entries[id & 0xff];
assert(!entry.hasResource());
return entry._data;
}
byte *BoltFile::memberAddrOffset(uint32 id) {
BoltGroup &group = _groups[id >> 24];
if (!group._loaded)
return NULL;
// If an entry already has a processed representation, we shouldn't
// still be accessing the raw data
BoltEntry &entry = group._entries[(id >> 16) & 0xff];
assert(!entry.hasResource());
return entry._data + (id & 0xffff);
}
/**
* Resolves an Id to an offset within a loaded resource
*/
void BoltFile::resolveIt(uint32 id, byte **p) {
if ((int32)id == -1) {
*p = NULL;
} else {
byte *ptr = memberAddrOffset(id);
if (ptr) {
*p = ptr;
} else {
*p = NULL;
assert(_state._resolves.size() < 1000);
_state._resolves.push_back(ResolveEntry(id, p));
}
}
}
void BoltFile::resolveFunction(uint32 id, GraphicMethodPtr *fn) {
if ((int32)id == -1) {
*fn = NULL;
} else {
error("Function fnTermGro array not supported");
}
}
/**
* Resolve any data references to within resources that weren't
* previously loaded, but are now
*/
void BoltFile::resolveAll() {
for (uint idx = 0; idx < _state._resolves.size(); ++idx)
*_state._resolves[idx]._p = memberAddrOffset(_state._resolves[idx]._id);
}
byte *BoltFile::getBoltMember(uint32 id) {
_state._curLibPtr = this;
// Get the group, and load it's entry list if not already loaded
_state._curGroupPtr = &_groups[(id >> 8) & 0xff];
if (!_state._curGroupPtr->_loaded)
_state._curGroupPtr->load();
// Get the entry
_state._curMemberPtr = &_state._curGroupPtr->_entries[id & 0xff];
if (_state._curMemberPtr->_field1)
initMem(_state._curMemberPtr->_field1);
// Return the data for the entry if it's already been loaded
if (_state._curMemberPtr->_data)
return _state._curMemberPtr->_data;
_state._xorMask = _state._curMemberPtr->_xorMask;
_state._encrypt = (_state._curMemberPtr->_mode & 0x10) != 0;
if (_state._curGroupPtr->_processed) {
// TODO: Figure out weird access type. Uncompressed read perhaps?
//int fileDiff = _state._curGroupPtr->_fileOffset - _state._curMemberPtr->_fileOffset;
error("TODO: processed bolt flag");
} else {
_state._bufStart = _state._decompressBuf;
_state._bufSize = DECOMPRESS_SIZE;
if ((_state._curFd != &_file) || (_state._curMemberPtr->_fileOffset < _state._bufferBegin)
|| (_state._curMemberPtr->_fileOffset >= _state._bufferEnd)) {
_state._bytesLeft = 0;
_state._bufPos = _state._bufStart;
_state._bufferBegin = -1;
_state._bufferEnd = _state._curMemberPtr->_fileOffset;
} else {
_state._bufPos = _state._curMemberPtr->_fileOffset - _state._bufferBegin + _state._bufStart;
_state._bytesLeft = _state._bufSize - (_state._bufPos - _state._bufStart);
}
}
_state._decompState = 0;
_state._historyIndex = 0;
// Initialise the resource
assert(_state._curMemberPtr->_initMethod < 25);
initResource(_state._curMemberPtr->_initMethod);
return _state._curMemberPtr->_data;
}
void BoltFile::initDefault() {
_state._curMemberPtr->_data = _state.decompress(0, _state._curMemberPtr->_size,
_state._curMemberPtr->_mode);
}
/*------------------------------------------------------------------------*/
BVoyBoltFile::BVoyBoltFile(BoltFilesState &state): BoltFile("bvoy.blt", state) {
}
void BVoyBoltFile::initResource(int resType) {
switch (resType) {
case 8:
sInitPic();
break;
case 10:
vInitCMap();
break;
case 11:
vInitCycl();
break;
case 15:
initViewPort();
break;
case 16:
initViewPortList();
break;
case 17:
initFont();
break;
case 18:
initFontInfo();
break;
case 19:
initSoundMap();
break;
default:
initDefault();
break;
}
}
void BVoyBoltFile::initViewPort() {
initDefault();
ViewPortResource *viewPort;
byte *src = _state._curMemberPtr->_data;
_state._curMemberPtr->_viewPortResource = viewPort = new ViewPortResource(_state, src);
// This is done post-constructor, since viewports can be self referential, so
// we ned the _viewPortResource field to have been set before resolving the pointer
viewPort->_parent = getBoltEntryFromLong(READ_LE_UINT32(src + 2))._viewPortResource;
}
void BVoyBoltFile::initViewPortList() {
initDefault();
ViewPortListResource *res;
_state._curMemberPtr->_viewPortListResource = res = new ViewPortListResource(
_state, _state._curMemberPtr->_data);
_state._vm->_graphicsManager._viewPortListPtr = res;
_state._vm->_graphicsManager._vPort = &res->_entries[0];
}
void BVoyBoltFile::initFontInfo() {
initDefault();
_state._curMemberPtr->_fontInfoResource = new FontInfoResource(
_state, _state._curMemberPtr->_data);
}
void BVoyBoltFile::initFont() {
initDefault();
_state._curMemberPtr->_fontResource = new FontResource(_state, _state._curMemberPtr->_data);
}
void BVoyBoltFile::initSoundMap() {
initDefault();
}
void BVoyBoltFile::sInitPic() {
// Read in the header data
_state._curMemberPtr->_data = _state.decompress(NULL, 24, _state._curMemberPtr->_mode);
_state._curMemberPtr->_picResource = new PictureResource(_state,
_state._curMemberPtr->_data);
}
void BVoyBoltFile::vInitCMap() {
initDefault();
_state._curMemberPtr->_cMapResource = new CMapResource(
_state, _state._curMemberPtr->_data);
}
void BVoyBoltFile::vInitCycl() {
initDefault();
_state._vm->_eventsManager.vStopCycle();
_state._curMemberPtr->_vInitCyclResource = new VInitCyclResource(
_state, _state._curMemberPtr->_data);
}
/*------------------------------------------------------------------------*/
StampBoltFile::StampBoltFile(BoltFilesState &state): BoltFile("stampblt.blt", state) {
}
void StampBoltFile::initResource(int resType) {
switch (resType) {
case 0:
initThread();
break;
case 6:
initPtr();
break;
case 24:
initControl();
break;
default:
initDefault();
break;
}
}
void StampBoltFile::initThread() {
initDefault();
_state._curMemberPtr->_threadResource = new ThreadResource(_state,
_state._curMemberPtr->_data);
}
void StampBoltFile::initPtr() {
initDefault();
_state._curMemberPtr->_ptrResource = new PtrResource(_state,
_state._curMemberPtr->_data);
}
void StampBoltFile::initControl() {
initDefault();
ControlResource *res;
_state._curMemberPtr->_controlResource = res = new ControlResource(_state,
_state._curMemberPtr->_data);
_state._vm->_controlGroupPtr = _state._curGroupPtr;
_state._vm->_controlPtr = res;
}
/*------------------------------------------------------------------------*/
BoltGroup::BoltGroup(Common::SeekableReadStream *f): _file(f) {
byte buffer[BOLT_GROUP_SIZE];
_loaded = false;
_file->read(&buffer[0], BOLT_GROUP_SIZE);
_processed = buffer[0] != 0;
_callInitGro = buffer[1] != 0;
_termGroIndex = buffer[2];
_count = buffer[3] ? buffer[3] : 256; // TODO: Added this in. Check it's okay
_fileOffset = READ_LE_UINT32(&buffer[8]);
}
BoltGroup::~BoltGroup() {
}
void BoltGroup::load() {
_file->seek(_fileOffset);
// Read the entries
for (int i = 0; i < _count; ++i)
_entries.push_back(BoltEntry(_file));
_loaded = true;
}
void BoltGroup::unload() {
if (!_loaded)
return;
_entries.clear();
_loaded = false;
}
/*------------------------------------------------------------------------*/
BoltEntry::BoltEntry(Common::SeekableReadStream *f): _file(f) {
_data = NULL;
_picResource = NULL;
_viewPortResource = NULL;
_viewPortListResource = NULL;
_fontResource = NULL;
_fontInfoResource = NULL;
_cMapResource = NULL;
_vInitCyclResource = NULL;
_ptrResource = NULL;
_controlResource = NULL;
_threadResource = NULL;
byte buffer[16];
_file->read(&buffer[0], 16);
_mode = buffer[0];
_field1 = buffer[1];
_initMethod = buffer[3];
_xorMask = buffer[4] & 0xff; // TODO: Is this right??
_size = READ_LE_UINT32(&buffer[4]);
_fileOffset = READ_LE_UINT32(&buffer[8]);
}
BoltEntry::~BoltEntry() {
delete[] _data;
delete _picResource;
delete _viewPortResource;
delete _viewPortListResource;
delete _fontResource;
delete _fontInfoResource;
delete _cMapResource;
delete _vInitCyclResource;
delete _ptrResource;
delete _controlResource;
}
void BoltEntry::load() {
// TODO: Currently, all entry loading and decompression is done in BoltFile::memberAddr.
// Ideally, a lot of the code should be moved here
}
/**
* Returns true if the given bolt entry has an attached resource
*/
bool BoltEntry::hasResource() const {
return _picResource || _viewPortResource || _viewPortListResource
|| _fontResource || _fontInfoResource || _cMapResource
|| _vInitCyclResource || _ptrResource || _controlResource
|| _threadResource;
}
/*------------------------------------------------------------------------*/
PictureResource::PictureResource(BoltFilesState &state, const byte *src) {
_flags = READ_LE_UINT16(src);
_select = src[2];
_pick = src[3];
_onOff = src[4];
_depth = src[5];
int xs = READ_LE_UINT16(&src[6]);
int ys = READ_LE_UINT16(&src[8]);
_bounds = Common::Rect(xs, ys, xs + READ_LE_UINT16(&src[10]),
ys + READ_LE_UINT16(&src[12]));
_maskData = READ_LE_UINT32(&src[14]);
_planeSize = READ_LE_UINT16(&src[22]);
_imgData = NULL;
int nbytes = _bounds.width() * _bounds.height();
if (_flags & 0x20) {
error("TODO: sInitPic flags&0x20");
} else if (_flags & 8) {
int mode = 0;
if (_bounds.width() == 320) {
mode = 147;
state._sImageShift = 2;
state._SVGAReset = false;
} else {
state._SVGAReset = true;
if (_bounds.width() == 640) {
if (_bounds.height() == 400) {
mode = 220;
state._sImageShift = 3;
} else {
mode = 221;
state._sImageShift = 3;
}
} else if (_bounds.width() == 800) {
mode = 222;
state._sImageShift = 3;
} else if (_bounds.width() == 1024) {
mode = 226;
state._sImageShift = 3;
}
}
if (mode != state._vm->_graphicsManager._SVGAMode) {
state._vm->_graphicsManager._SVGAMode = mode;
state._vm->_graphicsManager.clearPalette();
}
// byte *imgData = _imgData;
if (_flags & 0x10) {
// TODO: Figure out what it's doing. Looks like a direct clearing
// of the screen directly
} else {
// TODO: Figure out direct screen loading
}
} else {
if (_flags & 0x1000) {
if (!(_flags & 0x10))
nbytes = state._curMemberPtr->_size - 24;
int mask = (nbytes + 0x3FFF) >> 14;
_imgData = NULL;
if (state._boltPageFrame == 0)
state.EMSGetFrameAddr(&state._boltPageFrame);
if (state._boltPageFrame != 0) {
if (!state.EMSAllocatePages(&_planeSize)) {
_maskData = mask;
for (int idx = 0; idx < mask; ++idx)
state.EMSMapPageHandle(_planeSize, idx, idx);
state.decompress(state._boltPageFrame, nbytes, state._curMemberPtr->_mode);
return;
}
}
}
if (_flags & 0x10) {
_imgData = new byte[nbytes];
Common::fill(_imgData, _imgData + nbytes, 0);
} else {
_imgData = state.decompress(NULL, nbytes, state._curMemberPtr->_mode);
}
}
}
PictureResource::PictureResource() {
_flags = 0;
_select = 0;
_pick = 0;
_onOff = 0;
_depth = 0;
_maskData = 0;
_planeSize = 0;
_imgData = NULL;
}
PictureResource::PictureResource(int flags, int select, int pick, int onOff,
int depth, const Common::Rect &bounds, int maskData, byte *imgData,
int planeSize) {
_flags = flags;
_select = select;
_pick = pick;
_onOff = onOff;
_depth = depth;
_bounds = bounds;
_maskData = maskData;
_imgData = imgData;
_planeSize = planeSize;
}
PictureResource::~PictureResource() {
delete[] _imgData;
}
/*------------------------------------------------------------------------*/
ViewPortResource::ViewPortResource(BoltFilesState &state, const byte *src):
_fontChar(0, 0xff, 0xff, 0, 0, Common::Rect(), 0, NULL, 0), _state(state) {
_flags = READ_LE_UINT16(src);
_parent = NULL;
_pageCount = READ_LE_UINT16(src + 6);
_pageIndex = READ_LE_UINT16(src + 8);
_lastPage = READ_LE_UINT16(src + 10);
int xs = READ_LE_UINT16(src + 12);
int ys = READ_LE_UINT16(src + 14);
_bounds = Common::Rect(xs, ys, xs + READ_LE_UINT16(src + 16),
ys + READ_LE_UINT16(src + 18));
_field18 = READ_LE_UINT16(src + 0x18);
_currentPic = state._curLibPtr->getPictureResource(READ_LE_UINT32(src + 0x20));
_activePage = state._curLibPtr->getPictureResource(READ_LE_UINT32(src + 0x24));
_pages[0] = state._curLibPtr->getPictureResource(READ_LE_UINT32(src + 0x28));
_pages[1] = state._curLibPtr->getPictureResource(READ_LE_UINT32(src + 0x2C));
state._curLibPtr->resolveIt(READ_LE_UINT32(src + 0x30), &_field30);
// Get the rect list
for (int listIndex = 0; listIndex < 3; ++listIndex) {
_rectListCount[listIndex] = (int16)READ_LE_UINT16(src + 0x40 + 2 * listIndex);
uint32 id = READ_LE_UINT32(src + 0x34 + listIndex * 4);
if (id == -1) {
_rectListPtr[listIndex] = NULL;
} else {
_rectListPtr[listIndex] = new Common::Array<Common::Rect>();
if (_rectListCount[listIndex] > 0) {
int16 *rectList = (int16 *)state._curLibPtr->memberAddrOffset(id);
for (int i = 0; i < _rectListCount[listIndex]; ++i) {
int xs = FROM_LE_16(rectList[0]);
int ys = FROM_LE_16(rectList[1]);
_rectListPtr[i]->push_back(Common::Rect(xs, ys, xs + FROM_LE_16(rectList[2]),
ys + FROM_LE_16(rectList[3])));
}
}
}
}
xs = READ_LE_UINT16(src + 0x46);
ys = READ_LE_UINT16(src + 0x48);
_clipRect = Common::Rect(xs, ys, xs + READ_LE_UINT16(src + 0x4A),
ys + READ_LE_UINT16(src + 0x4C));
state._curLibPtr->resolveIt(READ_LE_UINT32(src + 0x7A), &_field7A);
state._curLibPtr->resolveFunction(READ_LE_UINT32(src + 0x7E), (GraphicMethodPtr *)&_fn1);
state._curLibPtr->resolveFunction(READ_LE_UINT32(src + 0x82), (GraphicMethodPtr *)&_setupFn);
state._curLibPtr->resolveFunction(READ_LE_UINT32(src + 0x86), (GraphicMethodPtr *)&_addFn);
state._curLibPtr->resolveFunction(READ_LE_UINT32(src + 0x8A), (GraphicMethodPtr *)&_restoreFn);
if (!_restoreFn && _addFn)
_addFn = &GraphicsManager::addRectNoSaveBack;
}
ViewPortResource::~ViewPortResource() {
for (int i = 0; i < 3; ++i)
delete _rectListPtr[i];
}
void ViewPortResource::setupViewPort(PictureResource *page, Common::Rect *clipRect,
ViewPortSetupPtr setupFn, ViewPortAddPtr addFn, ViewPortRestorePtr restoreFn) {
PictureResource *pic = _currentPic;
Common::Rect r = _bounds;
r.translate(pic->_bounds.left, pic->_bounds.top);
int xDiff, yDiff;
if (page) {
// Clip based on the passed picture resource
xDiff = page->_bounds.left - r.left;
yDiff = page->_bounds.top - r.top;
if (xDiff > 0) {
int width = r.width();
r.left = page->_bounds.left;
r.setWidth(xDiff <= width ? width - xDiff : 0);
}
if (yDiff > 0) {
int height = r.height();
r.top = page->_bounds.top;
r.setHeight(yDiff <= height ? height - yDiff : 0);
}
xDiff = r.right - page->_bounds.right;
yDiff = r.bottom - page->_bounds.bottom;
if (xDiff > 0)
r.setWidth(xDiff <= r.width() ? r.width() - xDiff : 0);
if (yDiff > 0)
r.setHeight(yDiff <= r.height() ? r.height() - yDiff : 0);
}
if (clipRect) {
// Clip based on the passed clip rectangles
xDiff = clipRect->left - r.left;
yDiff = clipRect->top - r.top;
if (xDiff > 0) {
int width = r.width();
r.left = clipRect->left;
r.setWidth(xDiff <= width ? width - xDiff : 0);
}
if (yDiff > 0) {
int height = r.height();
r.top = clipRect->top;
r.setHeight(yDiff <= height ? height - yDiff : 0);
}
xDiff = r.right - clipRect->right;
yDiff = r.right - clipRect->right;
if (xDiff > 0)
r.setWidth(xDiff <= r.width() ? r.width() - xDiff : 0);
if (yDiff > 0)
r.setHeight(yDiff <= r.height() ? r.height() - yDiff : 0);
}
// clip = (0x20, 0x14, width: 0x140, height: 0C8h
_activePage = page;
_field18 = 0;
_clipRect = r;
_setupFn = setupFn;
_addFn = addFn;
_restoreFn = restoreFn;
if (setupFn)
(_state._vm->_graphicsManager.*setupFn)(this);
}
void ViewPortResource::setupViewPort() {
setupViewPort(_state._vm->_graphicsManager._backgroundPage, NULL,
&GraphicsManager::setupMCGASaveRect, &GraphicsManager::addRectOptSaveRect,
&GraphicsManager::restoreMCGASaveRect);
}
int ViewPortResource::drawText(const Common::String &msg) {
GraphicsManager &gfxManager = _state._vm->_graphicsManager;
assert(gfxManager._fontPtr);
assert(gfxManager._fontPtr->_curFont);
FontInfoResource &fontInfo = *gfxManager._fontPtr;
FontResource &fontData = *fontInfo._curFont;
int xShadows[9] = { 0, 1, 1, 1, 0, -1, -1, -1, 0 };
int yShadows[9] = { 0, 1, 0, -1, -1, -1, 0, 1, 1 };
Common::Rect *clipPtr = gfxManager._clipPtr;
if (!(fontInfo._picFlags & DISPFLAG_1))
gfxManager._clipPtr = NULL;
int minChar = fontData._minChar;
int padding = fontData._padding;
int fontHeight = fontData._fontHeight;
int totalChars = fontData._maxChar - fontData._minChar;
int msgWidth = 0;
int xp = 0, yp = 0;
Common::Point pos = Common::Point(fontInfo._pos.x, fontInfo._pos.y + fontData._topPadding);
_fontChar._flags = fontInfo._picFlags | DISPFLAG_2;
_fontChar._select = fontInfo._picSelect;
_fontChar._bounds.setHeight(fontHeight);
if (gfxManager._drawTextPermFlag || (fontInfo._fontFlags & DISPFLAG_1) || fontInfo._justify ||
(gfxManager._saveBack && fontInfo._fontSaveBack && (_flags & DISPFLAG_VIEWPORT))) {
msgWidth = textWidth(msg);
yp = pos.y;
xp = pos.x;
switch (fontInfo._justify) {
case 1:
xp = pos.x + (fontInfo._justifyWidth / 2) - (msgWidth / 2);
break;
case 2:
xp = pos.x + fontInfo._justifyWidth - msgWidth;
break;
default:
break;
}
if (!(fontInfo._fontFlags & 3)) {
_fontRect.left = xp;
_fontRect.top = yp;
_fontRect.setWidth(msgWidth);
_fontRect.setHeight(fontHeight);
} else {
_fontRect.left = pos.x;
_fontRect.top = pos.y;
_fontRect.setWidth(fontInfo._justifyWidth);
_fontRect.setHeight(fontInfo._justifyHeight);
}
pos.x = xp;
pos.y = yp;
if (fontInfo._fontFlags & 4) {
if (fontInfo._shadow.x <= 0) {
_fontRect.left += fontInfo._shadow.x;
_fontRect.right -= fontInfo._shadow.x * 2;
} else {
_fontRect.right += fontInfo._shadow.x;
}
if (fontInfo._shadow.y <= 0) {
_fontRect.top += fontInfo._shadow.y;
_fontRect.bottom -= fontInfo._shadow.y * 2;
} else {
_fontRect.bottom += fontInfo._shadow.y;
}
} else if (fontInfo._fontFlags & 8) {
if (fontInfo._shadow.x <= 0) {
_fontRect.left += fontInfo._shadow.x;
_fontRect.right -= fontInfo._shadow.x * 3;
} else {
_fontRect.right += fontInfo._shadow.x * 3;
_fontRect.left -= fontInfo._shadow.x;
}
if (fontInfo._shadow.y <= 0) {
_fontRect.top += fontInfo._shadow.y;
_fontRect.bottom -= fontInfo._shadow.y * 3;
} else {
_fontRect.bottom += fontInfo._shadow.y * 3;
_fontRect.top -= fontInfo._shadow.y;
}
}
}
if (gfxManager._saveBack && fontInfo._fontSaveBack && (_flags & DISPFLAG_VIEWPORT)) {
addSaveRect(_pageIndex, _fontRect);
}
if (fontInfo._fontFlags & DISPFLAG_1) {
gfxManager._drawPtr->_pos = Common::Point(_fontRect.left, _fontRect.top);
gfxManager._drawPtr->_penColor = fontInfo._backColor;
sFillBox(_fontRect.width());
}
bool saveBack = gfxManager._saveBack;
gfxManager._saveBack = false;
int count = 0;
if (fontInfo._fontFlags & DISPFLAG_4)
count = 1;
else if (fontInfo._fontFlags & DISPFLAG_8)
count = 8;
for (int i = count; i >= 0; --i) {
xp = pos.x;
yp = pos.y;
switch (xShadows[i]) {
case -1:
xp -= fontInfo._shadow.x;
break;
case 1:
xp += fontInfo._shadow.x;
break;
default:
break;
}
switch (yShadows[i]) {
case -1:
yp -= fontInfo._shadow.y;
break;
case 1:
yp += fontInfo._shadow.y;
break;
default:
break;
}
if (i != 0) {
_fontChar._pick = 0;
_fontChar._onOff = fontInfo._shadowColor;
} else if (fontData.field2 == 1 || (fontInfo._fontFlags & 0x10)) {
_fontChar._pick = 0;
_fontChar._onOff = fontInfo._foreColor;
} else {
_fontChar._pick = fontInfo._picPick;
_fontChar._onOff = fontInfo._picOnOff;
_fontChar._depth = fontData.field2;
}
// Loop to draw each character in turn
msgWidth = -padding;
const char *msgP = msg.c_str();
char ch;
while ((ch = *msgP++) != '\0') {
int charValue = (int)ch - minChar;
if (charValue < 0 || charValue >= totalChars || fontData._charWidth[charValue] == 0)
charValue = fontData._maxChar - minChar;
int charWidth = fontData._charWidth[charValue];
_fontChar._bounds.setWidth(charWidth);
uint16 offset = READ_LE_UINT16(fontData._charOffsets + charValue * 2);
_fontChar._imgData = fontData._charImages + offset * 2;
gfxManager.sDrawPic(&_fontChar, this, Common::Point(xp, yp));
_fontChar._imgData = NULL;
xp += charWidth + padding;
msgWidth += charWidth + padding;
}
}
msgWidth = MAX(msgWidth, 0);
if (fontInfo._justify == ALIGN_LEFT)
fontInfo._pos.x = xp;
gfxManager._saveBack = saveBack;
gfxManager._clipPtr = clipPtr;
return msgWidth;
}
int ViewPortResource::textWidth(const Common::String &msg) {
if (msg.size() == 0)
return 0;
const char *msgP = msg.c_str();
FontResource &fontData = *_state._vm->_graphicsManager._fontPtr->_curFont;
int minChar = fontData._minChar;
int maxChar = fontData._maxChar;
int padding = fontData._padding;
int totalWidth = -padding;
char ch;
// Loop through the characters
while ((ch = *msgP++) != '\0') {
int charValue = (int)ch;
if (charValue < minChar || charValue > maxChar)
charValue = maxChar;
if (!fontData._charWidth[charValue - minChar])
charValue = maxChar;
totalWidth += fontData._charWidth[charValue - minChar] + padding;
}
if (totalWidth < 0)
totalWidth = 0;
return totalWidth;
}
void ViewPortResource::addSaveRect(int pageIndex, const Common::Rect &r) {
// TODO
}
void ViewPortResource::sFillBox(int width) {
// TODO
}
void ViewPortResource::fillPic(byte onOff) {
_state._vm->_graphicsManager.fillPic(this, onOff);
}
/*------------------------------------------------------------------------*/
ViewPortListResource::ViewPortListResource(BoltFilesState &state, const byte *src) {
uint count = READ_LE_UINT16(src);
_palIndex = READ_LE_UINT16(src + 2);
// Load palette map
byte *palData = state._curLibPtr->memberAddr(READ_LE_UINT32(src + 4));
for (uint i = 0; i < 256; ++i, palData += 16)
_palette.push_back(ViewPortPalEntry(palData));
// Load view port pointer list
uint32 *idP = (uint32 *)&src[8];
for (uint i = 0; i < count; ++i, ++idP) {
uint32 id = READ_LE_UINT32(idP);
BoltEntry &entry = state._curLibPtr->getBoltEntryFromLong(id);
assert(entry._viewPortResource);
_entries.push_back(entry._viewPortResource);
}
}
/*------------------------------------------------------------------------*/
ViewPortPalEntry::ViewPortPalEntry(const byte *src) {
uint16 *v = (uint16 *)src;
_rEntry = READ_LE_UINT16(v++);
_gEntry = READ_LE_UINT16(v++);
_bEntry = READ_LE_UINT16(v++);
_rChange = READ_LE_UINT16(v++);
_gChange = READ_LE_UINT16(v++);
_bChange = READ_LE_UINT16(v++);
_palIndex = READ_LE_UINT16(v++);
}
/*------------------------------------------------------------------------*/
FontResource::FontResource(BoltFilesState &state, byte *src) {
_minChar = src[0];
_maxChar = src[1];
field2 = src[2];
_padding = src[3];
_fontHeight = src[5];
_topPadding = (int8)src[6];
int totalChars = _maxChar - _minChar + 1;
_charWidth = new int[totalChars];
for (int i = 0; i < totalChars; ++i)
_charWidth[i] = READ_LE_UINT16(src + 8 + 2 * i);
_charOffsets = src + 8 + totalChars * 2;
_charImages = _charOffsets + totalChars * 2;
}
FontResource::~FontResource() {
delete[] _charWidth;
}
/*------------------------------------------------------------------------*/
FontInfoResource::FontInfoResource(BoltFilesState &state, const byte *src) {
_curFont = NULL;
_picFlags = src[4];
_picSelect = src[5];
_picPick = src[6];
_picOnOff = src[7];
_fontFlags = src[8];
_justify = (FontJustify)src[9];
_fontSaveBack = READ_LE_UINT16(src + 10);
_pos.x = (int16)READ_LE_UINT16(src + 12);
_pos.y = (int16)READ_LE_UINT16(src + 14);
_justifyWidth = READ_LE_UINT16(src + 16);
_justifyHeight = READ_LE_UINT16(src + 18);
_shadow.x = READ_LE_UINT16(src + 20);
_shadow.y = READ_LE_UINT16(src + 22);
_foreColor = READ_LE_UINT16(src + 24);
_backColor = READ_LE_UINT16(src + 26);
_shadowColor = READ_LE_UINT16(src + 28);
}
FontInfoResource::FontInfoResource() {
_curFont = NULL;
_picFlags = 3;
_picSelect = 0xff;
_picPick = 0xff;
_picOnOff = 0;
_fontFlags = 0;
_justify = ALIGN_LEFT;
_fontSaveBack = 0;
_justifyWidth = 1;
_justifyHeight = 1;
_shadow = Common::Point(1, 1);
_foreColor = 1;
_backColor = 0;
_shadowColor = 0;
}
FontInfoResource::FontInfoResource(byte picFlags, byte picSelect, byte picPick, byte picOnOff,
byte fontFlags, FontJustify justify, int fontSaveBack, const Common::Point &pos,
int justifyWidth, int justifyHeight, const Common::Point &shadow, int foreColor,
int backColor, int shadowColor) {
_curFont = NULL;
_picFlags = picFlags;
_picSelect = picSelect;
_picPick = picPick;
_picOnOff = picOnOff;
_fontFlags = fontFlags;
_justify = justify;
_fontSaveBack = fontSaveBack;
_pos = pos;
_justifyWidth = justifyWidth;
_justifyHeight = justifyHeight;
_shadow = shadow;
_foreColor = foreColor;
_backColor = backColor;
_shadowColor = shadowColor;
}
/*------------------------------------------------------------------------*/
CMapResource::CMapResource(BoltFilesState &state, const byte *src): _vm(state._vm) {
_steps = src[0];
_fadeStatus = src[1];
_start = READ_LE_UINT16(src + 2);
_end = READ_LE_UINT16(src + 4);
int count = _end - _start + 1;
_entries = new byte[count * 3];
Common::copy(src + 6, src + 6 + 3 * count, _entries);
int palIndex = state._vm->_graphicsManager._viewPortListPtr->_palIndex;
if (_end > palIndex)
_end = palIndex;
if (_start > palIndex)
_start = palIndex;
}
CMapResource::~CMapResource() {
delete[] _entries;
}
void CMapResource::startFade() {
_vm->_eventsManager.startFade(this);
}
/*------------------------------------------------------------------------*/
VInitCyclResource::VInitCyclResource(BoltFilesState &state, const byte *src) {
for (int i = 0; i < 4; ++i) {
state._curLibPtr->resolveIt(READ_LE_UINT32(src + 8 + i * 4), &_ptr[i]);
}
}
/*------------------------------------------------------------------------*/
int ThreadResource::_stampFlags = 0;
int ThreadResource::_useCount[8];
byte *ThreadResource::_threadDataPtr = NULL;
ThreadResource::ThreadResource(BoltFilesState &state, const byte *src):
_vm(state._vm) {
_flags = src[8];
}
bool ThreadResource::loadAStack(int idx) {
if (_stampFlags & 1) {
unloadAStack(_controlIndex);
if (!_useCount[idx]) {
BoltEntry &boltEntry = _vm->_stampLibPtr->boltEntry(_vm->_controlPtr->_memberIds[idx]);
if (!boltEntry._data)
return false;
_vm->_controlPtr->_entries[idx] = boltEntry._data;
}
++_useCount[idx];
}
_ctlPtr = _vm->_controlPtr->_entries[idx];
return true;
}
void ThreadResource::unloadAStack(int idx) {
if ((_stampFlags & 1) && _useCount[idx]) {
if (--_useCount[idx] == 0) {
_vm->_stampLibPtr->freeBoltMember(_vm->_controlPtr->_memberIds[idx]);
}
}
}
bool ThreadResource::doState() {
_flags |= 1;
if (!getStateInfo())
return false;
warning("TODO: stm_doState");
return true;
}
bool ThreadResource::getStateInfo() {
_field9 = 0;
int id = READ_LE_UINT16(_ctlPtr);
if (id <= _threadId) {
_field9 |= 0x80;
return false;
} else {
uint32 fld = READ_LE_UINT32(_ctlPtr + 2);
fld += _threadId << 3;
_field46 = READ_LE_UINT32(_ctlPtr + fld + 4);
fld = READ_LE_UINT32(_ctlPtr + fld);
byte *baseP = _ctlPtr + fld;
_field42 = READ_LE_UINT16(baseP);
_field40 = READ_LE_UINT16(baseP + 2);
_field44 = READ_LE_UINT16(baseP + 4);
_field28E = getDataOffset();
_field28E += READ_LE_UINT32(baseP + 6) / 2;
_field4A = baseP + 10;
getButtonsText();
return true;
}
}
byte *ThreadResource::getDataOffset() {
uint32 offset = READ_LE_UINT32(_ctlPtr + 10);
_threadDataPtr = _ctlPtr + offset;
return _threadDataPtr;
}
void ThreadResource::getButtonsText() {
warning("TODO: stm_getButtonsText");
}
void ThreadResource::unloadAllStacks(VoyeurEngine *vm) {
if (_stampFlags & 1) {
for (int i = 0; i < 8; ++i) {
if (_useCount[i])
vm->_stampLibPtr->freeBoltMember(vm->_controlPtr->_memberIds[i]);
}
}
}
void ThreadResource::initUseCount() {
Common::fill(&_useCount[0], &_useCount[8], 0);
}
/*------------------------------------------------------------------------*/
PtrResource::PtrResource(BoltFilesState &state, const byte *src) {
// Load pointer list
uint32 *idP = (uint32 *)&src[0];
int size = state._curMemberPtr->_size;
for (int i = 0; i < size / 4; ++i, ++idP) {
uint32 id = READ_LE_UINT32(idP);
BoltEntry &entry = state._curLibPtr->getBoltEntryFromLong(id);
_entries.push_back(&entry);
}
}
/*------------------------------------------------------------------------*/
ControlResource::ControlResource(BoltFilesState &state, const byte *src) {
// Get pointer
uint32 ptrId = READ_LE_UINT32(&src[0x32]);
_ptr = state._curLibPtr->getBoltEntryFromLong(ptrId)._data;
for (int i = 0; i < 8; ++i)
_memberIds[i] = READ_LE_UINT16(src + i * 2);
// Load pointer list
uint32 *idP = (uint32 *)&src[0x10];
int count = READ_LE_UINT16(&src[0x36]);
for (int i = 0; i < count; ++i, ++idP) {
uint32 id = READ_LE_UINT32(idP);
state._curLibPtr->resolveIt(id, &_entries[i]);
}
}
} // End of namespace Voyeur