mirror of
https://github.com/libretro/scummvm.git
synced 2025-01-05 17:20:30 +00:00
341 lines
8.3 KiB
C++
341 lines
8.3 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/endian.h"
|
|
#include "common/memstream.h"
|
|
#include "access/access.h"
|
|
#include "access/animation.h"
|
|
|
|
namespace Access {
|
|
|
|
AnimationResource::AnimationResource(AccessEngine *vm, Resource *res) {
|
|
int count = res->_stream->readUint16LE();
|
|
|
|
Common::Array<int> offsets;
|
|
for (int i = 0; i < count; ++i)
|
|
offsets.push_back(res->_stream->readUint32LE());
|
|
|
|
_animations.reserve(count);
|
|
for (int i = 0; i < count; ++i) {
|
|
res->_stream->seek(offsets[i]);
|
|
Animation *anim = new Animation(vm, res->_stream);
|
|
_animations.push_back(anim);
|
|
}
|
|
}
|
|
|
|
AnimationResource::~AnimationResource() {
|
|
for (int i = 0; i < (int)_animations.size(); ++i)
|
|
delete _animations[i];
|
|
}
|
|
|
|
/*------------------------------------------------------------------------*/
|
|
|
|
Animation::Animation(AccessEngine *vm, Common::SeekableReadStream *stream) : Manager(vm) {
|
|
uint32 startOfs = stream->pos();
|
|
|
|
_type = stream->readByte();
|
|
_scaling = stream->readSByte();
|
|
stream->readByte(); // unk
|
|
_frameNumber = stream->readByte();
|
|
_initialTicks = stream->readUint16LE();
|
|
stream->readUint16LE(); // unk
|
|
stream->readUint16LE(); // unk
|
|
_loopCount = stream->readSint16LE();
|
|
_countdownTicks = stream->readUint16LE();
|
|
_currentLoopCount = stream->readSint16LE();
|
|
stream->readUint16LE(); // unk
|
|
|
|
Common::Array<uint16> frameOffsets;
|
|
uint16 ofs;
|
|
while ((ofs = stream->readUint16LE()) != 0)
|
|
frameOffsets.push_back(ofs);
|
|
|
|
for (int i = 0; i < (int)frameOffsets.size(); i++) {
|
|
stream->seek(startOfs + frameOffsets[i]);
|
|
|
|
AnimationFrame *frame = new AnimationFrame(stream, startOfs);
|
|
_frames.push_back(frame);
|
|
}
|
|
}
|
|
|
|
Animation::~Animation() {
|
|
for (uint i = 0; i < _frames.size(); ++i)
|
|
delete _frames[i];
|
|
}
|
|
|
|
typedef void(Animation::*AnimationMethodPtr)();
|
|
|
|
void Animation::animate() {
|
|
static const AnimationMethodPtr METHODS[8] = {
|
|
&Animation::anim0, &Animation::anim1, &Animation::anim2, &Animation::anim3,
|
|
&Animation::anim4, &Animation::animNone, &Animation::animNone, &Animation::anim7
|
|
};
|
|
|
|
(this->*METHODS[_type])();
|
|
}
|
|
|
|
void Animation::anim0() {
|
|
if (_currentLoopCount != -1) {
|
|
if (_countdownTicks != 0) {
|
|
setFrame1(calcFrame());
|
|
} else {
|
|
_countdownTicks = _initialTicks;
|
|
++_frameNumber;
|
|
AnimationFrame *frame = calcFrame();
|
|
|
|
if (frame == nullptr) {
|
|
_frameNumber = 0;
|
|
_currentLoopCount = -1;
|
|
frame = calcFrame();
|
|
}
|
|
|
|
setFrame(frame);
|
|
}
|
|
}
|
|
}
|
|
|
|
void Animation::anim1() {
|
|
if (_currentLoopCount == -1 || _countdownTicks != 0) {
|
|
setFrame1(calcFrame());
|
|
} else {
|
|
_countdownTicks = _initialTicks;
|
|
++_frameNumber;
|
|
AnimationFrame *frame = calcFrame();
|
|
|
|
if (frame == nullptr) {
|
|
--_frameNumber;
|
|
_currentLoopCount = -1;
|
|
frame = calcFrame();
|
|
}
|
|
|
|
setFrame(frame);
|
|
}
|
|
}
|
|
|
|
void Animation::anim2() {
|
|
if (_countdownTicks != 0) {
|
|
setFrame1(calcFrame());
|
|
} else {
|
|
_countdownTicks = _initialTicks;
|
|
++_frameNumber;
|
|
AnimationFrame *frame = calcFrame();
|
|
|
|
if (frame == nullptr) {
|
|
_frameNumber = 0;
|
|
frame = calcFrame();
|
|
}
|
|
|
|
setFrame(frame);
|
|
}
|
|
}
|
|
|
|
void Animation::anim3() {
|
|
if (_currentLoopCount != -1) {
|
|
if (_countdownTicks != 0) {
|
|
setFrame1(calcFrame());
|
|
} else {
|
|
_countdownTicks = _initialTicks;
|
|
++_frameNumber;
|
|
AnimationFrame *frame = calcFrame();
|
|
|
|
if (frame == nullptr) {
|
|
--_currentLoopCount;
|
|
_frameNumber = 0;
|
|
frame = calcFrame();
|
|
}
|
|
|
|
setFrame(frame);
|
|
}
|
|
}
|
|
}
|
|
|
|
void Animation::anim4() {
|
|
if (_currentLoopCount == -1 || _countdownTicks != 0) {
|
|
setFrame1(calcFrame());
|
|
} else {
|
|
_countdownTicks = _initialTicks;
|
|
++_frameNumber;
|
|
AnimationFrame *frame = calcFrame();
|
|
|
|
if (frame == nullptr) {
|
|
if (--_currentLoopCount == -1) {
|
|
setFrame1(calcFrame());
|
|
return;
|
|
} else {
|
|
_frameNumber = 0;
|
|
frame = calcFrame();
|
|
}
|
|
}
|
|
|
|
setFrame(frame);
|
|
}
|
|
}
|
|
|
|
void Animation::animNone() {
|
|
// Empty implementation
|
|
}
|
|
|
|
void Animation::anim7() {
|
|
setFrame(calcFrame1());
|
|
}
|
|
|
|
AnimationFrame *Animation::calcFrame() {
|
|
return (_frameNumber < (int)_frames.size()) ? _frames[_frameNumber] : nullptr;
|
|
}
|
|
|
|
AnimationFrame *Animation::calcFrame1() {
|
|
return _frames[0];
|
|
}
|
|
|
|
void Animation::setFrame(AnimationFrame *frame) {
|
|
assert(frame);
|
|
_countdownTicks += frame->_frameDelay;
|
|
setFrame1(frame);
|
|
}
|
|
|
|
void Animation::setFrame1(AnimationFrame *frame) {
|
|
_vm->_animation->_base.x = frame->_baseX;
|
|
_vm->_animation->_base.y = frame->_baseY;
|
|
|
|
// Loop to add image draw requests for the parts of the frame
|
|
for (uint i = 0; i < frame->_parts.size(); ++i) {
|
|
AnimationFramePart *part = frame->_parts[i];
|
|
ImageEntry ie;
|
|
|
|
// Set the flags
|
|
ie._flags = part->_flags & ~IMGFLAG_UNSCALED;
|
|
if (_vm->_animation->_frameScale == -1)
|
|
ie._flags |= IMGFLAG_UNSCALED;
|
|
|
|
// Set the other fields
|
|
ie._spritesPtr = _vm->_objectsTable[part->_spritesIndex];
|
|
ie._frameNumber = part->_frameIndex;
|
|
ie._position = part->_position + _vm->_animation->_base;
|
|
ie._offsetY = part->_offsetY - ie._position.y;
|
|
|
|
_vm->_images.addToList(ie);
|
|
}
|
|
}
|
|
|
|
/*------------------------------------------------------------------------*/
|
|
|
|
AnimationFrame::AnimationFrame(Common::SeekableReadStream *stream, int startOffset) {
|
|
uint16 nextOffset;
|
|
|
|
stream->readByte(); // unk
|
|
_baseX = stream->readUint16LE();
|
|
_baseY = stream->readUint16LE();
|
|
_frameDelay = stream->readUint16LE();
|
|
nextOffset = stream->readUint16LE();
|
|
|
|
while (nextOffset != 0) {
|
|
stream->seek(startOffset + nextOffset);
|
|
|
|
AnimationFramePart *framePart = new AnimationFramePart(stream);
|
|
_parts.push_back(framePart);
|
|
|
|
nextOffset = stream->readUint16LE();
|
|
}
|
|
}
|
|
|
|
AnimationFrame::~AnimationFrame() {
|
|
for (int i = 0; i < (int)_parts.size(); ++i)
|
|
delete _parts[i];
|
|
}
|
|
|
|
/*------------------------------------------------------------------------*/
|
|
|
|
AnimationFramePart::AnimationFramePart(Common::SeekableReadStream *stream) {
|
|
_flags = stream->readByte();
|
|
_spritesIndex = stream->readByte();
|
|
_frameIndex = stream->readByte();
|
|
_position.x = stream->readUint16LE();
|
|
_position.y = stream->readUint16LE();
|
|
_offsetY = stream->readUint16LE();
|
|
}
|
|
|
|
/*------------------------------------------------------------------------*/
|
|
|
|
AnimationManager::AnimationManager(AccessEngine *vm) : Manager(vm) {
|
|
_animation = nullptr;
|
|
_animStart = nullptr;
|
|
_frameScale = 0;
|
|
}
|
|
|
|
AnimationManager::~AnimationManager() {
|
|
delete _animation;
|
|
}
|
|
|
|
void AnimationManager::freeAnimationData() {
|
|
delete _animation;
|
|
_animation = nullptr;
|
|
_animStart = nullptr;
|
|
}
|
|
|
|
void AnimationManager::clearTimers() {
|
|
_animationTimers.clear();
|
|
}
|
|
|
|
void AnimationManager::loadAnimations(Resource *res) {
|
|
_animationTimers.clear();
|
|
delete _animation;
|
|
_animation = new AnimationResource(_vm, res);
|
|
}
|
|
|
|
|
|
Animation *AnimationManager::setAnimation(int animId) {
|
|
Animation *anim = findAnimation(animId);
|
|
if (!anim)
|
|
return nullptr;
|
|
|
|
anim->_countdownTicks = anim->_initialTicks;
|
|
anim->_frameNumber = 0;
|
|
|
|
anim->_currentLoopCount = (anim->_type != 3 && anim->_type != 4) ? 0 : anim->_loopCount;
|
|
|
|
return anim;
|
|
}
|
|
|
|
void AnimationManager::setAnimTimer(Animation *anim) {
|
|
_animationTimers.push_back(anim);
|
|
}
|
|
|
|
Animation *AnimationManager::findAnimation(int animId) {
|
|
_animStart = (_animation == nullptr) ? nullptr : _animation->getAnimation(animId);
|
|
return _animStart;
|
|
}
|
|
|
|
void AnimationManager::animate(int animId) {
|
|
Animation *anim = findAnimation(animId);
|
|
_frameScale = anim->_scaling;
|
|
anim->animate();
|
|
}
|
|
|
|
void AnimationManager::updateTimers() {
|
|
for (uint idx = 0; idx < _animationTimers.size(); ++idx) {
|
|
if (_animationTimers[idx]->_countdownTicks > 0)
|
|
_animationTimers[idx]->_countdownTicks--;
|
|
}
|
|
}
|
|
|
|
} // End of namespace Access
|