scummvm/engines/gob/scenery.cpp
Simon Delamarre 0099c3624c GOB: coordinates adjustment for games existing in several resolutions
This mechanism was used to accommodate games existing in several resolutions, while keeping the same script logic across versions.
Opcodes manipulating coordinates and lengths call the adjustCoords() function to double or halve their values depending on the adjustment needed.

Its usage has been confirmed only in a German version of Adibou1 for now, restrict the feature to this game.
2023-04-21 00:13:44 +02:00

1294 lines
35 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 "common/endian.h"
#include "common/stream.h"
#include "gob/gob.h"
#include "gob/scenery.h"
#include "gob/global.h"
#include "gob/draw.h"
#include "gob/game.h"
#include "gob/script.h"
#include "gob/resources.h"
#include "gob/inter.h"
#include "gob/map.h"
#include "gob/videoplayer.h"
namespace Gob {
Scenery::Scenery(GobEngine *vm) : _vm(vm) {
for (int i = 0; i < 20; i++) {
_spriteRefs[i] = 0;
_spriteResId[i] = 0;
}
for (int i = 0; i < 10; i++) {
_staticPictCount[i] = 0;
_staticResId[i] = 0;
_animPictCount[i] = 0;
_animResId[i] = 0;
}
_curStatic = 0;
_curStaticLayer = 0;
_toRedrawLeft = 0;
_toRedrawRight = 0;
_toRedrawTop = 0;
_toRedrawBottom = 0;
_animTop = 0;
_animLeft = 0;
_pCaptureCounter = nullptr;
for (int i = 0; i < 70; i++) {
_staticPictToSprite[i] = 0;
_animPictToSprite[i] = 0;
}
}
Scenery::~Scenery() {
for (int i = 0; i < 10; i++) {
freeStatic(i);
freeAnim(i);
}
}
void Scenery::init() {
for (int i = 0; i < 10; i++) {
if (_vm->getGameType() == kGameTypeFascination) {
freeAnim(i);
freeStatic(i);
}
_animPictCount[i] = 0;
_staticPictCount[i] = -1;
}
for (int i = 0; i < 20; i++) {
_spriteRefs[i] = 0;
_spriteResId[i] = -1;
}
_curStaticLayer = -1;
_curStatic = -1;
}
int16 Scenery::loadStatic(char search) {
int16 size;
byte *backsPtr;
int16 picsCount;
int16 resId;
int16 sceneryIndex;
Static *ptr;
int16 width;
int16 height;
int16 sprResId;
int16 sprIndex;
_vm->_game->_script->evalExpr(&sceneryIndex);
size = _vm->_game->_script->readInt16();
backsPtr = _vm->_game->_script->getData() + _vm->_game->_script->pos();
_vm->_game->_script->skip(size * 2);
picsCount = _vm->_game->_script->readInt16();
resId = _vm->_game->_script->readInt16();
if (search) {
int i;
for (i = 0; i < 10; i++) {
if ((_staticPictCount[i] != -1) && (_staticResId[i] == resId)) {
_vm->_game->_script->skip(8 * _staticPictCount[i]);
return i;
}
if (_staticPictCount[i] == -1 && i < sceneryIndex)
sceneryIndex = i;
}
}
_staticPictCount[sceneryIndex] = picsCount;
_staticResId[sceneryIndex] = resId;
Resource *resource = _vm->_game->_resources->getResource((uint16) resId);
if (!resource)
return 0;
ptr = &_statics[sceneryIndex];
ptr->layersCount = resource->stream()->readSint16LE();
ptr->layers = new StaticLayer[ptr->layersCount];
for (int i = 0; i < ptr->layersCount; i++) {
Common::SeekableReadStream &layerData = *resource->stream();
layerData.seek(2 + i * 2);
layerData.seek(layerData.readUint16LE());
ptr->layers[i].backResId = layerData.readSint16LE();
ptr->layers[i].planeCount = layerData.readSint16LE();
if (ptr->layers[i].planeCount > 0) {
ptr->layers[i].planes = new StaticPlane[ptr->layers[i].planeCount];
for (int j = 0; j < ptr->layers[i].planeCount; j++) {
ptr->layers[i].planes[j].pictIndex = layerData.readByte();
ptr->layers[i].planes[j].pieceIndex = layerData.readByte();
ptr->layers[i].planes[j].drawOrder = layerData.readByte();
ptr->layers[i].planes[j].destX = layerData.readSint16LE();
ptr->layers[i].planes[j].destY = layerData.readSint16LE();
ptr->layers[i].planes[j].transp = layerData.readSByte();
}
} else
ptr->layers[i].planes = nullptr;
ptr->layers[i].backResId = (int16) READ_LE_UINT16(backsPtr);
backsPtr += 2;
}
ptr->pieces = new PieceDesc*[picsCount];
ptr->piecesCount = new uint32[picsCount];
for (int i = 0; i < picsCount; i++) {
int16 pictDescId = _vm->_game->_script->readInt16();
loadPieces(pictDescId, ptr->pieces[i], ptr->piecesCount[i]);
width = _vm->_game->_script->readInt16();
height = _vm->_game->_script->readInt16();
sprResId = _vm->_game->_script->readInt16();
for (sprIndex = 0; sprIndex < 20; sprIndex++) {
if (_spriteResId[sprIndex] == sprResId)
break;
}
if (sprIndex < 20) {
_staticPictToSprite[7 * sceneryIndex + i] = sprIndex;
_spriteRefs[sprIndex]++;
} else {
for (sprIndex = 19; _vm->_draw->_spritesArray[sprIndex] != nullptr; sprIndex--) { }
_staticPictToSprite[7 * sceneryIndex + i] = sprIndex;
_spriteRefs[sprIndex] = 1;
_spriteResId[sprIndex] = sprResId;
_vm->_draw->initSpriteSurf(sprIndex, width, height, 2);
_vm->_draw->_spritesArray[sprIndex]->clear();
_vm->_draw->_destSurface = sprIndex;
_vm->_draw->_spriteLeft = sprResId;
_vm->_draw->_transparency = 0;
_vm->_draw->_destSpriteX = 0;
_vm->_draw->_destSpriteY = 0;
_vm->_draw->spriteOperation(DRAW_LOADSPRITE);
}
}
delete resource;
return sceneryIndex + 100;
}
void Scenery::freeStatic(int16 index) {
int16 spr;
if (index == -1)
_vm->_game->_script->evalExpr(&index);
if (_staticPictCount[index] == -1)
return;
for (int i = 0; i < _staticPictCount[index]; i++) {
delete[] _statics[index].pieces[i];
spr = _staticPictToSprite[index * 7 + i];
_spriteRefs[spr]--;
if (_spriteRefs[spr] == 0) {
_vm->_draw->freeSprite(spr);
_spriteResId[spr] = -1;
}
}
for (int i = 0; i < _statics[index].layersCount; i++)
delete[] _statics[index].layers[i].planes;
delete[] _statics[index].layers;
delete[] _statics[index].pieces;
delete[] _statics[index].piecesCount;
_statics[index].layersCount = 0;
_staticPictCount[index] = -1;
}
void Scenery::renderStatic(int16 scenery, int16 layer) {
Static *ptr;
StaticLayer *layerPtr;
StaticPlane *planePtr;
int16 planeCount;
int16 order;
int16 plane;
uint16 pieceIndex;
uint16 pictIndex;
int16 left;
int16 right;
int16 top;
int16 bottom;
ptr = &_statics[scenery];
if (layer >= ptr->layersCount)
return;
layerPtr = &ptr->layers[layer];
_vm->_draw->_spriteLeft = layerPtr->backResId;
if (_vm->_draw->_spriteLeft != -1) {
_vm->_draw->_destSpriteX = 0;
_vm->_draw->_destSpriteY = 0;
_vm->_draw->_destSurface = Draw::kBackSurface;
_vm->_draw->_transparency = 0;
_vm->_draw->spriteOperation(DRAW_LOADSPRITE);
}
planeCount = layerPtr->planeCount;
for (order = 0; order < 100; order++) {
for (plane = 0, planePtr = layerPtr->planes; plane < planeCount; plane++, planePtr++) {
if (planePtr->drawOrder != order)
continue;
pieceIndex = planePtr->pieceIndex;
pictIndex = planePtr->pictIndex - 1;
if (pictIndex >= _staticPictCount[scenery])
continue;
if (!ptr->pieces || !ptr->pieces[pictIndex])
continue;
if (pieceIndex >= ptr->piecesCount[pictIndex])
continue;
_vm->_draw->_destSpriteX = planePtr->destX;
_vm->_draw->_destSpriteY = planePtr->destY;
left = ptr->pieces[pictIndex][pieceIndex].left;
right = ptr->pieces[pictIndex][pieceIndex].right;
top = ptr->pieces[pictIndex][pieceIndex].top;
bottom = ptr->pieces[pictIndex][pieceIndex].bottom;
_vm->_draw->_sourceSurface =
_staticPictToSprite[scenery * 7 + pictIndex];
_vm->_draw->_destSurface = Draw::kBackSurface;
_vm->_draw->_spriteLeft = left;
_vm->_draw->_spriteTop = top;
_vm->_draw->_spriteRight = right - left + 1;
_vm->_draw->_spriteBottom = bottom - top + 1;
_vm->_draw->_transparency = planePtr->transp ? 3 : 0;
_vm->_draw->spriteOperation(DRAW_BLITSURF);
}
}
}
void Scenery::updateStatic(int16 orderFrom, byte index, byte layer) {
StaticLayer *layerPtr;
PieceDesc **pictPtr;
StaticPlane *planePtr;
int16 planeCount;
int16 order;
int16 plane;
uint16 pieceIndex;
uint16 pictIndex;
int16 left;
int16 right;
int16 top;
int16 bottom;
if ((index >= 10) || layer >= _statics[index].layersCount)
return;
layerPtr = &_statics[index].layers[layer];
pictPtr = _statics[index].pieces;
planeCount = layerPtr->planeCount;
for (order = orderFrom; order < 100; order++) {
for (planePtr = layerPtr->planes, plane = 0;
plane < planeCount; plane++, planePtr++) {
if (planePtr->drawOrder != order)
continue;
pieceIndex = planePtr->pieceIndex;
pictIndex = planePtr->pictIndex - 1;
if (pictIndex >= _staticPictCount[index])
continue;
if (!pictPtr || !pictPtr[pictIndex])
continue;
if (pieceIndex >= _statics[index].piecesCount[pictIndex])
continue;
_vm->_draw->_destSpriteX = planePtr->destX;
_vm->_draw->_destSpriteY = planePtr->destY;
left = pictPtr[pictIndex][pieceIndex].left;
right = pictPtr[pictIndex][pieceIndex].right;
top = pictPtr[pictIndex][pieceIndex].top;
bottom = pictPtr[pictIndex][pieceIndex].bottom;
if (_vm->_draw->_destSpriteX > _toRedrawRight)
continue;
if (_vm->_draw->_destSpriteY > _toRedrawBottom)
continue;
if (_vm->_draw->_destSpriteX < _toRedrawLeft) {
left += _toRedrawLeft - _vm->_draw->_destSpriteX;
_vm->_draw->_destSpriteX = _toRedrawLeft;
}
if (_vm->_draw->_destSpriteY < _toRedrawTop) {
top += _toRedrawTop - _vm->_draw->_destSpriteY;
_vm->_draw->_destSpriteY = _toRedrawTop;
}
_vm->_draw->_spriteLeft = left;
_vm->_draw->_spriteTop = top;
_vm->_draw->_spriteRight = right - left + 1;
_vm->_draw->_spriteBottom = bottom - top + 1;
if ((_vm->_draw->_spriteRight <= 0) ||
(_vm->_draw->_spriteBottom <= 0))
continue;
if ((_vm->_draw->_destSpriteX + _vm->_draw->_spriteRight - 1) >
_toRedrawRight)
_vm->_draw->_spriteRight =
_toRedrawRight - _vm->_draw->_destSpriteX + 1;
if ((_vm->_draw->_destSpriteY + _vm->_draw->_spriteBottom - 1) >
_toRedrawBottom)
_vm->_draw->_spriteBottom =
_toRedrawBottom - _vm->_draw->_destSpriteY + 1;
_vm->_draw->_sourceSurface =
_staticPictToSprite[index * 7 + pictIndex];
_vm->_draw->_destSurface = Draw::kBackSurface;
_vm->_draw->_transparency = planePtr->transp ? 3 : 0;
_vm->_draw->spriteOperation(DRAW_BLITSURF);
}
}
}
void Scenery::updateStatic(int16 orderFrom) {
if (_curStatic == -1)
return;
if (_curStatic < 10000) {
updateStatic(orderFrom, _curStatic & 0xFF, _curStaticLayer & 0xFF);
if (_curStatic & 0xFF00)
updateStatic(orderFrom, ((_curStatic >> 8) & 0xFF) - 1,
(_curStaticLayer >> 8) & 0xFF);
} else
for (int i = 0; i < (_curStatic - 10000); i++)
updateStatic(orderFrom, i, 0);
}
int16 Scenery::loadAnim(char search) {
int16 picsCount;
int16 resId;
int16 i;
int16 j;
int16 sceneryIndex;
int16 framesCount;
Animation *ptr;
int16 width;
int16 height;
int16 sprResId;
int16 sprIndex;
uint32 layerPos;
_vm->_game->_script->evalExpr(&sceneryIndex);
picsCount = _vm->_game->_script->readInt16();
resId = _vm->_game->_script->readInt16();
if (search) {
for (i = 0; i < 10; i++) {
if ((_animPictCount[i] != 0) && (_animResId[i] == resId)) {
_vm->_game->_script->skip(8 * _animPictCount[i]);
return i;
}
if ((_animPictCount[i] == 0) && (i < sceneryIndex))
sceneryIndex = i;
}
}
_animPictCount[sceneryIndex] = picsCount;
_animResId[sceneryIndex] = resId;
Resource *resource = _vm->_game->_resources->getResource((uint16) resId);
if (!resource)
return 0;
ptr = &_animations[sceneryIndex];
ptr->layersCount = resource->stream()->readSint16LE();
ptr->layers = new AnimLayer[ptr->layersCount];
for (i = 0; i < ptr->layersCount; i++) {
Common::SeekableReadStream &layerData = *resource->stream();
layerData.seek(2 + i * 2);
layerData.seek(layerData.readUint16LE());
ptr->layers[i].unknown0 = layerData.readSint16LE();
ptr->layers[i].posX = layerData.readSint16LE();
ptr->layers[i].posY = layerData.readSint16LE();
ptr->layers[i].animDeltaX = layerData.readSint16LE();
ptr->layers[i].animDeltaY = layerData.readSint16LE();
ptr->layers[i].transp = layerData.readSByte();
ptr->layers[i].framesCount = layerData.readSint16LE();
// Going through the AnimFramePiece, finding the end for each
layerPos = layerData.pos();
framesCount = 0;
for (j = 0; j < ptr->layers[i].framesCount; j++) {
layerData.skip(4); // pictIndex, pieceIndex, destX, destY
while (layerData.readByte() == 1) {
framesCount++;
layerData.skip(4); // pictIndex, pieceIndex, destX, destY
}
framesCount++;
}
layerData.seek(layerPos);
ptr->layers[i].frames = new AnimFramePiece[framesCount];
for (j = 0; j < framesCount; j++) {
ptr->layers[i].frames[j].pictIndex = layerData.readByte();
ptr->layers[i].frames[j].pieceIndex = layerData.readByte();
ptr->layers[i].frames[j].destX = layerData.readSByte();
ptr->layers[i].frames[j].destY = layerData.readSByte();
ptr->layers[i].frames[j].notFinal = layerData.readSByte();
}
}
ptr->pieces = new PieceDesc*[picsCount];
ptr->piecesCount = new uint32[picsCount];
for (i = 0; i < picsCount; i++) {
int16 pictDescId = _vm->_game->_script->readInt16();
loadPieces(pictDescId, ptr->pieces[i], ptr->piecesCount[i]);
width = _vm->_game->_script->readInt16();
height = _vm->_game->_script->readInt16();
sprResId = _vm->_game->_script->readInt16();
for (sprIndex = 0; sprIndex < 20; sprIndex++)
if (_spriteResId[sprIndex] == sprResId)
break;
if (sprIndex < 20) {
_animPictToSprite[7 * sceneryIndex + i] = sprIndex;
_spriteRefs[sprIndex]++;
} else {
for (sprIndex = 19; _vm->_draw->_spritesArray[sprIndex];
sprIndex--)
;
_animPictToSprite[7 * sceneryIndex + i] = sprIndex;
_spriteRefs[sprIndex] = 1;
_spriteResId[sprIndex] = sprResId;
_vm->_draw->adjustCoords(0, &width, &height);
_vm->_draw->initSpriteSurf(sprIndex, width, height, 2);
_vm->_draw->_spritesArray[sprIndex]->clear();
_vm->_draw->_destSurface = sprIndex;
_vm->_draw->_spriteLeft = sprResId;
_vm->_draw->_transparency = 0;
_vm->_draw->_destSpriteX = 0;
_vm->_draw->_destSpriteY = 0;
_vm->_draw->spriteOperation(DRAW_LOADSPRITE);
}
}
delete resource;
return sceneryIndex + 100;
}
void Scenery::freeAnim(int16 index) {
int16 spr;
if (index == -1)
_vm->_game->_script->evalExpr(&index);
if (_animPictCount[index] == 0)
return;
for (int i = 0; i < _animPictCount[index]; i++) {
delete[] _animations[index].pieces[i];
spr = _animPictToSprite[index * 7 + i];
_spriteRefs[spr]--;
if (_spriteRefs[spr] == 0) {
_vm->_draw->freeSprite(spr);
_spriteResId[spr] = -1;
}
}
for (int i = 0; i < _animations[index].layersCount; i++)
delete[] _animations[index].layers[i].frames;
delete[] _animations[index].layers;
delete[] _animations[index].pieces;
delete[] _animations[index].piecesCount;
_animPictCount[index] = 0;
}
void clipInRect(int16 left, int16 top, int16 width, int16 height, int16 *deltaX, int16 *deltaY,
int16 *leftPtr, int16 *topPtr, int16 *rightPtr, int16 *bottomPtr,
int16 layer) {
int16 posX = (int16)*leftPtr;
if (posX < left) {
if (!(layer & 0x80)) {
*deltaX += left - posX;
}
*leftPtr = left;
}
int16 right = left + width;
if (*rightPtr >= right) {
if (layer & 0x80) {
*deltaX += *rightPtr - (right - 1);
}
*rightPtr = right - 1;
}
int16 posY = (int16)*topPtr;
if (posY < top) {
*deltaY += top - posY;
*topPtr = top;
}
int16 bottom = top + height;
if (*bottomPtr >= bottom) {
*bottomPtr = bottom - 1;
}
}
void Scenery::updateAnimObjectVideo(int16 layer, int16 frame, int16 animation, int16 flags,
int16 drawDeltaX, int16 drawDeltaY, char doDraw) {
int16 left;
int16 right;
int16 top;
int16 bottom;
int16 destX;
int16 destY;
if (flags & 1) { // Do capture
updateAnim(layer, frame, animation, 0, drawDeltaX, drawDeltaY, 0);
if (_toRedrawLeft == -12345)
return;
_vm->_game->capturePush(_toRedrawLeft, _toRedrawTop,
_toRedrawRight - _toRedrawLeft + 1,
_toRedrawBottom - _toRedrawTop + 1);
*_pCaptureCounter = *_pCaptureCounter + 1;
}
Mult::Mult_Object &obj = _vm->_mult->_objects[-animation - 1];
if ((obj.videoSlot == 0) || !_vm->_vidPlayer->slotIsOpen(obj.videoSlot - 1)) {
if (_vm->getGameType() == kGameTypeAdibou2) {
if (!(flags & 4))
_toRedrawLeft = -12345;
int16 deltaX = 0;
int16 deltaY = 0;
int16 sprite_dest_left = 0;
int16 sprite_dest_top = 0;
int16 sprite_dest_right = 0;
int16 sprite_dest_bottom = 0;
int16 sprite_width = 0;
if (obj.animName[0] != '\0') {
if (obj.videoSlot == 0) {
VideoPlayer::Properties props;
props.x = 0;
props.y = 0;
props.startFrame = 0;
props.lastFrame = 0;
props.breakKey = 0;
props.flags = 0;
props.palStart = 0;
props.palEnd = 0;
props.sprite = 50 - obj.pAnimData->animation - 1;
_vm->_mult->openObjVideo(obj.animName, props, animation);
}
if (obj.videoSlot != 0 && _vm->_vidPlayer->getFlags(obj.videoSlot - 1) & 0x800) {
if (doDraw) {
if (frame == (int32) _vm->_vidPlayer->getFrameCount(obj.videoSlot - 1) - 1 &&
(int16) *obj.pPosX >= _animLeft &&
(int16) *obj.pPosX <= _animTop) {
if ((int16) *obj.pPosY + _vm->_vidPlayer->getHeight(obj.videoSlot - 1) <= _animBottom
&& !(flags & 0x4)) {
if (frame != obj.lastFrameIndex) {
//_vm->_vidPlayer->setXY();
VideoPlayer::Properties props;
props.x = *obj.pPosX;
props.y = *obj.pPosY;
props.startFrame = 0;
props.lastFrame = 0;
props.breakKey = 0;
props.flags = 0;
props.palStart = 0;
props.palEnd = 0;
props.sprite = 50 - obj.pAnimData->animation - 1;
_vm->_mult->openObjVideo(obj.animName, props, animation);
sprite_dest_left = obj.spriteDestLeft;
sprite_dest_top = obj.spriteDestTop;
sprite_dest_right = obj.spriteDestRight;
sprite_dest_bottom = obj.spriteDestBottom;
if (flags & 0x2) {
clipInRect(_vm->_mult->_animLeft,
_vm->_mult->_animTop,
_vm->_mult->_animWidth,
_vm->_mult->_animHeight,
&deltaX,
&deltaY,
&sprite_dest_left,
&sprite_dest_top,
&sprite_dest_right,
&sprite_dest_bottom,
layer);
}
} else if (flags & 4) {
clipInRect(_toRedrawLeft,
_toRedrawTop,
_toRedrawRight - _toRedrawLeft + 1,
_toRedrawBottom - _toRedrawTop + 1,
&deltaX,
&deltaY,
&sprite_dest_left,
&sprite_dest_top,
&sprite_dest_right,
&sprite_dest_bottom,
layer);
} else {
_toRedrawRight = sprite_dest_right;
_toRedrawBottom = sprite_dest_bottom;
_toRedrawLeft = sprite_dest_left;
_toRedrawTop = sprite_dest_top;
}
if (!(flags & 4)) {
_toRedrawRight = sprite_dest_right;
_toRedrawBottom = sprite_dest_bottom;
_toRedrawLeft = sprite_dest_left;
_toRedrawTop = sprite_dest_top;
}
}
}
} else {
if (frame != (int16) _vm->_vidPlayer->getFrameCount(obj.videoSlot - 1) - 1) {
//getFrameCoords
}
sprite_dest_left = obj.spriteDestLeft;
sprite_dest_top = obj.spriteDestTop;
sprite_dest_right = obj.spriteDestRight;
sprite_dest_bottom = obj.spriteDestBottom;
}
}
// if (obj.field4F == 0)
if (frame != obj.lastFrameIndex) {
if ((obj.videoSlot != 0 && _vm->_vidPlayer->getFlags(obj.videoSlot - 1) & 0x800) ||
doDraw) {
if (obj.videoSlot != 0 && _vm->_vidPlayer->getFlags(obj.videoSlot - 1) & 0x800)
warning("updateAnim Adibou2 stub obj.field_38 & 0x800 != 0");
VideoPlayer::Properties props;
props.x = 0;
props.y = 0;
props.startFrame = 0;
props.lastFrame = 0;
props.breakKey = 0;
props.flags = 0;
props.palStart = 0;
props.palEnd = 0;
props.sprite = 50 - obj.pAnimData->animation - 1;
if (obj.videoSlot > 0)
_vm->_vidPlayer->closeVideo(obj.videoSlot - 1);
int slot = _vm->_vidPlayer->openVideo(false, obj.animName, props);
obj.videoSlot = slot + 1;
}
}
if (obj.videoSlot != 0 && _vm->_vidPlayer->getFlags(obj.videoSlot - 1) & 0x8000) {
deltaX = obj.spriteDestLeft;
deltaY = obj.spriteDestTop;
sprite_dest_left = *obj.pPosX + deltaX;
sprite_dest_top = *obj.pPosY + deltaY;
sprite_width = obj.spriteDestRight- obj.spriteDestLeft + 1;
if (layer & 0x80) {
sprite_dest_left = *obj.pPosX + _vm->_vidPlayer->getWidth(obj.videoSlot - 1) - deltaX - sprite_width;
}
}
else {
sprite_dest_left = *obj.pPosX ;
sprite_dest_top = *obj.pPosY;
sprite_dest_right = sprite_dest_left + _vm->_vidPlayer->getWidth(obj.videoSlot - 1) - 1;
sprite_dest_bottom = sprite_dest_top + _vm->_vidPlayer->getHeight(obj.videoSlot - 1) - 1;
}
} else {
auto &sprite = _vm->_draw->_spritesArray[50 - animation - 1];
if (sprite == nullptr)
return;
sprite_dest_left = *obj.pPosX;
sprite_dest_top = *obj.pPosY;
sprite_dest_right = sprite_dest_left + sprite->getWidth() - 1;
sprite_dest_bottom = sprite_dest_top + sprite->getHeight() - 1;
if (flags & 2) {
clipInRect(_vm->_mult->_animLeft,
_vm->_mult->_animTop,
_vm->_mult->_animWidth,
_vm->_mult->_animHeight,
&deltaX,
&deltaY,
&sprite_dest_left,
&sprite_dest_top,
&sprite_dest_right,
&sprite_dest_bottom,
layer);
} else if (flags & 4) {
clipInRect(_toRedrawLeft,
_toRedrawTop,
_toRedrawRight - _toRedrawLeft + 1,
_toRedrawBottom - _toRedrawTop + 1,
&deltaX,
&deltaY,
&sprite_dest_left,
&sprite_dest_top,
&sprite_dest_right,
&sprite_dest_bottom,
layer);
} else {
_toRedrawRight = sprite_dest_right;
_toRedrawBottom = sprite_dest_bottom;
_toRedrawLeft = sprite_dest_left;
_toRedrawTop = sprite_dest_top;
}
if (doDraw) {
if (sprite_dest_left > sprite_dest_right)
return;
if (sprite_dest_top > sprite_dest_bottom)
return;
_vm->_draw->_sourceSurface = 50 - animation - 1;
_vm->_draw->_destSurface = Draw::kBackSurface;
clipInRect(_vm->_mult->_animLeft,
_vm->_mult->_animTop,
_vm->_mult->_animWidth,
_vm->_mult->_animHeight,
&deltaX,
&deltaY,
&sprite_dest_left,
&sprite_dest_top,
&sprite_dest_right,
&sprite_dest_bottom,
layer);
_vm->_draw->_spriteLeft = deltaX;
_vm->_draw->_spriteTop = deltaY;
_vm->_draw->_spriteRight = sprite_dest_right - sprite_dest_left + 1;
_vm->_draw->_spriteBottom = sprite_dest_bottom - sprite_dest_top + 1;
_vm->_draw->_destSpriteX = sprite_dest_left,
_vm->_draw->_destSpriteY = sprite_dest_top,
_vm->_draw->_transparency = layer;
_vm->_draw->spriteOperation(DRAW_BLITSURF);
}
if (!(flags & 4)) {
_toRedrawLeft = sprite_dest_left;
_toRedrawTop = sprite_dest_top;
_toRedrawRight = sprite_dest_right;
_toRedrawBottom = sprite_dest_bottom;
}
}
} else {
// Woodruff
_toRedrawLeft = -12345;
}
return;
}
if (frame >= (int32)_vm->_vidPlayer->getFrameCount(obj.videoSlot - 1))
frame = _vm->_vidPlayer->getFrameCount(obj.videoSlot - 1) - 1;
if ((int32)_vm->_vidPlayer->getCurrentFrame(obj.videoSlot - 1) >= 255) {
// Allow for object videos with more than 255 frames, although the
// object frame counter is just a byte.
uint32 curFrame = _vm->_vidPlayer->getCurrentFrame(obj.videoSlot - 1) + 1;
uint16 frameWrap = curFrame / 256;
frame = ((frame + 1) % 256) + frameWrap * 256;
}
if (frame != (int32)_vm->_vidPlayer->getCurrentFrame(obj.videoSlot - 1)) {
// Seek to frame
VideoPlayer::Properties props;
props.forceSeek = true;
props.waitEndFrame = false;
props.lastFrame = frame;
if ((int32)_vm->_vidPlayer->getCurrentFrame(obj.videoSlot - 1) < frame)
props.startFrame = _vm->_vidPlayer->getCurrentFrame(obj.videoSlot - 1) + 1;
else
props.startFrame = frame;
_vm->_vidPlayer->play(obj.videoSlot - 1, props);
}
int32 subtitle = _vm->_vidPlayer->getSubtitleIndex(obj.videoSlot - 1);
if (subtitle != -1)
_vm->_draw->printTotText(subtitle);
destX = 0;
destY = 0;
left = *(obj.pPosX);
top = *(obj.pPosY);
right = left + _vm->_vidPlayer->getWidth(obj.videoSlot - 1) - 1;
bottom = top + _vm->_vidPlayer->getHeight(obj.videoSlot - 1) - 1;
if (flags & 2) {
if (left < _vm->_mult->_animLeft) {
destX += _vm->_mult->_animLeft - left;
left = _vm->_mult->_animLeft;
}
if ((_vm->_mult->_animLeft + _vm->_mult->_animWidth) <= right)
right = _vm->_mult->_animLeft + _vm->_mult->_animWidth - 1;
if (top < _vm->_mult->_animTop) {
destY += _vm->_mult->_animTop - top;
top = _vm->_mult->_animTop;
}
if ((_vm->_mult->_animTop + _vm->_mult->_animHeight) <= bottom)
bottom = _vm->_mult->_animTop + _vm->_mult->_animHeight - 1;
} else if (flags & 4) {
if (left < _toRedrawLeft) {
destX += _toRedrawLeft - left;
left = _toRedrawLeft;
}
if (right > _toRedrawRight)
right = _toRedrawRight;
if (top < _toRedrawTop) {
destY += _toRedrawTop - top;
top = _toRedrawTop;
}
if (bottom > _toRedrawBottom)
bottom = _toRedrawBottom;
} else {
_toRedrawTop = top;
_toRedrawLeft = left;
_toRedrawRight = right;
_toRedrawBottom = bottom;
}
if (doDraw) {
if ((left > right) || (top > bottom))
return;
if (left < _vm->_mult->_animLeft) {
destX += _vm->_mult->_animLeft - left;
left = _vm->_mult->_animLeft;
}
if ((_vm->_mult->_animLeft + _vm->_mult->_animWidth) <= right)
right = _vm->_mult->_animLeft + _vm->_mult->_animWidth - 1;
if (top < _vm->_mult->_animTop) {
destY += _vm->_mult->_animTop - top;
top = _vm->_mult->_animTop;
}
if ((_vm->_mult->_animTop + _vm->_mult->_animHeight) <= bottom)
bottom = _vm->_mult->_animTop + _vm->_mult->_animHeight - 1;
_vm->_draw->_spriteLeft = destX;
_vm->_draw->_spriteTop = destY;
_vm->_draw->_spriteRight = right - left + 1;
_vm->_draw->_spriteBottom = bottom - top + 1;
_vm->_draw->_destSpriteX = left;
_vm->_draw->_destSpriteY = top;
_vm->_draw->_transparency = layer;
if (layer & 0x80)
_vm->_draw->_spriteLeft = _vm->_vidPlayer->getWidth(obj.videoSlot - 1) -
(destX + _vm->_draw->_spriteRight);
_vm->_vidPlayer->copyFrame(obj.videoSlot - 1, *_vm->_draw->_backSurface,
_vm->_draw->_spriteLeft, _vm->_draw->_spriteTop,
_vm->_draw->_spriteRight, _vm->_draw->_spriteBottom,
_vm->_draw->_destSpriteX, _vm->_draw->_destSpriteY,
(_vm->_draw->_transparency != 0) ? 0 : -1,
(_vm->_draw->_transparency & 0x80));
_vm->_draw->invalidateRect(_vm->_draw->_destSpriteX, _vm->_draw->_destSpriteY,
_vm->_draw->_destSpriteX + _vm->_draw->_spriteRight - 1,
_vm->_draw->_destSpriteY + _vm->_draw->_spriteBottom - 1);
}
if (!(flags & 4)) {
_animLeft = _toRedrawLeft = left;
_animTop = _toRedrawTop = top;
_animRight = _toRedrawRight = right;
_animBottom = _toRedrawBottom = bottom;
}
}
// flags & 1 - do capture all area animation is occupying
// flags & 4 == 0 - calculate animation final size
// flags & 2 != 0 - don't check with "toRedraw"'s
// flags & 4 != 0 - checkk view toRedraw
void Scenery::updateAnim(int16 layer, int16 frame, int16 animation, int16 flags,
int16 drawDeltaX, int16 drawDeltaY, char doDraw) {
AnimLayer *layerPtr;
PieceDesc **pictPtr;
AnimFramePiece *framePtr;
uint16 pieceIndex;
uint16 pictIndex;
int16 left;
int16 right;
int16 top;
int16 bottom;
byte highX;
byte highY;
int16 i;
int16 transp;
int16 destX;
int16 destY;
if ((animation < 0) &&
((_vm->getGameType() == kGameTypeWoodruff) ||
(_vm->getGameType() == kGameTypeAdibou2))) {
// Object video
updateAnimObjectVideo(layer, frame, animation, flags, drawDeltaX, drawDeltaY, doDraw);
return;
}
if ((_vm->getGameType() == kGameTypeAdibou2) && animation >= 0) {
_toRedrawRight = 1000;
_toRedrawBottom = 1000;
_toRedrawLeft = 1000;
_toRedrawTop = 1000;
}
if ((animation < 0) || (animation >= 10))
return;
if ((_animPictCount[animation] == 0) || (layer < 0))
return;
if (layer >= _animations[animation].layersCount)
return;
layerPtr = &_animations[animation].layers[layer];
if (frame >= layerPtr->framesCount)
return;
if (flags & 1) { // Do capture
updateAnim(layer, frame, animation, 0, drawDeltaX, drawDeltaY, 0);
if (_toRedrawLeft == -12345)
return;
_vm->_game->capturePush(_toRedrawLeft, _toRedrawTop,
_toRedrawRight - _toRedrawLeft + 1,
_toRedrawBottom - _toRedrawTop + 1);
*_pCaptureCounter = *_pCaptureCounter + 1;
}
pictPtr = _animations[animation].pieces;
framePtr = layerPtr->frames;
for (i = 0; i < frame; i++, framePtr++)
while (framePtr->notFinal == 1)
framePtr++;
if (flags & 4) {
_toRedrawLeft = MAX(_toRedrawLeft, _vm->_mult->_animLeft);
_toRedrawTop = MAX(_toRedrawTop, _vm->_mult->_animTop);
_toRedrawRight = MIN(_toRedrawRight,
(int16)(_vm->_mult->_animLeft + _vm->_mult->_animWidth - 1));
_toRedrawBottom = MIN(_toRedrawBottom,
(int16)(_vm->_mult->_animTop + _vm->_mult->_animHeight - 1));
} else
_toRedrawLeft = -12345;
transp = layerPtr->transp ? 3 : 0;
framePtr--;
do {
framePtr++;
pieceIndex = framePtr->pieceIndex;
pictIndex = framePtr->pictIndex;
destX = framePtr->destX;
destY = framePtr->destY;
highX = pictIndex & 0xC0;
highY = pictIndex & 0x30;
highX >>= 6;
highY >>= 4;
if (destX >= 0)
destX += ((uint16)highX) << 7;
else
destX -= ((uint16)highX) << 7;
if (destY >= 0)
destY += ((uint16)highY) << 7;
else
destY -= ((uint16)highY) << 7;
if (drawDeltaX == 1000)
destX += layerPtr->posX;
else
destX += drawDeltaX;
if (drawDeltaY == 1000)
destY += layerPtr->posY;
else
destY += drawDeltaY;
pictIndex = (pictIndex & 15) - 1;
if (pictIndex >= _animPictCount[animation])
continue;
if (!pictPtr[pictIndex])
continue;
if (pieceIndex >= _animations[animation].piecesCount[pictIndex])
continue;
left = pictPtr[pictIndex][pieceIndex].left;
right = pictPtr[pictIndex][pieceIndex].right;
top = pictPtr[pictIndex][pieceIndex].top;
bottom = pictPtr[pictIndex][pieceIndex].bottom;
if (flags & 2) {
if (destX < _vm->_mult->_animLeft) {
left += _vm->_mult->_animLeft - destX;
destX = _vm->_mult->_animLeft;
}
if ((left <= right) && ((destX + right - left) >=
(_vm->_mult->_animLeft + _vm->_mult->_animWidth)))
right -= (destX + right - left) -
(_vm->_mult->_animLeft + _vm->_mult->_animWidth) + 1;
if (destY < _vm->_mult->_animTop) {
top += _vm->_mult->_animTop - destY;
destY = _vm->_mult->_animTop;
}
if ((top <= bottom) && ((destY + bottom - top) >=
(_vm->_mult->_animTop + _vm->_mult->_animHeight)))
bottom -= (destY + bottom - top) -
(_vm->_mult->_animTop + _vm->_mult->_animHeight) + 1;
} else if (flags & 4) {
if (destX < _toRedrawLeft) {
left += _toRedrawLeft - destX;
destX = _toRedrawLeft;
}
if ((left <= right) && ((destX + right - left) > _toRedrawRight))
right -= destX + right - left - _toRedrawRight;
if (destY < _toRedrawTop) {
top += _toRedrawTop - destY;
destY = _toRedrawTop;
}
if ((top <= bottom) && ((destY + bottom - top) > _toRedrawBottom))
bottom -= destY + bottom - top - _toRedrawBottom;
}
if ((left > right) || (top > bottom))
continue;
if (doDraw) {
_vm->_draw->_sourceSurface =
_animPictToSprite[animation * 7 + pictIndex];
_vm->_draw->_destSurface = Draw::kBackSurface;
_vm->_draw->_spriteLeft = left;
_vm->_draw->_spriteTop = top;
_vm->_draw->_spriteRight = right - left + 1;
_vm->_draw->_spriteBottom = bottom - top + 1;
_vm->_draw->_destSpriteX = destX;
_vm->_draw->_destSpriteY = destY;
_vm->_draw->_transparency = transp;
_vm->_draw->spriteOperation(DRAW_BLITSURF);
}
if (!(flags & 4)) {
if (_toRedrawLeft == -12345) {
_toRedrawLeft = destX;
_animLeft = destX;
_toRedrawTop = destY;
_animTop = destY;
_toRedrawRight = destX + right - left;
_animRight = destX + right - left;
_toRedrawBottom = destY + bottom - top;
_animBottom = destY + bottom - top;
} else {
_toRedrawLeft = MIN(_toRedrawLeft, destX);
_toRedrawTop = MIN(_toRedrawTop, destY);
_toRedrawRight =
MAX(_toRedrawRight, (int16)(destX + right - left));
_toRedrawBottom =
MAX(_toRedrawBottom, (int16)(destY + bottom - top));
}
}
} while (framePtr->notFinal == 1);
}
void Scenery::writeAnimLayerInfo(uint16 index, uint16 layer,
int16 varDX, int16 varDY, int16 varUnk0, int16 varFrames) {
assert(index < 10);
// WORKAROUND - Fascination Hebrew is using scripts from the CD versions, but of course
// no CD track, so the anim syncing failed, and the anims were suppressed. But they
// didn't updated the scripts. Skipping the wrong anims is a solution.
if ((_vm->getGameType() == kGameTypeFascination) && (layer >= _animations[index].layersCount)) {
WRITE_VAR_OFFSET(varDX, 0);
WRITE_VAR_OFFSET(varDY, 0);
WRITE_VAR_OFFSET(varUnk0, 0);
WRITE_VAR_OFFSET(varFrames, 0);
} else {
assert(layer < _animations[index].layersCount);
AnimLayer &animLayer = _animations[index].layers[layer];
WRITE_VAR_OFFSET(varDX, animLayer.animDeltaX);
WRITE_VAR_OFFSET(varDY, animLayer.animDeltaY);
WRITE_VAR_OFFSET(varUnk0, animLayer.unknown0);
WRITE_VAR_OFFSET(varFrames, animLayer.framesCount);
}
}
int16 Scenery::getStaticLayersCount(uint16 index) {
assert(index < 10);
return _statics[index].layersCount;
}
int16 Scenery::getAnimLayersCount(uint16 index) {
assert(index < 10);
return _animations[index].layersCount;
}
Scenery::StaticLayer *Scenery::getStaticLayer(uint16 index, uint16 layer) {
assert(index < 10);
assert(layer < _statics[index].layersCount);
return &_statics[index].layers[layer];
}
Scenery::AnimLayer *Scenery::getAnimLayer(uint16 index, uint16 layer) {
assert(index < 10);
assert(layer < _animations[index].layersCount);
return &_animations[index].layers[layer];
}
void Scenery::loadPieces(int16 pictDescId, PieceDesc *&pieceDesc, uint32 &piecesCount) {
Resource *resource = _vm->_game->_resources->getResource(pictDescId);
if (!resource) {
warning("Scenery::loadPieces(): Can't load %d", pictDescId);
return;
}
piecesCount = resource->getSize() / 8;
pieceDesc = new PieceDesc[piecesCount];
for (uint32 i = 0; i < piecesCount; i++) {
pieceDesc[i].left = resource->stream()->readSint16LE();
pieceDesc[i].right = resource->stream()->readSint16LE();
pieceDesc[i].top = resource->stream()->readSint16LE();
pieceDesc[i].bottom = resource->stream()->readSint16LE();
}
delete resource;
}
} // End of namespace Gob