scummvm/engines/mohawk/riven_graphics.cpp
2021-12-26 18:48:43 +01:00

1287 lines
37 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 "mohawk/resource.h"
#include "mohawk/riven.h"
#include "mohawk/riven_card.h"
#include "mohawk/riven_graphics.h"
#include "mohawk/riven_sound.h"
#include "mohawk/riven_stack.h"
#include "mohawk/riven_video.h"
#include "common/system.h"
#include "common/memstream.h"
#include "engines/util.h"
#include "graphics/fontman.h"
#include "graphics/font.h"
#include "graphics/fonts/ttf.h"
namespace Mohawk {
class TransitionEffect {
public:
TransitionEffect(OSystem *system, Graphics::Surface *mainScreen, Graphics::Surface *effectScreen,
RivenTransition type, uint duration, const Common::Rect &rect) :
_system(system),
_mainScreen(mainScreen),
_effectScreen(effectScreen),
_type(type),
_duration(duration),
_timeBased(false),
_rect(rect) {
}
virtual ~TransitionEffect() {}
bool isTimeBased() const { return _timeBased; }
virtual bool drawFrame(uint32 elapsed) = 0;
protected:
Common::Rect makeDirectionalInitalArea() const {
Common::Rect initialArea = _rect;
switch (_type) {
case kRivenTransitionWipeLeft:
case kRivenTransitionPanLeft:
initialArea.left = _rect.right;
break;
case kRivenTransitionWipeRight:
case kRivenTransitionPanRight:
initialArea.right = _rect.left;
break;
case kRivenTransitionWipeUp:
case kRivenTransitionPanUp:
initialArea.top = _rect.bottom;
break;
case kRivenTransitionWipeDown:
case kRivenTransitionPanDown:
initialArea.bottom = _rect.top;
break;
default:
error("Unhandled transition type: %d", _type);
}
return initialArea;
}
OSystem *_system;
RivenTransition _type;
uint _duration;
Common::Rect _rect;
bool _timeBased;
Graphics::Surface *_mainScreen;
Graphics::Surface *_effectScreen;
};
class TransitionEffectWipe : public TransitionEffect {
public:
TransitionEffectWipe(OSystem *system, Graphics::Surface *mainScreen, Graphics::Surface *effectScreen,
RivenTransition type, uint duration, const Common::Rect &rect) :
TransitionEffect(system, mainScreen, effectScreen, type, duration, rect) {
_timeBased = true;
_lastCopyArea = makeDirectionalInitalArea();
}
bool drawFrame(uint32 elapsed) override {
Common::Rect copyArea;
switch (_type) {
case kRivenTransitionWipeLeft:
copyArea.top = _lastCopyArea.top;
copyArea.bottom = _lastCopyArea.bottom;
copyArea.right = _lastCopyArea.left;
copyArea.left = _rect.width() - elapsed * _rect.width() / _duration;
break;
case kRivenTransitionWipeRight:
copyArea.top = _lastCopyArea.top;
copyArea.bottom = _lastCopyArea.bottom;
copyArea.left = _lastCopyArea.right;
copyArea.right = elapsed * _rect.width() / _duration;
break;
case kRivenTransitionWipeUp:
copyArea.left = _lastCopyArea.left;
copyArea.right = _lastCopyArea.right;
copyArea.bottom = _lastCopyArea.top;
copyArea.top = _rect.height() - elapsed * _rect.height() / _duration;
break;
case kRivenTransitionWipeDown:
copyArea.left = _lastCopyArea.left;
copyArea.right = _lastCopyArea.right;
copyArea.top = _lastCopyArea.bottom;
copyArea.bottom = elapsed * _rect.height() / _duration;
break;
default:
error("Unhandled transition type: %d", _type);
}
_lastCopyArea = copyArea;
if (copyArea.isEmpty()) {
// Nothing to draw
return false;
}
_effectScreen->copyRectToSurface(*_mainScreen, copyArea.left, copyArea.top, copyArea);
_system->copyRectToScreen(_effectScreen->getBasePtr(copyArea.left, copyArea.top), _effectScreen->pitch,
copyArea.left, copyArea.top, copyArea.width(), copyArea.height());
return false;
}
private:
Common::Rect _lastCopyArea;
};
class TransitionEffectPan : public TransitionEffect {
public:
TransitionEffectPan(OSystem *system, Graphics::Surface *mainScreen, Graphics::Surface *effectScreen,
RivenTransition type, uint duration, const Common::Rect &rect, int16 offset) :
TransitionEffect(system, mainScreen, effectScreen, type, duration, rect) {
_timeBased = true;
_offset = offset;
_initialArea = makeDirectionalInitalArea();
complete = false;
}
bool drawFrame(uint32 elapsed) override {
Common::Rect newArea;
switch (_type) {
case kRivenTransitionPanLeft:
newArea.top = _initialArea.top;
newArea.bottom = _initialArea.bottom;
newArea.right = _initialArea.right;
newArea.left = _rect.width() - elapsed * _rect.width() / _duration;
break;
case kRivenTransitionPanRight:
newArea.top = _initialArea.top;
newArea.bottom = _initialArea.bottom;
newArea.left = _initialArea.left;
newArea.right = elapsed * _rect.width() / _duration;
break;
case kRivenTransitionPanUp:
newArea.left = _initialArea.left;
newArea.right = _initialArea.right;
newArea.bottom = _initialArea.bottom;
newArea.top = _rect.height() - elapsed * _rect.height() / _duration;
break;
case kRivenTransitionPanDown:
newArea.left = _initialArea.left;
newArea.right = _initialArea.right;
newArea.top = _initialArea.top;
newArea.bottom = elapsed * _rect.height() / _duration;
break;
default:
error("Unhandled transition type: %d", _type);
}
if (newArea.isEmpty()) {
// Nothing to draw
return false;
}
Common::Rect oldArea;
if (newArea != _rect) {
oldArea = Common::Rect(
newArea.right != _rect.right ? _rect.left + newArea.width() : _rect.left,
newArea.bottom != _rect.bottom ? _rect.top + newArea.height() : _rect.top,
newArea.left != _rect.left ? _rect.right - newArea.width() : _rect.right,
newArea.top != _rect.top ? _rect.bottom - newArea.height() : _rect.bottom
);
}
int oldX = newArea.left != _rect.left ? _rect.left + newArea.width() : _rect.left;
int oldY = newArea.top != _rect.top ? _rect.top + newArea.height() : _rect.top;
int newX = newArea.right != _rect.right ? _rect.left + oldArea.width() : _rect.left;
int newY = newArea.bottom != _rect.bottom ? _rect.top + oldArea.height() : _rect.top;
if (_offset != -1) {
if (_type == kRivenTransitionPanDown && oldArea.height() - _offset > 0) {
newY -= _offset;
} else if (_type == kRivenTransitionPanUp && newArea.height() + _offset < _rect.height()) {
newY += _offset;
} else if (_type == kRivenTransitionPanRight && oldArea.width() - _offset > 0) {
newX -= _offset;
} else if (_type == kRivenTransitionPanLeft && newArea.width() + _offset < _rect.width()) {
newX += _offset;
} else {
newX = 0;
newY = 0;
newArea = _rect;
oldArea = Common::Rect();
}
}
if (!oldArea.isEmpty()) {
_system->copyRectToScreen(_effectScreen->getBasePtr(oldX, oldY), _effectScreen->pitch,
oldArea.left, oldArea.top, oldArea.width(), oldArea.height());
}
if (!newArea.isEmpty()) {
_system->copyRectToScreen(_mainScreen->getBasePtr(newX, newY), _mainScreen->pitch,
newArea.left, newArea.top, newArea.width(), newArea.height());
}
if (newArea == _rect) {
_effectScreen->copyRectToSurface(*_mainScreen, _rect.left, _rect.top, _rect);
return true; // The transition is complete
} else {
return false;
}
}
private:
Common::Rect _initialArea;
int16 _offset;
bool complete;
};
class TransitionEffectBlend : public TransitionEffect {
public:
TransitionEffectBlend(OSystem *system, Graphics::Surface *mainScreen, Graphics::Surface *effectScreen,
RivenTransition type, uint duration, const Common::Rect &rect) :
TransitionEffect(system, mainScreen, effectScreen, type, duration, rect) {
_timeBased = false;
}
bool drawFrame(uint32 elapsed) override {
assert(_mainScreen->format.bytesPerPixel == 2);
assert(_effectScreen->format.bytesPerPixel == 2);
if (elapsed == _duration) {
_effectScreen->copyRectToSurface(*_mainScreen, 0, 0, Common::Rect(_mainScreen->w, _mainScreen->h));
_system->copyRectToScreen(_effectScreen->getBasePtr(0, 0), _effectScreen->pitch, 0, 0, _effectScreen->w, _effectScreen->h);
return true; // The transition is complete
} else {
Graphics::Surface *screen = _system->lockScreen();
uint alpha = elapsed * 255 / _duration;
for (int y = 0; y < _mainScreen->h; y++) {
uint16 *src1 = (uint16 *) _mainScreen->getBasePtr(0, y);
uint16 *src2 = (uint16 *) _effectScreen->getBasePtr(0, y);
uint16 *dst = (uint16 *) screen->getBasePtr(0, y);
for (int x = 0; x < _mainScreen->w; x++) {
uint8 r1, g1, b1, r2, g2, b2;
_mainScreen->format.colorToRGB(*src1++, r1, g1, b1);
_effectScreen->format.colorToRGB(*src2++, r2, g2, b2);
uint r = r1 * alpha + r2 * (255 - alpha);
uint g = g1 * alpha + g2 * (255 - alpha);
uint b = b1 * alpha + b2 * (255 - alpha);
r /= 255;
g /= 255;
b /= 255;
*dst++ = (uint16) screen->format.RGBToColor(r, g, b);
}
}
_system->unlockScreen();
return false;
}
}
};
RivenGraphics::RivenGraphics(MohawkEngine_Riven* vm) :
GraphicsManager(),
_vm(vm),
_screenUpdateNesting(0),
_screenUpdateRunning(false),
_enableCardUpdateScript(true),
_scheduledTransition(kRivenTransitionNone),
_dirtyScreen(false),
_creditsImage(kRivenCreditsZeroImage),
_creditsPos(0),
_transitionMode(kRivenTransitionModeFastest),
_transitionOffset(-1),
_waterEffect(nullptr),
_fliesEffect(nullptr),
_menuFont(nullptr),
_transitionFrames(0),
_transitionDuration(0) {
_bitmapDecoder = new MohawkBitmap();
// Restrict ourselves to a single pixel format to simplify the effects implementation
_pixelFormat = Graphics::PixelFormat(2, 5, 6, 5, 0, 11, 5, 0, 0);
initGraphics(608, 436, &_pixelFormat);
// The actual game graphics only take up the first 392 rows. The inventory
// occupies the rest of the screen and we don't use the buffer to hold that.
_mainScreen = new Graphics::Surface();
_mainScreen->create(608, 392, _pixelFormat);
_effectScreen = new Graphics::Surface();
_effectScreen->create(608, 392, _pixelFormat);
if (_vm->isGameVariant(GF_25TH)) {
loadMenuFont();
}
}
RivenGraphics::~RivenGraphics() {
_effectScreen->free();
delete _effectScreen;
_mainScreen->free();
delete _mainScreen;
delete _bitmapDecoder;
clearFliesEffect();
clearWaterEffect();
delete _menuFont;
}
MohawkSurface *RivenGraphics::decodeImage(uint16 id) {
Common::SeekableReadStream *resourceStream = _vm->getResource(ID_TBMP, id);
Common::SeekableReadStream *memResourceStream = resourceStream->readStream(resourceStream->size());
delete resourceStream;
MohawkSurface *surface = _bitmapDecoder->decodeImage(memResourceStream);
surface->convertToTrueColor();
return surface;
}
void RivenGraphics::copyImageToScreen(uint16 image, uint32 left, uint32 top, uint32 right, uint32 bottom) {
Graphics::Surface *surface = findImage(image)->getSurface();
beginScreenUpdate();
// Clip the width to fit on the screen. Fixes some images.
if (left + surface->w > 608)
surface->w = 608 - left;
for (uint16 i = 0; i < surface->h; i++)
memcpy(_mainScreen->getBasePtr(left, i + top), surface->getBasePtr(0, i), surface->w * surface->format.bytesPerPixel);
_dirtyScreen = true;
applyScreenUpdate();
}
void RivenGraphics::updateScreen() {
if (_dirtyScreen) {
// Copy to screen if there's no transition. Otherwise transition.
if (_scheduledTransition == kRivenTransitionNone
|| _transitionMode == kRivenTransitionModeDisabled) {
const Common::Rect updateRect = Common::Rect(0, 0, 608, 392);
// mainScreen -> effectScreen -> systemScreen
_effectScreen->copyRectToSurface(*_mainScreen, updateRect.left, updateRect.top, updateRect);
_vm->_system->copyRectToScreen(_effectScreen->getBasePtr(updateRect.left, updateRect.top), _effectScreen->pitch, updateRect.left, updateRect.top, updateRect.width(), updateRect.height());
_scheduledTransition = kRivenTransitionNone;
} else {
runScheduledTransition();
}
_dirtyScreen = false;
}
}
void RivenGraphics::scheduleWaterEffect(uint16 sfxeID) {
clearWaterEffect();
_waterEffect = new WaterEffect(_vm, sfxeID);
}
void RivenGraphics::clearWaterEffect() {
delete _waterEffect;
_waterEffect = nullptr;
}
WaterEffect::WaterEffect(MohawkEngine_Riven *vm, uint16 sfxeID) :
_vm(vm) {
Common::SeekableReadStream *sfxeStream = _vm->getResource(ID_SFXE, sfxeID);
if (sfxeStream->readUint16BE() != 'SL')
error ("Unknown sfxe tag");
// Read in header info
uint16 frameCount = sfxeStream->readUint16BE();
uint32 offsetTablePosition = sfxeStream->readUint32BE();
_rect.left = sfxeStream->readUint16BE();
_rect.top = sfxeStream->readUint16BE();
_rect.right = sfxeStream->readUint16BE();
_rect.bottom = sfxeStream->readUint16BE();
_speed = sfxeStream->readUint16BE();
// Skip the rest of the fields...
// Read in offsets
sfxeStream->seek(offsetTablePosition);
Common::Array<uint32> frameOffsets;
frameOffsets.resize(frameCount);
for (uint16 i = 0; i < frameCount; i++)
frameOffsets[i] = sfxeStream->readUint32BE();
// Read in the scripts
sfxeStream->seek(frameOffsets[0]);
for (uint16 i = 0; i < frameCount; i++) {
uint scriptLength = (i == frameCount - 1) ? sfxeStream->size() - frameOffsets[i] : frameOffsets[i + 1] - frameOffsets[i];
_frameScripts.push_back(sfxeStream->readStream(scriptLength));
}
// Set it to the first frame
_curFrame = 0;
_lastFrameTime = 0;
delete sfxeStream;
}
void WaterEffect::update() {
if (_vm->_system->getMillis() <= _lastFrameTime + 1000 / _speed) {
return; // Nothing to do yet
}
// Make sure the script is at the starting point
Common::SeekableReadStream *script = _frameScripts[_curFrame];
script->seek(0);
Graphics::Surface *screen = _vm->_system->lockScreen();
Graphics::Surface *mainScreen = _vm->_gfx->getBackScreen();
assert(screen->format == mainScreen->format);
// Run script
uint16 curRow = 0;
for (uint16 op = script->readUint16BE(); op != 4; op = script->readUint16BE()) {
if (op == 1) { // Increment Row
curRow++;
} else if (op == 3) { // Copy Pixels
uint16 dstLeft = script->readUint16BE();
uint16 srcLeft = script->readUint16BE();
uint16 srcTop = script->readUint16BE();
uint16 rowWidth = script->readUint16BE();
byte *src = (byte *)mainScreen->getBasePtr(srcLeft, srcTop);
byte *dst = (byte *)screen->getBasePtr(dstLeft, curRow + _rect.top);
memcpy(dst, src, rowWidth * screen->format.bytesPerPixel);
} else if (op != 4) { // End of Script
error ("Unknown SFXE opcode %d", op);
}
}
_vm->_system->unlockScreen();
// Increment frame
_curFrame++;
if (_curFrame == _frameScripts.size())
_curFrame = 0;
// Set the new time
_lastFrameTime = _vm->_system->getMillis();
}
WaterEffect::~WaterEffect() {
for (uint i = 0; i < _frameScripts.size(); i++) {
delete _frameScripts[i];
}
}
void RivenGraphics::setTransitionMode(RivenTransitionMode mode) {
_transitionMode = mode;
switch (_transitionMode) {
case kRivenTransitionModeFastest:
_transitionFrames = 8;
_transitionDuration = 300;
break;
case kRivenTransitionModeNormal:
_transitionFrames = 16;
_transitionDuration = 500;
break;
case kRivenTransitionModeBest:
_transitionFrames = 32;
_transitionDuration = 700;
break;
case kRivenTransitionModeDisabled:
_transitionFrames = 0;
_transitionDuration = 0;
break;
default:
error("Unknown transition mode %d", _transitionMode);
}
}
RivenTransitionMode RivenGraphics::sanitizeTransitionMode(int mode) {
if (mode != kRivenTransitionModeDisabled
&& mode != kRivenTransitionModeFastest
&& mode != kRivenTransitionModeNormal
&& mode != kRivenTransitionModeBest) {
return kRivenTransitionModeFastest;
}
return static_cast<RivenTransitionMode>(mode);
}
void RivenGraphics::scheduleTransition(RivenTransition id, const Common::Rect &rect) {
_scheduledTransition = id;
_transitionRect = rect;
RivenHotspot *hotspot = _vm->getCard()->getCurHotspot();
if (hotspot) {
_transitionOffset = hotspot->getTransitionOffset();
} else {
_transitionOffset = -1;
}
}
void RivenGraphics::runScheduledTransition() {
if (_scheduledTransition == kRivenTransitionNone)
return;
// Note: Transitions 0-11 are actual transitions, but none are used in-game.
// There's no point in implementing them if they're not used. These extra
// transitions were found by hacking scripts.
TransitionEffect *effect = nullptr;
switch (_scheduledTransition) {
case kRivenTransitionWipeLeft:
case kRivenTransitionWipeRight:
case kRivenTransitionWipeUp:
case kRivenTransitionWipeDown: {
effect = new TransitionEffectWipe(_vm->_system, _mainScreen, _effectScreen,
_scheduledTransition, _transitionDuration, _transitionRect);
break;
}
case kRivenTransitionPanLeft:
case kRivenTransitionPanRight:
case kRivenTransitionPanUp:
case kRivenTransitionPanDown: {
effect = new TransitionEffectPan(_vm->_system, _mainScreen, _effectScreen,
_scheduledTransition, _transitionDuration, _transitionRect, _transitionOffset);
break;
}
case kRivenTransitionBlend:
case kRivenTransitionBlend2: // (tspit CARD 155)
effect = new TransitionEffectBlend(_vm->_system, _mainScreen, _effectScreen,
_scheduledTransition, _transitionFrames, _transitionRect);
break;
default:
error("Unhandled transition type: %d", _scheduledTransition);
}
if (effect->isTimeBased()) {
uint32 startTime = _vm->_system->getMillis();
uint32 timeElapsed = 0;
bool transitionComplete = false;
while (timeElapsed < _transitionDuration && !transitionComplete && !_vm->hasGameEnded()) {
transitionComplete = effect->drawFrame(timeElapsed);
_vm->doFrame();
timeElapsed = _vm->_system->getMillis() - startTime;
}
if (!transitionComplete) {
effect->drawFrame(_transitionDuration);
}
} else {
for (uint frame = 1; frame <= _transitionFrames && !_vm->hasGameEnded(); frame++) {
effect->drawFrame(frame);
_vm->doFrame();
}
}
delete effect;
_scheduledTransition = kRivenTransitionNone; // Clear scheduled transition
_transitionOffset = -1;
}
void RivenGraphics::clearMainScreen() {
_mainScreen->fillRect(Common::Rect(0, 0, 608, 392), _pixelFormat.RGBToColor(0, 0, 0));
}
void RivenGraphics::fadeToBlack() {
// The transition speed is forced to best here
setTransitionMode(kRivenTransitionModeBest);
scheduleTransition(kRivenTransitionBlend);
clearMainScreen();
runScheduledTransition();
}
void RivenGraphics::drawExtrasImageToScreen(uint16 id, const Common::Rect &rect) {
MohawkSurface *mhkSurface = _bitmapDecoder->decodeImage(_vm->getExtrasResource(ID_TBMP, id));
mhkSurface->convertToTrueColor();
Graphics::Surface *surface = mhkSurface->getSurface();
_vm->_system->copyRectToScreen(surface->getPixels(), surface->pitch, rect.left, rect.top, surface->w, surface->h);
delete mhkSurface;
}
void RivenGraphics::drawRect(const Common::Rect &rect, bool active) {
// Useful with debugging. Shows where hotspots are on the screen and whether or not they're active.
Graphics::Surface *screen = _vm->_system->lockScreen();
if (active)
screen->frameRect(rect, _pixelFormat.RGBToColor(0, 255, 0));
else
screen->frameRect(rect, _pixelFormat.RGBToColor(255, 0, 0));
_vm->_system->unlockScreen();
}
void RivenGraphics::drawImageRect(uint16 id, const Common::Rect &srcRect, const Common::Rect &dstRect) {
// Draw tBMP id from srcRect to dstRect
Graphics::Surface *surface = findImage(id)->getSurface();
assert(srcRect.width() == dstRect.width() && srcRect.height() == dstRect.height());
for (uint16 i = 0; i < srcRect.height(); i++)
memcpy(_mainScreen->getBasePtr(dstRect.left, i + dstRect.top), surface->getBasePtr(srcRect.left, i + srcRect.top), srcRect.width() * surface->format.bytesPerPixel);
_dirtyScreen = true;
}
void RivenGraphics::drawExtrasImage(uint16 id, const Common::Rect &dstRect) {
MohawkSurface *mhkSurface = _bitmapDecoder->decodeImage(_vm->getExtrasResource(ID_TBMP, id));
mhkSurface->convertToTrueColor();
Graphics::Surface *surface = mhkSurface->getSurface();
assert(dstRect.width() == surface->w);
for (uint16 i = 0; i < surface->h; i++)
memcpy(_mainScreen->getBasePtr(dstRect.left, i + dstRect.top), surface->getBasePtr(0, i), surface->pitch);
delete mhkSurface;
_dirtyScreen = true;
}
void RivenGraphics::beginCredits() {
// Clear the old cache
clearCache();
_creditsImage = kRivenCreditsZeroImage;
_creditsPos = 0;
// Now cache all the credits images
for (uint16 i = kRivenCreditsZeroImage; i <= kRivenCreditsLastImage; i++) {
MohawkSurface *surface = _bitmapDecoder->decodeImage(_vm->getExtrasResource(ID_TBMP, i));
surface->convertToTrueColor();
addImageToCache(i, surface);
}
// And clear our screen too
clearMainScreen();
_effectScreen->fillRect(Common::Rect(0, 0, 608, 392), _pixelFormat.RGBToColor(0, 0, 0));
}
void RivenGraphics::updateCredits() {
if ((_creditsImage == kRivenCreditsFirstImage || _creditsImage == kRivenCreditsSecondImage) && _creditsPos == 0)
fadeToBlack();
if (_creditsImage < kRivenCreditsSecondImage) {
// For the first two credit images, they are faded from black to the image and then out again
scheduleTransition(kRivenTransitionBlend);
Graphics::Surface *frame = findImage(_creditsImage++)->getSurface();
for (int y = 0; y < frame->h; y++)
memcpy(_mainScreen->getBasePtr(124, y), frame->getBasePtr(0, y), frame->pitch);
runScheduledTransition();
} else {
// Otherwise, we're scrolling
// This is done by 1) moving the screen up one row and
// 2) adding a new row at the bottom that is the current row of the current image or
// not and it defaults to being empty (a black row).
// Move the screen up one row
memmove(_mainScreen->getPixels(), _mainScreen->getBasePtr(0, 1), _mainScreen->pitch * (_mainScreen->h - 1));
// Copy the next row to the bottom of the screen and keep incrementing the credit images and which row we are on until we reach the last.
if (_creditsImage <= kRivenCreditsLastImage) {
Graphics::Surface *frame = findImage(_creditsImage)->getSurface();
memcpy(_mainScreen->getBasePtr(124, _mainScreen->h - 1), frame->getBasePtr(0, _creditsPos), frame->pitch);
_creditsPos++;
if (_creditsPos == (uint)_mainScreen->h) {
_creditsImage++;
_creditsPos = 0;
}
}
// Now flush the new screen
_vm->_system->copyRectToScreen(_mainScreen->getPixels(), _mainScreen->pitch, 0, 0, _mainScreen->w, _mainScreen->h);
}
}
void RivenGraphics::beginScreenUpdate() {
_screenUpdateNesting++;
}
void RivenGraphics::applyScreenUpdate(bool force) {
if (force) {
_screenUpdateNesting = 0;
} else {
_screenUpdateNesting--;
}
// The screen is only updated when the outermost screen update ends
if (_screenUpdateNesting <= 0 && !_screenUpdateRunning) {
_screenUpdateRunning = true;
if (_enableCardUpdateScript) {
_vm->getCard()->runScript(kCardUpdateScript);
}
_vm->_sound->triggerDrawSound();
updateScreen();
_screenUpdateNesting = 0;
_screenUpdateRunning = false;
}
}
void RivenGraphics::setFliesEffect(uint16 count, bool fireflies) {
delete _fliesEffect;
_fliesEffect = new FliesEffect(_vm, count, fireflies);
}
void RivenGraphics::clearFliesEffect() {
delete _fliesEffect;
_fliesEffect = nullptr;
}
Graphics::Surface *RivenGraphics::getBackScreen() {
return _mainScreen;
}
Graphics::Surface *RivenGraphics::getEffectScreen() {
return _effectScreen;
}
void RivenGraphics::updateEffects() {
if (_waterEffect && _vm->_vars["waterenabled"] != 0) {
_waterEffect->update();
}
if (_fliesEffect) {
_fliesEffect->update();
}
}
void RivenGraphics::copySystemRectToScreen(const Common::Rect &rect) {
Graphics::Surface *screen = _vm->_system->lockScreen();
_mainScreen->copyRectToSurface(*screen, rect.left, rect.top, rect);
_effectScreen->copyRectToSurface(*screen, rect.left, rect.top, rect);
_vm->_system->unlockScreen();
}
void RivenGraphics::enableCardUpdateScript(bool enable) {
_enableCardUpdateScript = enable;
}
void RivenGraphics::drawText(const Common::U32String &text, const Common::Rect &dest, uint8 greyLevel) {
_mainScreen->fillRect(dest, _pixelFormat.RGBToColor(0, 0, 0));
uint32 color = _pixelFormat.RGBToColor(greyLevel, greyLevel, greyLevel);
const Graphics::Font *font = getMenuFont();
font->drawString(_mainScreen, text, dest.left, dest.top, dest.width(), color);
_dirtyScreen = true;
}
void RivenGraphics::loadMenuFont() {
delete _menuFont;
_menuFont = nullptr;
const char *fontName;
if (_vm->getLanguage() != Common::JA_JPN) {
fontName = "FreeSans.ttf";
} else {
fontName = "mplus-2c-regular.ttf";
}
#if defined(USE_FREETYPE2)
int fontHeight;
if (_vm->getLanguage() != Common::JA_JPN) {
fontHeight = 12;
} else {
fontHeight = 11;
}
Common::SeekableReadStream *stream = SearchMan.createReadStreamForMember(fontName);
if (stream) {
_menuFont = Graphics::loadTTFFont(*stream, fontHeight);
delete stream;
}
#endif
if (!_menuFont) {
warning("Cannot load font %s", fontName);
}
}
const Graphics::Font *RivenGraphics::getMenuFont() const {
const Graphics::Font *font;
if (_menuFont) {
font = _menuFont;
} else {
font = FontMan.getFontByUsage(Graphics::FontManager::kBigGUIFont);
}
return font;
}
const FliesEffect::FliesEffectData FliesEffect::_firefliesParameters = {
true,
true,
true,
true,
3.0F,
0.7F,
40,
2.0F,
1.0F,
8447718,
30,
10
};
const FliesEffect::FliesEffectData FliesEffect::_fliesParameters = {
false,
false,
false,
true,
8.0F,
3.0F,
80,
3.0F,
1.0F,
661528,
30,
10
};
FliesEffect::FliesEffect(MohawkEngine_Riven *vm, uint16 count, bool fireflies) :
_vm(vm) {
_effectSurface = _vm->_gfx->getEffectScreen();
_backSurface = _vm->_gfx->getBackScreen();
_gameRect = Common::Rect(608, 392);
if (fireflies) {
_parameters = &_firefliesParameters;
} else {
_parameters = &_fliesParameters;
}
_updatePeriodMs = 66;
_nextUpdateTime = _vm->_system->getMillis();
initFlies(count);
}
FliesEffect::~FliesEffect() {
}
void FliesEffect::initFlies(uint16 count) {
_fly.resize(count);
for (uint16 i = 0; i < _fly.size(); i++) {
initFlyRandomPosition(i);
}
}
void FliesEffect::initFlyRandomPosition(uint index) {
int posX = _vm->_rnd->getRandomNumber(_gameRect.right - 4);
int posY = _vm->_rnd->getRandomNumber(_gameRect.bottom - 4);
if (posY < 100) {
posY = 100;
}
initFlyAtPosition(index, posX, posY, 15);
}
int FliesEffect::randomBetween(int min, int max) {
return _vm->_rnd->getRandomNumber(max - min) + min;
}
void FliesEffect::initFlyAtPosition(uint index, int posX, int posY, int posZ) {
FliesEffectEntry &fly = _fly[index];
fly.posX = posX;
fly.posXFloat = posX;
fly.posY = posY;
fly.posYFloat = posY;
fly.posZ = posZ;
fly.light = true;
fly.framesTillLightSwitch = randomBetween(_parameters->minFramesLit, _parameters->minFramesLit + _parameters->maxLightDuration);
fly.hasBlur = false;
fly.directionAngleRad = randomBetween(0, 300) / 100.0F;
fly.directionAngleRadZ = randomBetween(0, 300) / 100.0F;
fly.speed = randomBetween(0, 100) / 100.0F;
}
void FliesEffect::update() {
if (_nextUpdateTime <= _vm->_system->getMillis()) {
_nextUpdateTime = _updatePeriodMs + _vm->_system->getMillis();
updateFlies();
draw();
updateScreen();
}
}
void FliesEffect::updateFlies() {
for (uint i = 0; i < _fly.size(); i++) {
updateFlyPosition(i);
if (_fly[i].posX < 1 || _fly[i].posX > _gameRect.right - 4 || _fly[i].posY > _gameRect.bottom - 4) {
initFlyRandomPosition(i);
}
if (_parameters->lightable) {
_fly[i].framesTillLightSwitch--;
if (_fly[i].framesTillLightSwitch <= 0) {
_fly[i].light = !_fly[i].light;
_fly[i].framesTillLightSwitch = randomBetween(_parameters->minFramesLit, _parameters->minFramesLit + _parameters->maxLightDuration);
_fly[i].hasBlur = false;
}
}
}
}
void FliesEffect::updateFlyPosition(uint index) {
FliesEffectEntry &fly = _fly[index];
if (fly.directionAngleRad > 2.0F * M_PI) {
fly.directionAngleRad = fly.directionAngleRad - 2.0F * M_PI;
}
if (fly.directionAngleRad < 0.0F) {
fly.directionAngleRad = fly.directionAngleRad + 2.0F * M_PI;
}
if (fly.directionAngleRadZ > 2.0F * M_PI) {
fly.directionAngleRadZ = fly.directionAngleRadZ - 2.0F * M_PI;
}
if (fly.directionAngleRadZ < 0.0F) {
fly.directionAngleRadZ = fly.directionAngleRadZ + 2.0F * M_PI;
}
fly.posXFloat += cos(fly.directionAngleRad) * fly.speed;
fly.posYFloat += sin(fly.directionAngleRad) * fly.speed;
fly.posX = fly.posXFloat;
fly.posY = fly.posYFloat;
selectAlphaMap(
fly.posXFloat - fly.posX >= 0.5,
fly.posYFloat - fly.posY >= 0.5,
&fly.alphaMap,
&fly.width,
&fly.height);
fly.posZFloat += cos(fly.directionAngleRadZ) * (fly.speed / 2.0F);
fly.posZ = fly.posZFloat;
if (_parameters->canBlur && fly.speed > _parameters->blurSpeedTreshold) {
fly.hasBlur = true;
float blurPosXFloat = cos(fly.directionAngleRad + M_PI) * _parameters->blurDistance + fly.posXFloat;
float blurPosYFloat = sin(fly.directionAngleRad + M_PI) * _parameters->blurDistance + fly.posYFloat;
fly.blurPosX = blurPosXFloat;
fly.blurPosY = blurPosYFloat;
selectAlphaMap(
blurPosXFloat - fly.blurPosX >= 0.5,
blurPosYFloat - fly.blurPosY >= 0.5,
&fly.blurAlphaMap,
&fly.blurWidth,
&fly.blurHeight);
}
if (fly.posY >= 100) {
int maxAngularSpeed = _parameters->maxAcceleration;
if (fly.posZ > 15) {
maxAngularSpeed /= 2;
}
int angularSpeed = randomBetween(-maxAngularSpeed, maxAngularSpeed);
fly.directionAngleRad += angularSpeed / 100.0F;
} else {
// Make the flies go down if they are too high in the screen
int angularSpeed = randomBetween(0, 50);
if (fly.directionAngleRad >= M_PI / 2.0F && fly.directionAngleRad <= 3.0F * M_PI / 2.0F) {
// Going down
fly.directionAngleRad -= angularSpeed / 100.0F;
} else {
// Going up
fly.directionAngleRad += angularSpeed / 100.0F;
}
if (fly.posY < 1) {
initFlyRandomPosition(index);
}
}
if (fly.posZ >= 0) {
int distanceToScreenEdge;
if (fly.posX / 10 >= (_gameRect.right - fly.posX) / 10) {
distanceToScreenEdge = (_gameRect.right - fly.posX) / 10;
} else {
distanceToScreenEdge = fly.posX / 10;
}
if (distanceToScreenEdge > (_gameRect.bottom - fly.posY) / 10) {
distanceToScreenEdge = (_gameRect.bottom - fly.posY) / 10;
}
if (distanceToScreenEdge > 30) {
distanceToScreenEdge = 30;
}
if (fly.posZ <= distanceToScreenEdge) {
fly.directionAngleRadZ += randomBetween(-_parameters->maxAcceleration, _parameters->maxAcceleration) / 100.0F;
} else {
fly.posZ = distanceToScreenEdge;
fly.directionAngleRadZ += (float)M_PI;
}
} else {
fly.posZ = 0;
fly.directionAngleRadZ += (float)M_PI;
}
float minSpeed = _parameters->minSpeed - fly.posZ / 40.0F;
float maxSpeed = _parameters->maxSpeed - fly.posZ / 20.0F;
fly.speed += randomBetween(-_parameters->maxAcceleration, _parameters->maxAcceleration) / 100.0F;
if (fly.speed > maxSpeed) {
fly.speed -= randomBetween(0, 50) / 100.0F;
}
if (fly.speed < minSpeed) {
fly.speed += randomBetween(0, 50) / 100.0F;
}
}
void FliesEffect::selectAlphaMap(bool horGridOffset, bool vertGridoffset, const uint16 **alphaMap, uint *width, uint *height) {
static const uint16 alpha1[12] = {
8, 16, 8,
16, 32, 16,
8, 16, 8,
0, 0, 0
};
static const uint16 alpha2[12] = {
4, 12, 12, 4,
8, 24, 24, 8,
4, 12, 12, 4
};
static const uint16 alpha3[12] = {
4, 8, 4,
12, 24, 12,
12, 24, 12,
4, 8, 4
};
static const uint16 alpha4[16] = {
2, 6, 6, 2,
6, 18, 18, 6,
6, 18, 18, 6,
2, 6, 6, 2
};
static const uint16 alpha5[12] = {
4, 8, 4,
8, 32, 8,
4, 8, 4,
0, 0, 0
};
static const uint16 alpha6[12] = {
2, 6, 6, 2,
4, 24, 24, 4,
2, 6, 6, 2
};
static const uint16 alpha7[12] = {
2, 4, 2,
6, 24, 6,
6, 24, 6,
2, 4, 2
};
static const uint16 alpha8[16] = {
1, 3, 3, 1,
3, 18, 18, 3,
3, 18, 18, 3,
1, 3, 3, 1
};
struct AlphaMap {
bool horizontalGridOffset;
bool verticalGridOffset;
bool isLarge;
uint16 width;
uint16 height;
const uint16 *pixels;
};
static const AlphaMap alphaSelector[] = {
{ true, true, true, 4, 4, alpha4 },
{ true, true, false, 4, 4, alpha8 },
{ true, false, true, 4, 3, alpha2 },
{ true, false, false, 4, 3, alpha6 },
{ false, true, true, 3, 4, alpha3 },
{ false, true, false, 3, 4, alpha7 },
{ false, false, true, 3, 3, alpha1 },
{ false, false, false, 3, 3, alpha5 }
};
for (uint i = 0; i < ARRAYSIZE(alphaSelector); i++) {
if (alphaSelector[i].horizontalGridOffset == horGridOffset
&& alphaSelector[i].verticalGridOffset == vertGridoffset
&& alphaSelector[i].isLarge == _parameters->isLarge) {
*alphaMap = alphaSelector[i].pixels;
*width = alphaSelector[i].width;
*height = alphaSelector[i].height;
return;
}
}
error("Unknown flies alpha map case");
}
void FliesEffect::draw() {
const Graphics::PixelFormat format = _effectSurface->format;
for (uint i = 0; i < _fly.size(); i++) {
FliesEffectEntry &fly = _fly[i];
uint32 color = _parameters->color32;
if (!fly.light) {
color = _fliesParameters.color32;
}
bool hoveringBrightBackground = false;
for (uint y = 0; y < fly.height; y++) {
uint16 *pixel = (uint16 *) _effectSurface->getBasePtr(fly.posX, fly.posY + y);
for (uint x = 0; x < fly.width; x++) {
byte r, g, b;
format.colorToRGB(*pixel, r, g, b);
if (_parameters->unlightIfTooBright) {
if (r >= 192 || g >= 192 || b >= 192) {
hoveringBrightBackground = true;
}
}
colorBlending(color, r, g, b, fly.alphaMap[fly.width * y + x] - fly.posZ);
*pixel = format.RGBToColor(r, g, b);
++pixel;
}
}
Common::Rect drawRect = Common::Rect(fly.width, fly.height);
drawRect.translate(fly.posX, fly.posY);
addToScreenDirtyRects(drawRect);
addToEffectsDirtyRects(drawRect);
if (fly.hasBlur) {
for (uint y = 0; y < fly.blurHeight; y++) {
uint16 *pixel = (uint16 *) _effectSurface->getBasePtr(fly.blurPosX, fly.blurPosY + y);
for (uint x = 0; x < fly.blurWidth; x++) {
byte r, g, b;
format.colorToRGB(*pixel, r, g, b);
colorBlending(color, r, g, b, fly.blurAlphaMap[fly.blurWidth * y + x] - fly.posZ);
*pixel = format.RGBToColor(r, g, b);
++pixel;
}
}
Common::Rect drawRect2 = Common::Rect(fly.blurWidth, fly.blurHeight);
drawRect2.translate(fly.blurPosX, fly.blurPosY);
addToScreenDirtyRects(drawRect2);
addToEffectsDirtyRects(drawRect2);
fly.hasBlur = false;
}
if (hoveringBrightBackground) {
fly.hasBlur = false;
if (_parameters->lightable) {
fly.light = false;
fly.framesTillLightSwitch = randomBetween(_parameters->minFramesLit, _parameters->minFramesLit + _parameters->maxLightDuration);
}
if (_vm->_rnd->getRandomBit()) {
fly.directionAngleRad += (float)M_PI / 2.0F;
} else {
fly.directionAngleRad -= (float)M_PI / 2.0F;
}
}
}
}
void FliesEffect::colorBlending(uint32 flyColor, byte &r, byte &g, byte &b, int alpha) {
alpha = CLIP(alpha, 0, 32);
byte flyR = (flyColor & 0x000000FF) >> 0;
byte flyG = (flyColor & 0x0000FF00) >> 8;
byte flyB = (flyColor & 0x00FF0000) >> 16;
r = (32 * r + alpha * (flyR - r)) / 32;
g = (32 * g + alpha * (flyG - g)) / 32;
b = (32 * b + alpha * (flyB - b)) / 32;
}
void FliesEffect::updateScreen() {
for (uint i = 0; i < _screenSurfaceDirtyRects.size(); i++) {
const Common::Rect &rect = _screenSurfaceDirtyRects[i];
_vm->_system->copyRectToScreen(_effectSurface->getBasePtr(rect.left, rect.top),
_effectSurface->pitch, rect.left, rect.top,
rect.width(), rect.height()
);
}
_screenSurfaceDirtyRects.clear();
restoreEffectsSurface();
}
void FliesEffect::addToScreenDirtyRects(const Common::Rect &rect) {
for (uint i = 0; i < _screenSurfaceDirtyRects.size(); i++) {
if (rect.intersects(_screenSurfaceDirtyRects[i])) {
_screenSurfaceDirtyRects[i].extend(rect);
return;
}
}
_screenSurfaceDirtyRects.push_back(rect);
}
void FliesEffect::addToEffectsDirtyRects(const Common::Rect &rect) {
for (uint i = 0; i < _effectsSurfaceDirtyRects.size(); i++) {
if (rect.intersects(_effectsSurfaceDirtyRects[i])) {
_effectsSurfaceDirtyRects[i].extend(rect);
return;
}
}
_effectsSurfaceDirtyRects.push_back(rect);
}
void FliesEffect::restoreEffectsSurface() {
for (uint i = 0; i < _effectsSurfaceDirtyRects.size(); i++) {
const Common::Rect &rect = _effectsSurfaceDirtyRects[i];
_effectSurface->copyRectToSurface(*_backSurface, rect.left, rect.top, rect);
addToScreenDirtyRects(rect);
}
_effectsSurfaceDirtyRects.clear();
}
} // End of namespace Mohawk