mirror of
https://github.com/libretro/scummvm.git
synced 2025-01-01 23:18:44 +00:00
1043 lines
26 KiB
C++
1043 lines
26 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/debug.h"
|
|
#include "common/random.h"
|
|
|
|
#include "prince/hero.h"
|
|
#include "prince/hero_set.h"
|
|
#include "prince/animation.h"
|
|
#include "prince/resource.h"
|
|
#include "prince/prince.h"
|
|
#include "prince/graphics.h"
|
|
#include "prince/flags.h"
|
|
#include "prince/script.h"
|
|
|
|
namespace Prince {
|
|
|
|
Hero::Hero(PrinceEngine *vm, GraphicsMan *graph) : _vm(vm), _graph(graph),
|
|
_number(0), _visible(false), _state(kHeroStateStay), _middleX(0), _middleY(0),
|
|
_boreNum(1), _currHeight(0), _moveDelay(0), _shadMinus(0), _moveSetType(0), _zoomedHeroSurface(nullptr),
|
|
_lastDirection(kHeroDirDown), _destDirection(kHeroDirDown), _talkTime(0), _boredomTime(0), _phase(0),
|
|
_specAnim(nullptr), _drawX(0), _drawY(0), _drawZ(0), _zoomFactor(0), _scaleValue(0),
|
|
_shadZoomFactor(0), _shadScaleValue(0), _shadLineLen(0), _shadDrawX(0), _shadDrawY(0),
|
|
_frameXSize(0), _frameYSize(0), _scaledFrameXSize(0), _scaledFrameYSize(0), _color(0),
|
|
_coords(nullptr), _dirTab(nullptr), _currCoords(nullptr), _currDirTab(nullptr), _step(0),
|
|
_maxBoredom(200), _leftRightMainDir(0), _upDownMainDir(0), _animSetNr(0)
|
|
{
|
|
_shadowLine = (byte *)malloc(kShadowLineArraySize);
|
|
}
|
|
|
|
Hero::~Hero() {
|
|
if (_shadowLine != nullptr) {
|
|
free(_shadowLine);
|
|
_shadowLine = nullptr;
|
|
}
|
|
freeHeroAnim();
|
|
freeOldMove();
|
|
freeZoomedSurface();
|
|
}
|
|
|
|
bool Hero::loadAnimSet(uint32 animSetNr) {
|
|
_animSetNr = animSetNr;
|
|
|
|
if (animSetNr > sizeof(heroSetTable)) {
|
|
return false;
|
|
}
|
|
|
|
_shadMinus = heroSetBack[animSetNr];
|
|
|
|
for (uint32 i = 0; i < _moveSet.size(); i++) {
|
|
delete _moveSet[i];
|
|
}
|
|
|
|
const HeroSetAnimNames &animSet = *heroSetTable[animSetNr];
|
|
|
|
_moveSet.resize(kMoveSetSize);
|
|
for (uint32 i = 0; i < kMoveSetSize; i++) {
|
|
debug("Anim set item %d %s", i, animSet[i]);
|
|
Animation *anim = NULL;
|
|
if (animSet[i] != NULL) {
|
|
anim = new Animation();
|
|
Resource::loadResource(anim, animSet[i], false);
|
|
}
|
|
_moveSet[i] = anim;
|
|
}
|
|
|
|
return true;
|
|
}
|
|
|
|
Graphics::Surface *Hero::getSurface() {
|
|
Animation *heroAnim = nullptr;
|
|
if (_specAnim != nullptr) {
|
|
heroAnim = _specAnim;
|
|
} else {
|
|
heroAnim = _moveSet[_moveSetType];
|
|
}
|
|
|
|
if (heroAnim != nullptr) {
|
|
int16 phaseFrameIndex = heroAnim->getPhaseFrameIndex(_phase);
|
|
Graphics::Surface *heroFrame = heroAnim->getFrame(phaseFrameIndex);
|
|
return heroFrame;
|
|
}
|
|
return NULL;
|
|
}
|
|
|
|
uint16 Hero::getData(AttrId dataId) {
|
|
switch (dataId) {
|
|
case kHeroLastDir:
|
|
return _lastDirection;
|
|
case kHeroAnimSet:
|
|
return _moveSetType;
|
|
default:
|
|
assert(false);
|
|
return 0;
|
|
}
|
|
}
|
|
|
|
int Hero::getScaledValue(int size) {
|
|
int newSize = 0;
|
|
int16 initScaleValue = _scaleValue;
|
|
if (_scaleValue != 10000) {
|
|
for(int i = 0; i < size; i++) {
|
|
initScaleValue -= 100;
|
|
if(initScaleValue >= 0) {
|
|
newSize++;
|
|
} else {
|
|
initScaleValue += _scaleValue;
|
|
}
|
|
}
|
|
return newSize;
|
|
} else {
|
|
return size;
|
|
}
|
|
}
|
|
|
|
Graphics::Surface *Hero::zoomSprite(Graphics::Surface *heroFrame) {
|
|
Graphics::Surface *zoomedFrame = new Graphics::Surface();
|
|
zoomedFrame->create(_scaledFrameXSize, _scaledFrameYSize, Graphics::PixelFormat::createFormatCLUT8());
|
|
|
|
int sprZoomX;
|
|
int sprZoomY = _scaleValue;
|
|
uint xSource = 0;
|
|
uint ySource = 0;
|
|
uint xDest = 0;
|
|
uint yDest = 0;
|
|
|
|
for (int i = 0; i < _scaledFrameYSize; i++) {
|
|
// linear_loop:
|
|
while (1) {
|
|
sprZoomY -= 100;
|
|
if (sprZoomY >= 0 || _scaleValue == 10000) {
|
|
// all_r_y
|
|
sprZoomX = _scaleValue;
|
|
break; // to loop_lin
|
|
} else {
|
|
sprZoomY += _scaleValue;
|
|
xSource = 0;
|
|
ySource++;
|
|
}
|
|
}
|
|
// loop_lin:
|
|
for (int j = 0; j < _scaledFrameXSize; j++) {
|
|
sprZoomX -= 100;
|
|
if (sprZoomX >= 0) {
|
|
// its_all_r
|
|
memcpy(zoomedFrame->getBasePtr(xDest, yDest), heroFrame->getBasePtr(xSource, ySource), 1);
|
|
xDest++;
|
|
} else {
|
|
sprZoomX += _scaleValue;
|
|
j--;
|
|
}
|
|
xSource++;
|
|
}
|
|
xDest = 0;
|
|
yDest++;
|
|
xSource = 0;
|
|
ySource++;
|
|
}
|
|
return zoomedFrame;
|
|
}
|
|
|
|
void Hero::countDrawPosition() {
|
|
Animation *heroAnim = nullptr;
|
|
if (_specAnim != nullptr) {
|
|
heroAnim = _specAnim;
|
|
} else {
|
|
heroAnim = _moveSet[_moveSetType];
|
|
}
|
|
if (heroAnim != nullptr) {
|
|
int16 tempMiddleY;
|
|
int16 baseX = heroAnim->getBaseX();
|
|
int16 baseY = heroAnim->getBaseY();
|
|
// any chance?
|
|
if (baseX == 320) {
|
|
tempMiddleY = _middleY - (baseY - 240);
|
|
if (baseY != 240) {
|
|
error("Hero::countDrawPosition() - tempMiddleY");
|
|
}
|
|
} else {
|
|
tempMiddleY = _middleY;
|
|
}
|
|
int phaseFrameIndex = heroAnim->getPhaseFrameIndex(_phase);
|
|
Graphics::Surface *heroSurface = heroAnim->getFrame(phaseFrameIndex);
|
|
_frameXSize = heroSurface->w;
|
|
_frameYSize = heroSurface->h;
|
|
_scaledFrameXSize = getScaledValue(_frameXSize);
|
|
_scaledFrameYSize = getScaledValue(_frameYSize);
|
|
|
|
// any use of this?
|
|
if (!heroAnim->testId()) {
|
|
error("countDrawPosition - !heroAnim->testId()");
|
|
//int diffX = heroAnim->getIdXDiff();
|
|
//int diffY = heroAnim->getIdYDiff();
|
|
}
|
|
|
|
if (_zoomFactor != 0) {
|
|
//notfullSize
|
|
_drawX = _middleX - _scaledFrameXSize / 2;
|
|
_drawY = tempMiddleY + 1 - _scaledFrameYSize;
|
|
_vm->checkMasks(_drawX, _drawY - 1, _scaledFrameXSize, _scaledFrameYSize, _middleY);
|
|
} else {
|
|
//fullSize
|
|
_drawX = _middleX - _frameXSize / 2;
|
|
_drawY = tempMiddleY + 1 - _frameYSize;
|
|
_vm->checkMasks(_drawX, _drawY - 1, _frameXSize, _frameYSize, _middleY);
|
|
}
|
|
|
|
_drawZ = tempMiddleY;
|
|
}
|
|
}
|
|
|
|
void Hero::plotPoint(int x, int y) {
|
|
WRITE_UINT16(&_shadowLine[_shadLineLen * 4], x);
|
|
WRITE_UINT16(&_shadowLine[_shadLineLen * 4 + 2], y);
|
|
}
|
|
|
|
static void plot(int x, int y, int color, void *data) {
|
|
Hero *shadowLine = (Hero *)data;
|
|
shadowLine->plotPoint(x, y);
|
|
shadowLine->_shadLineLen++;
|
|
}
|
|
|
|
// FIXME - shadows are badly cutted off in left down corner of locations 7, 12
|
|
// and horizontal line with wrong colors in the middle of shadow appears in loc. 12, 23
|
|
void Hero::showHeroShadow(Graphics::Surface *screen, DrawNode *drawNode) {
|
|
Hero *heroData = (Hero *)drawNode->data;
|
|
int16 heroSurfaceWidth = drawNode->s->w;
|
|
int16 heroSurfaceHeight = drawNode->s->h;
|
|
|
|
Graphics::Surface *makeShadow = new Graphics::Surface();
|
|
makeShadow->create(heroSurfaceWidth, heroSurfaceHeight, Graphics::PixelFormat::createFormatCLUT8());
|
|
|
|
for (int y = 0; y < heroSurfaceHeight; y++) {
|
|
byte *src = (byte *)drawNode->s->getBasePtr(0, y);
|
|
byte *dst = (byte *)makeShadow->getBasePtr(0, y);
|
|
for (int x = 0; x < heroSurfaceWidth; x++, dst++, src++) {
|
|
if (*src != 0xFF) {
|
|
*dst = GraphicsMan::kShadowColor;
|
|
} else {
|
|
*dst = *src;
|
|
}
|
|
}
|
|
}
|
|
|
|
if (drawNode->posY > 1 && drawNode->posY < PrinceEngine::kMaxPicHeight) {
|
|
int shadDirection;
|
|
if (heroData->_vm->_lightY > drawNode->posY) {
|
|
shadDirection = 1;
|
|
} else {
|
|
shadDirection = 0;
|
|
}
|
|
|
|
heroData->_shadLineLen = 0;
|
|
Graphics::drawLine(heroData->_vm->_lightX, heroData->_vm->_lightY, drawNode->posX, drawNode->posY, 0, &plot, heroData);
|
|
|
|
byte *sprShadow = (byte *)heroData->_graph->_shadowTable70;
|
|
|
|
heroData->_shadDrawX = drawNode->posX - heroData->_vm->_picWindowX;
|
|
heroData->_shadDrawY = drawNode->posY - heroData->_vm->_picWindowY;
|
|
|
|
int shadPosX = heroData->_shadDrawX;
|
|
int shadPosY = heroData->_shadDrawY;
|
|
int shadBitAddr = drawNode->posY * PrinceEngine::kMaxPicWidth / 8 + drawNode->posX / 8;
|
|
int shadBitMask = 128 >> (drawNode->posX % 8);
|
|
|
|
int shadZoomY2 = heroData->_shadScaleValue;
|
|
int shadZoomY = heroData->_scaleValue;
|
|
|
|
int diffX = 0;
|
|
int diffY = 0;
|
|
|
|
int shadowHeroX = 0;
|
|
int shadowHeroY = heroSurfaceHeight - 1;
|
|
|
|
int shadLastY = 0;
|
|
|
|
byte *shadowHero = (byte *)makeShadow->getBasePtr(shadowHeroX, shadowHeroY); // first pixel from last row of shadow hero
|
|
byte *background = (byte *)screen->getBasePtr(heroData->_shadDrawX, heroData->_shadDrawY); // pixel of background where shadow sprite starts
|
|
|
|
// banked2
|
|
byte *shadowLineStart = heroData->_shadowLine + 8;
|
|
|
|
int shadWallDown = 0;
|
|
int shadWallBitAddr = 0;
|
|
int shadWallBitMask = 0;
|
|
byte *shadWallDestAddr = 0;
|
|
int shadWallPosY = 0;
|
|
int shadWallSkipX = 0;
|
|
int shadWallModulo = 0;
|
|
|
|
// linear_loop
|
|
for (int i = 0; i < heroSurfaceHeight; i++) {
|
|
int shadSkipX = 0;
|
|
int ct_loop = 0;
|
|
int sprModulo = 0;
|
|
|
|
int j;
|
|
//retry_line:
|
|
for (j = heroSurfaceHeight - i; j > 0; j--) {
|
|
shadZoomY -= 100;
|
|
if (shadZoomY < 0 && heroData->_scaleValue != 10000) {
|
|
shadZoomY += heroData->_scaleValue;
|
|
shadowHeroY--;
|
|
if (shadowHeroY < 0) {
|
|
break;
|
|
}
|
|
shadowHero = (byte *)makeShadow->getBasePtr(shadowHeroX, shadowHeroY);
|
|
} else {
|
|
break;
|
|
}
|
|
}
|
|
if (j == 0) {
|
|
break;
|
|
}
|
|
if (shadowHeroY < 0) {
|
|
break;
|
|
}
|
|
//line_y_ok
|
|
if (shadLastY != shadPosY && shadPosY >= 0 && shadPosY < 480 && shadPosX < 640) {
|
|
shadLastY = shadPosY;
|
|
if (shadPosX < 0) {
|
|
shadSkipX = -1 * shadPosX;
|
|
background += shadSkipX;
|
|
if (heroSurfaceWidth > shadSkipX) {
|
|
shadowHero += shadSkipX;
|
|
shadBitAddr += shadSkipX / 8;
|
|
if ((shadSkipX % 8) != 0) {
|
|
//loop_rotate:
|
|
for (int a = 0; a < (shadSkipX % 8); a++) {
|
|
if (shadBitMask == 1) {
|
|
shadBitMask = 128;
|
|
shadBitAddr++;
|
|
} else {
|
|
shadBitMask >>= 1;
|
|
}
|
|
}
|
|
}
|
|
} else {
|
|
//skip_line
|
|
//test it
|
|
}
|
|
} else {
|
|
//x1_ok
|
|
if (shadPosX + heroSurfaceWidth > 640) {
|
|
ct_loop = 640 - shadPosX; // test it
|
|
sprModulo = shadPosX + heroSurfaceWidth - 640;
|
|
} else {
|
|
//draw_line
|
|
ct_loop = heroSurfaceWidth;
|
|
}
|
|
}
|
|
//draw_line1
|
|
//retry_line2
|
|
int k;
|
|
for (k = j; k > 0; k--) {
|
|
shadZoomY2 -= 100;
|
|
if (shadZoomY2 < 0 && heroData->_shadScaleValue != 10000) {
|
|
shadZoomY2 += heroData->_shadScaleValue;
|
|
shadowHeroY--;
|
|
if (shadowHeroY < 0) {
|
|
break;
|
|
}
|
|
shadowHero = (byte *)makeShadow->getBasePtr(shadowHeroX, shadowHeroY);
|
|
} else {
|
|
break;
|
|
}
|
|
}
|
|
if (shadowHeroY < 0) {
|
|
break;
|
|
}
|
|
if (k == 0) {
|
|
break;
|
|
}
|
|
//line_y_ok_2:
|
|
//copy_trans
|
|
bool shadWDFlag = false;
|
|
int shadZoomX = heroData->_scaleValue;
|
|
int backgroundDiff = 0;
|
|
int shadBitMaskCopyTrans = shadBitMask;
|
|
int shadBitAddrCopyTrans = shadBitAddr;
|
|
//ct_loop:
|
|
for (int l = 0; l < ct_loop; l++) {
|
|
shadZoomX -= 100;
|
|
if (shadZoomX < 0 && heroData->_scaleValue != 10000) {
|
|
shadZoomX += heroData->_scaleValue;
|
|
} else {
|
|
if (*shadowHero == GraphicsMan::kShadowColor) {
|
|
if ((shadBitMaskCopyTrans & heroData->_vm->_shadowBitmap[shadBitAddrCopyTrans]) != 0) {
|
|
if (shadWallDown == 0) {
|
|
if ((shadBitMaskCopyTrans & heroData->_vm->_shadowBitmap[shadBitAddrCopyTrans + heroData->_vm->kShadowBitmapSize]) != 0) {
|
|
shadWDFlag = true;
|
|
//shadow
|
|
*background = *(sprShadow + *background);
|
|
}
|
|
}
|
|
} else {
|
|
//shadow
|
|
*background = *(sprShadow + *background);
|
|
}
|
|
}
|
|
//ct_next
|
|
if (shadBitMaskCopyTrans == 1) {
|
|
shadBitMaskCopyTrans = 128;
|
|
shadBitAddrCopyTrans++;
|
|
} else {
|
|
shadBitMaskCopyTrans >>= 1;
|
|
}
|
|
//okok
|
|
backgroundDiff++;
|
|
background = (byte *)screen->getBasePtr(heroData->_shadDrawX + diffX + backgroundDiff, heroData->_shadDrawY + diffY);
|
|
}
|
|
shadowHeroX++;
|
|
shadowHero = (byte *)makeShadow->getBasePtr(shadowHeroX, shadowHeroY);
|
|
}
|
|
//byebyebye
|
|
if (!shadWallDown && shadWDFlag) {
|
|
shadWallDown = shadPosX;
|
|
shadWallBitAddr = shadBitAddr;
|
|
shadWallDestAddr = (byte *)screen->getBasePtr(heroData->_shadDrawX + diffX, heroData->_shadDrawY + diffY);
|
|
shadWallBitMask = shadBitMask;
|
|
shadWallPosY = shadPosY;
|
|
shadWallSkipX = shadSkipX;
|
|
shadWallModulo = sprModulo;
|
|
}
|
|
//byebye
|
|
if (shadDirection != 0 && shadWallDown != 0) {
|
|
int shadBitMaskWallCopyTrans = shadWallBitMask;
|
|
int shadBitAddrWallCopyTrans = shadWallBitAddr;
|
|
background = shadWallDestAddr;
|
|
shadowHero = (byte *)makeShadow->getBasePtr(shadWallSkipX, shadowHeroY);
|
|
|
|
if (ct_loop > shadWallSkipX && ct_loop - shadWallSkipX > shadWallModulo) {
|
|
//WALL_copy_trans
|
|
shadWDFlag = false;
|
|
int shadZoomXWall = heroData->_scaleValue;
|
|
int backgroundDiffWall = 0;
|
|
int shadowHeroXWall = 0;
|
|
//ct_loop:
|
|
for (int m = 0; m < ct_loop; m++) {
|
|
shadZoomXWall -= 100;
|
|
if (shadZoomXWall < 0 && heroData->_scaleValue != 10000) {
|
|
shadZoomXWall += heroData->_scaleValue;
|
|
} else {
|
|
//point_ok:
|
|
if (*shadowHero == GraphicsMan::kShadowColor) {
|
|
if ((shadBitMaskWallCopyTrans & heroData->_vm->_shadowBitmap[shadBitAddrWallCopyTrans + heroData->_vm->kShadowBitmapSize]) != 0) {
|
|
*background = *(sprShadow + *background);
|
|
}
|
|
}
|
|
//ct_next
|
|
if (shadBitMaskWallCopyTrans == 1) {
|
|
shadBitMaskWallCopyTrans = 128;
|
|
shadBitAddrWallCopyTrans++;
|
|
} else {
|
|
shadBitMaskWallCopyTrans >>= 1;
|
|
}
|
|
//okok
|
|
backgroundDiffWall++;
|
|
background = shadWallDestAddr + backgroundDiffWall;
|
|
}
|
|
shadowHeroXWall++;
|
|
shadowHero = (byte *)makeShadow->getBasePtr(shadWallSkipX + shadowHeroXWall, shadowHeroY);
|
|
}
|
|
}
|
|
//krap2
|
|
shadWallDestAddr -= PrinceEngine::kNormalWidth;
|
|
shadWallBitAddr -= PrinceEngine::kMaxPicWidth / 8;
|
|
shadWallPosY--;
|
|
}
|
|
}
|
|
//skip_line
|
|
//next_line
|
|
if (*(shadowLineStart + 2) < *(shadowLineStart - 2)) {
|
|
//minus_y
|
|
shadBitAddr -= PrinceEngine::kMaxPicWidth / 8;
|
|
shadPosY--;
|
|
diffY--;
|
|
} else if (*(shadowLineStart + 2) > *(shadowLineStart - 2)) {
|
|
shadBitAddr += PrinceEngine::kMaxPicWidth / 8;
|
|
shadPosY++;
|
|
diffY++;
|
|
}
|
|
//no_change_y
|
|
if (*shadowLineStart < *(shadowLineStart - 4)) {
|
|
//minus_x
|
|
shadPosX--;
|
|
//rol
|
|
if (shadBitMask == 128) {
|
|
shadBitMask = 1;
|
|
shadBitAddr--;
|
|
} else {
|
|
shadBitMask <<= 1;
|
|
}
|
|
diffX--;
|
|
} else if (*shadowLineStart > *(shadowLineStart - 4)) {
|
|
shadPosX++;
|
|
//ror
|
|
if (shadBitMask == 1) {
|
|
shadBitMask = 128;
|
|
shadBitAddr++;
|
|
} else {
|
|
shadBitMask >>= 1;
|
|
}
|
|
diffX++;
|
|
}
|
|
//no_change_x
|
|
shadowLineStart += 4;
|
|
shadowHeroY--;
|
|
if (shadowHeroY < 0) {
|
|
break;
|
|
}
|
|
shadowHeroX = 0;
|
|
background = (byte *)screen->getBasePtr(heroData->_shadDrawX + diffX, heroData->_shadDrawY + diffY);
|
|
shadowHero = (byte *)makeShadow->getBasePtr(shadowHeroX, shadowHeroY);
|
|
}
|
|
//koniec_bajki - end_of_a_story
|
|
}
|
|
makeShadow->free();
|
|
delete makeShadow;
|
|
}
|
|
|
|
void Hero::setScale(int8 zoomBitmapValue) {
|
|
if (zoomBitmapValue == 0) {
|
|
_zoomFactor = 0;
|
|
_scaleValue = 10000;
|
|
} else {
|
|
_zoomFactor = zoomBitmapValue;
|
|
_scaleValue = 10000 / _zoomFactor;
|
|
}
|
|
}
|
|
|
|
void Hero::selectZoom() {
|
|
int8 zoomBitmapValue = *(_vm->_zoomBitmap + _middleY / 4 * _vm->kZoomBitmapWidth + _middleX / 4);
|
|
setScale(zoomBitmapValue);
|
|
}
|
|
|
|
void Hero::setShadowScale(int32 shadowScale) {
|
|
shadowScale = 100 - shadowScale;
|
|
if (shadowScale == 0) {
|
|
_shadZoomFactor = 0;
|
|
_shadScaleValue = 10000;
|
|
} else {
|
|
_shadZoomFactor = shadowScale;
|
|
_shadScaleValue = 10000 / _shadZoomFactor;
|
|
}
|
|
}
|
|
|
|
int Hero::rotateHero(int oldDirection, int newDirection) {
|
|
switch (oldDirection) {
|
|
case kHeroDirLeft:
|
|
switch (newDirection) {
|
|
case kHeroDirRight:
|
|
return kMove_MLR;
|
|
case kHeroDirUp:
|
|
return kMove_MLU;
|
|
case kHeroDirDown:
|
|
return kMove_MLD;
|
|
}
|
|
break;
|
|
case kHeroDirRight:
|
|
switch (newDirection) {
|
|
case kHeroDirLeft:
|
|
return kMove_MRL;
|
|
case kHeroDirUp:
|
|
return kMove_MRU;
|
|
case kHeroDirDown:
|
|
return kMove_MRD;
|
|
}
|
|
break;
|
|
case kHeroDirUp:
|
|
switch (newDirection) {
|
|
case kHeroDirLeft:
|
|
return kMove_MUL;
|
|
case kHeroDirRight:
|
|
return kMove_MUR;
|
|
case kHeroDirDown:
|
|
return kMove_MUD;
|
|
}
|
|
break;
|
|
case kHeroDirDown:
|
|
switch (newDirection) {
|
|
case kHeroDirLeft:
|
|
return kMove_MDL;
|
|
case kHeroDirRight:
|
|
return kMove_MDR;
|
|
case kHeroDirUp:
|
|
return kMove_MDU;
|
|
}
|
|
break;
|
|
}
|
|
error("rotateHero - wrong directions - old %d, new %d", oldDirection, newDirection);
|
|
}
|
|
|
|
void Hero::heroStanding() {
|
|
_phase = 0;
|
|
switch (_lastDirection) {
|
|
case kHeroDirLeft:
|
|
_moveSetType = kMove_SL;
|
|
break;
|
|
case kHeroDirRight:
|
|
_moveSetType = kMove_SR;
|
|
break;
|
|
case kHeroDirUp:
|
|
_moveSetType = kMove_SU;
|
|
break;
|
|
case kHeroDirDown:
|
|
_moveSetType = kMove_SD;
|
|
break;
|
|
}
|
|
}
|
|
|
|
void Hero::showHero() {
|
|
if (_visible && !_vm->_flags->getFlagValue(Flags::NOHEROATALL)) {
|
|
|
|
if (_talkTime != 0) {
|
|
_talkTime--;
|
|
}
|
|
|
|
// Scale of hero
|
|
selectZoom();
|
|
|
|
if (_state != kHeroStateStay) {
|
|
_boredomTime = 0;
|
|
}
|
|
|
|
if (_state == kHeroStateSpec) {
|
|
if (_specAnim != nullptr) {
|
|
if (_phase < _specAnim->getPhaseCount() - 1) {
|
|
_phase++;
|
|
} else {
|
|
_phase = 0;
|
|
freeHeroAnim();
|
|
if (!_talkTime) {
|
|
_state = kHeroStateStay;
|
|
} else {
|
|
_state = kHeroStateTalk;
|
|
}
|
|
}
|
|
} else {
|
|
_state = kHeroStateStay;
|
|
}
|
|
}
|
|
|
|
if (_state == kHeroStateTalk) {
|
|
if (_talkTime) {
|
|
switch (_lastDirection) {
|
|
case kHeroDirLeft:
|
|
_moveSetType = kMove_TL;
|
|
break;
|
|
case kHeroDirRight:
|
|
_moveSetType = kMove_TR;
|
|
break;
|
|
case kHeroDirUp:
|
|
_moveSetType = kMove_TU;
|
|
break;
|
|
case kHeroDirDown:
|
|
_moveSetType = kMove_TD;
|
|
break;
|
|
}
|
|
if (_phase < _moveSet[_moveSetType]->getPhaseCount() - 1) {
|
|
_phase++;
|
|
} else {
|
|
_phase = _moveSet[_moveSetType]->getLoopCount();
|
|
}
|
|
} else {
|
|
_state = kHeroStateStay;
|
|
}
|
|
}
|
|
|
|
if (_state == kHeroStateBore) {
|
|
switch (_boreNum) {
|
|
case 0:
|
|
_moveSetType = kMove_BORED1;
|
|
break;
|
|
case 1:
|
|
_moveSetType = kMove_BORED2;
|
|
break;
|
|
}
|
|
if (_moveSet[_moveSetType] != nullptr) {
|
|
if (_phase < _moveSet[_moveSetType]->getPhaseCount() - 1) {
|
|
_phase++;
|
|
} else {
|
|
_phase = 0;
|
|
_lastDirection = kHeroDirDown;
|
|
_state = kHeroStateStay;
|
|
}
|
|
} else {
|
|
_state = kHeroStateStay;
|
|
}
|
|
}
|
|
|
|
if (_state == kHeroStateStay) {
|
|
if (!_vm->_optionsFlag) {
|
|
if (!_vm->_interpreter->getLastOPCode() || !_vm->_interpreter->getFgOpcodePC()) {
|
|
_boredomTime++;
|
|
if (_boredomTime == _maxBoredom) {
|
|
_boreNum =_vm->_randomSource.getRandomNumber(1); // rand one of two 'bored' animation
|
|
_phase = 0;
|
|
_state = kHeroStateBore;
|
|
if (_lastDirection == kHeroDirUp) {
|
|
_lastDirection = kHeroDirLeft;
|
|
} else {
|
|
_lastDirection = kHeroDirDown;
|
|
}
|
|
}
|
|
} else {
|
|
_boredomTime = 0;
|
|
}
|
|
} else {
|
|
_boredomTime = 0;
|
|
}
|
|
heroStanding();
|
|
}
|
|
|
|
if (_state == kHeroStateTurn) {
|
|
if (_destDirection && (_lastDirection != _destDirection)) {
|
|
_phase = 0;
|
|
int rotateDir = rotateHero(_lastDirection, _destDirection);
|
|
_lastDirection = _destDirection;
|
|
if (rotateDir) {
|
|
_moveSetType = rotateDir;
|
|
_state = kHeroStateTran;
|
|
} else {
|
|
_state = kHeroStateStay;
|
|
heroStanding();
|
|
}
|
|
} else {
|
|
_state = kHeroStateStay;
|
|
heroStanding();
|
|
}
|
|
}
|
|
|
|
if (_state == kHeroStateTran) {
|
|
if (_moveSet[_moveSetType] != nullptr) {
|
|
// only in bear form
|
|
if (_phase < _moveSet[_moveSetType]->getPhaseCount() - 2) {
|
|
_phase += 2;
|
|
} else {
|
|
_state = kHeroStateStay;
|
|
heroStanding();
|
|
}
|
|
} else {
|
|
_state = kHeroStateStay;
|
|
heroStanding();
|
|
}
|
|
}
|
|
|
|
if (_state == kHeroStateMvan) {
|
|
if (_moveSet[_moveSetType] != nullptr) {
|
|
// only in bear form
|
|
if (_phase < _moveSet[_moveSetType]->getPhaseCount() - 2) {
|
|
_phase += 2;
|
|
} else {
|
|
_state = kHeroStateMove;
|
|
}
|
|
} else {
|
|
_state = kHeroStateMove;
|
|
}
|
|
}
|
|
|
|
if (_state == kHeroStateDelayMove) {
|
|
_moveDelay--;
|
|
if (!_moveDelay) {
|
|
_state = kHeroStateMove;
|
|
}
|
|
}
|
|
|
|
int x, y, dir;
|
|
|
|
if (_state == kHeroStateMove || _state == kHeroStateRun) {
|
|
//go_for_it:
|
|
while (1) {
|
|
if (_currCoords != nullptr) {
|
|
if (READ_UINT32(_currCoords) != 0xFFFFFFFF) {
|
|
x = READ_UINT16(_currCoords);
|
|
y = READ_UINT16(_currCoords + 2);
|
|
_currCoords += 4;
|
|
dir = *_currDirTab;
|
|
_currDirTab++;
|
|
if (_lastDirection != dir) {
|
|
_phase = 0;
|
|
int rotateDir = rotateHero(_lastDirection, dir);
|
|
_lastDirection = dir;
|
|
if (_moveSet[rotateDir] != nullptr) {
|
|
// only in bear form
|
|
_state = kHeroStateMvan;
|
|
_moveSetType = rotateDir;
|
|
if (_phase < _moveSet[_moveSetType]->getPhaseCount() - 2) {
|
|
_phase += 2;
|
|
break;
|
|
} else {
|
|
_state = kHeroStateMove;
|
|
continue;
|
|
}
|
|
} else {
|
|
continue;
|
|
}
|
|
}
|
|
//no_need_direction_change
|
|
if (dir == kHeroDirLeft) {
|
|
if (_middleX - x >= _step) {
|
|
heroMoveGotIt(x, y, dir);
|
|
break;
|
|
}
|
|
} else if (dir == kHeroDirRight) {
|
|
if (x - _middleX >= _step) {
|
|
heroMoveGotIt(x, y, dir);
|
|
break;
|
|
}
|
|
} else if (dir == kHeroDirUp) {
|
|
if (_middleY - y >= _step) {
|
|
heroMoveGotIt(x, y, dir);
|
|
break;
|
|
}
|
|
} else if (dir == kHeroDirDown) {
|
|
if (y - _middleY >= _step) {
|
|
heroMoveGotIt(x, y, dir);
|
|
break;
|
|
}
|
|
}
|
|
} else {
|
|
//finito
|
|
_middleX = READ_UINT16(_currCoords - 4);
|
|
_middleY = READ_UINT16(_currCoords - 2);
|
|
selectZoom();
|
|
|
|
if (_coords != nullptr) {
|
|
free(_coords);
|
|
_coords = nullptr;
|
|
_currCoords = nullptr;
|
|
}
|
|
|
|
if (_dirTab != nullptr) {
|
|
free(_dirTab);
|
|
_dirTab = nullptr;
|
|
_currDirTab = nullptr;
|
|
}
|
|
|
|
_boredomTime = 0;
|
|
_phase = 0;
|
|
_state = kHeroStateTurn;
|
|
|
|
if (!_destDirection) {
|
|
_destDirection = _lastDirection;
|
|
}
|
|
|
|
heroStanding();
|
|
|
|
break;
|
|
}
|
|
} else {
|
|
heroStanding();
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
countDrawPosition();
|
|
}
|
|
}
|
|
|
|
void Hero::drawHero() {
|
|
if (_visible && !_vm->_flags->getFlagValue(Flags::NOHEROATALL)) {
|
|
freeZoomedSurface();
|
|
Graphics::Surface *mainHeroSurface = getSurface();
|
|
if (mainHeroSurface) {
|
|
DrawNode newDrawNode;
|
|
newDrawNode.posX = _drawX;
|
|
newDrawNode.posY = _drawY;
|
|
newDrawNode.posZ = _drawZ;
|
|
newDrawNode.width = 0;
|
|
newDrawNode.height = 0;
|
|
newDrawNode.originalRoomSurface = nullptr;
|
|
newDrawNode.data = nullptr;
|
|
newDrawNode.drawFunction = &_graph->drawTransparentDrawNode;
|
|
|
|
if (_zoomFactor) {
|
|
_zoomedHeroSurface = zoomSprite(mainHeroSurface);
|
|
newDrawNode.s = _zoomedHeroSurface;
|
|
} else {
|
|
newDrawNode.s = mainHeroSurface;
|
|
}
|
|
_vm->_drawNodeList.push_back(newDrawNode);
|
|
|
|
drawHeroShadow(mainHeroSurface);
|
|
|
|
}
|
|
}
|
|
}
|
|
|
|
void Hero::drawHeroShadow(Graphics::Surface *heroFrame) {
|
|
DrawNode newDrawNode;
|
|
newDrawNode.posX = _middleX - _scaledFrameXSize / 2;
|
|
newDrawNode.posY = _middleY - _shadMinus - 1;
|
|
newDrawNode.posZ = kHeroShadowZ;
|
|
newDrawNode.width = 0;
|
|
newDrawNode.height = 0;
|
|
newDrawNode.originalRoomSurface = nullptr;
|
|
newDrawNode.data = this;
|
|
newDrawNode.drawFunction = &showHeroShadow;
|
|
newDrawNode.s = heroFrame;
|
|
_vm->_drawNodeList.push_back(newDrawNode);
|
|
}
|
|
|
|
void Hero::heroMoveGotIt(int x, int y, int dir) {
|
|
_middleX = x;
|
|
_middleY = y;
|
|
selectZoom();
|
|
|
|
switch (dir) {
|
|
case kHeroDirLeft:
|
|
_moveSetType = kMove_ML;
|
|
break;
|
|
case kHeroDirRight:
|
|
_moveSetType = kMove_MR;
|
|
break;
|
|
case kHeroDirUp:
|
|
_moveSetType = kMove_MU;
|
|
break;
|
|
case kHeroDirDown:
|
|
_moveSetType = kMove_MD;
|
|
break;
|
|
}
|
|
|
|
if (_vm->_flags->getFlagValue(Flags::HEROFAST) || _state == kHeroStateRun) {
|
|
if (_phase < _moveSet[_moveSetType]->getPhaseCount() - 2) {
|
|
_phase += 2;
|
|
} else {
|
|
_phase = 0;
|
|
}
|
|
} else {
|
|
if (_phase < _moveSet[_moveSetType]->getPhaseCount() - 1) {
|
|
_phase++;
|
|
} else {
|
|
_phase = 0;
|
|
}
|
|
}
|
|
|
|
_step = kStepLeftRight;
|
|
if (_moveSetType == kMove_MU || _moveSetType == kMove_MD) {
|
|
_step = kStepUpDown;
|
|
}
|
|
if (_vm->_flags->getFlagValue(Flags::HEROFAST)) {
|
|
_step *= 2.5;
|
|
} else if (_state == kHeroStateRun) {
|
|
_step *= 2;
|
|
}
|
|
}
|
|
|
|
//TODO - test this
|
|
void Hero::scrollHero() {
|
|
int scrollType = _vm->_flags->getFlagValue(Flags::SCROLLTYPE);
|
|
int position = _middleX;
|
|
int scrollValue, scrollValue2;
|
|
|
|
switch (scrollType) {
|
|
case 0:
|
|
position = _middleX;
|
|
break;
|
|
case 1:
|
|
scrollValue = _vm->_flags->getFlagValue(Flags::SCROLLVALUE);
|
|
position = _vm->_normAnimList[scrollValue]._currX + _vm->_normAnimList[scrollValue]._currW / 2;
|
|
break;
|
|
case 2:
|
|
scrollValue = _vm->_flags->getFlagValue(Flags::SCROLLVALUE);
|
|
scrollValue2 = _vm->_flags->getFlagValue(Flags::SCROLLVALUE2);
|
|
position = scrollValue;
|
|
if (scrollValue < scrollValue2) {
|
|
_vm->_flags->setFlagValue(Flags::SCROLLVALUE, 0);
|
|
} else {
|
|
_vm->_flags->setFlagValue(Flags::SCROLLVALUE, scrollValue - scrollValue2);
|
|
}
|
|
break;
|
|
}
|
|
|
|
int locationWidth = _vm->_sceneWidth;
|
|
int difference = locationWidth - _vm->kNormalWidth / 2;
|
|
|
|
int destValue = 0;
|
|
if (position > _vm->kNormalWidth / 2) {
|
|
destValue = difference - _vm->kNormalWidth / 2;
|
|
}
|
|
if (position < difference) {
|
|
destValue = position - _vm->kNormalWidth / 2;
|
|
}
|
|
|
|
if(destValue < 0) {
|
|
destValue = 0;
|
|
}
|
|
_vm->_picWindowX = destValue;
|
|
_drawX -= destValue;
|
|
}
|
|
|
|
void Hero::freeOldMove() {
|
|
if (_coords != nullptr) {
|
|
free(_coords);
|
|
_coords = nullptr;
|
|
}
|
|
if (_dirTab != nullptr) {
|
|
free(_dirTab);
|
|
_dirTab = nullptr;
|
|
}
|
|
_step = 0;
|
|
_phase = 0;
|
|
_moveDelay = 0;
|
|
_state = Hero::kHeroStateStay;
|
|
}
|
|
|
|
void Hero::freeHeroAnim() {
|
|
if (_specAnim != nullptr) {
|
|
delete _specAnim;
|
|
_specAnim = nullptr;
|
|
}
|
|
}
|
|
|
|
void Hero::freeZoomedSurface() {
|
|
if (_zoomedHeroSurface != nullptr) {
|
|
_zoomedHeroSurface->free();
|
|
delete _zoomedHeroSurface;
|
|
_zoomedHeroSurface = nullptr;
|
|
}
|
|
}
|
|
|
|
} // End of namespace Prince
|