scummvm/engines/kyra/animator_v3.cpp
Johannes Schickel ffc1e56cbd More kyra3 item drop down code.
svn-id: r31685
2008-04-24 13:05:14 +00:00

658 lines
18 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.
*
* $URL$
* $Id$
*
*/
#include "kyra/kyra_v3.h"
#include "kyra/wsamovie.h"
namespace Kyra {
void KyraEngine_v3::clearAnimObjects() {
debugC(9, kDebugLevelAnimator, "KyraEngine_v3::clearAnimObjects()");
for (int i = 0; i < 67; ++i)
_animObjects[i].enabled = false;
_animObjects[0].index = 0;
_animObjects[0].type = 0;
_animObjects[0].enabled = true;
_animObjects[0].unk8 = 1;
_animObjects[0].flags = 0x800;
_animObjects[0].width = 57;
_animObjects[0].height = 91;
_animObjects[0].width2 = 4;
_animObjects[0].height2 = 10;
for (int i = 1; i < 17; ++i) {
_animObjects[i].index = i;
_animObjects[i].type = 2;
_animObjects[i].flags = 0;
_animObjects[i].enabled = false;
_animObjects[i].needRefresh = 0;
_animObjects[i].unk8 = 1;
}
for (int i = 17; i <= 66; ++i) {
_animObjects[i].index = i;
_animObjects[i].type = 1;
_animObjects[i].unk8 = 1;
_animObjects[i].flags = 0x800;
_animObjects[i].width = 24;
_animObjects[i].height = 20;
_animObjects[i].width2 = 0;
_animObjects[i].height2 = 0;
}
}
KyraEngine_v3::AnimObj *KyraEngine_v3::initAnimList(AnimObj *list, AnimObj *entry) {
debugC(9, kDebugLevelAnimator, "KyraEngine_v3::initAnimList(%p, %p)", (const void*)list, (const void*)entry);
entry->nextObject = list;
return entry;
}
KyraEngine_v3::AnimObj *KyraEngine_v3::addToAnimListSorted(AnimObj *list, AnimObj *add) {
debugC(9, kDebugLevelAnimator, "KyraEngine_v3::addToAnimListSorted(%p, %p)", (const void*)list, (const void*)add);
add->nextObject = 0;
if (!list)
return add;
if (add->yPos1 <= list->yPos1) {
add->nextObject = list;
return add;
}
AnimObj *cur = list;
AnimObj *prev = list;
while (add->yPos1 > cur->yPos1) {
AnimObj *temp = cur->nextObject;
if (!temp)
break;
prev = cur;
cur = temp;
}
if (add->yPos1 <= cur->yPos1) {
prev->nextObject = add;
add->nextObject = cur;
} else {
cur->nextObject = add;
add->nextObject = 0;
}
return list;
}
KyraEngine_v3::AnimObj *KyraEngine_v3::deleteAnimListEntry(AnimObj *list, AnimObj *entry) {
debugC(9, kDebugLevelAnimator, "KyraEngine_v3::addToAnimListSorted(%p, %p)", (const void*)list, (const void*)entry);
if (!list)
return 0;
AnimObj *old = 0;
AnimObj *cur = list;
while (true) {
if (cur == entry)
break;
if (!cur->nextObject)
break;
old = cur;
cur = cur->nextObject;
}
if (cur != entry)
return list;
if (cur == list) {
if (!cur->nextObject)
return 0;
cur = cur->nextObject;
return cur;
}
if (!cur->nextObject) {
if (!old)
return 0;
old->nextObject = 0;
return list;
}
if (cur != entry)
return list;
old->nextObject = entry->nextObject;
return list;
}
void KyraEngine_v3::animSetupPaletteEntry(AnimObj *anim) {
debugC(9, kDebugLevelAnimator, "KyraEngine_v3::animSetupPaletteEntry(%p)", (const void*)anim);
int layer = _screen->getLayer(anim->xPos1, anim->yPos1) - 1;
int16 count = 0;
for (int i = 0; i < 3; ++i)
count += _sceneDatPalette[layer*3+i];
count /= 3;
count *= -1;
count = MAX<int16>(0, MIN<int16>(count, 10));
anim->palette = count / 3;
}
void KyraEngine_v3::restorePage3() {
debugC(9, kDebugLevelAnimator, "KyraEngine_v3::restorePage3()");
musicUpdate(0);
_screen->copyBlockToPage(3, 0, 0, 320, 200, _gamePlayBuffer);
musicUpdate(0);
}
void KyraEngine_v3::drawAnimObjects() {
debugC(9, kDebugLevelAnimator, "KyraEngine_v3::drawAnimObjects()");
for (AnimObj *curObject = _animList; curObject; curObject = curObject->nextObject) {
if (!curObject->enabled)
continue;
int x = curObject->xPos2 - (_screen->getScreenDim(2)->sx << 3);
int y = curObject->yPos2 - _screen->getScreenDim(2)->sy;
int layer = 7;
if (curObject->flags & 0x800) {
if (!curObject->unk8)
layer = 0;
else
layer = getDrawLayer(curObject->xPos1, curObject->yPos1);
}
if (curObject->index)
drawSceneAnimObject(curObject, x, y, layer);
else
drawCharacterAnimObject(curObject, x, y, layer);
}
}
void KyraEngine_v3::drawSceneAnimObject(AnimObj *obj, int x, int y, int layer) {
debugC(9, kDebugLevelAnimator, "KyraEngine_v3::drawSceneAnimObject(%p, %d, %d, %d)", (const void*)obj, x, y, layer);
if (obj->type == 1) {
if (obj->shapeIndex == 0xFFFF)
return;
int scale = getScale(obj->xPos1, obj->yPos1);
_screen->drawShape(2, getShapePtr(obj->shapeIndex), x, y, 2, obj->flags | 0x104, _paletteOverlay, obj->palette, layer, scale, scale);
} else {
if (obj->shapePtr) {
_screen->drawShape(2, obj->shapePtr, x, y, 2, obj->flags, 7);
} else {
if (obj->shapeIndex3 == 0xFFFF || obj->animNum == 0xFFFF)
return;
uint16 flags = 0x4000;
if (obj->flags & 0x800)
flags |= 0x8000;
x = obj->xPos2 - _sceneAnimMovie[obj->animNum]->xAdd();
y = obj->yPos2 - _sceneAnimMovie[obj->animNum]->yAdd();
_sceneAnimMovie[obj->animNum]->setDrawPage(2);
_sceneAnimMovie[obj->animNum]->setX(x);
_sceneAnimMovie[obj->animNum]->setY(y);
_sceneAnimMovie[obj->animNum]->displayFrame(obj->shapeIndex3, flags | layer);
}
}
}
void KyraEngine_v3::drawCharacterAnimObject(AnimObj *obj, int x, int y, int layer) {
debugC(9, kDebugLevelAnimator, "KyraEngine_v3::drawCharacterAnimObject(%p, %d, %d, %d)", (const void*)obj, x, y, layer);
if (_drawNoShapeFlag)
return;
if (_mainCharacter.animFrame < 9)
_mainCharacter.animFrame = 87;
if (obj->shapeIndex == 0xFFFF || _mainCharacter.animFrame == 87)
return;
_screen->drawShape(2, getShapePtr(421), _mainCharacter.x3, _mainCharacter.y3, 2, obj->flags | 0x304, _paletteOverlay, 3, layer, _charScale, _charScale);
uint8 *shape = getShapePtr(_mainCharacter.animFrame);
if (shape)
_screen->drawShape(2, shape, x, y, 2, obj->flags | 4, layer, _charScale, _charScale);
}
void KyraEngine_v3::refreshAnimObjects(int force) {
debugC(9, kDebugLevelAnimator, "KyraEngine_v3::refreshAnimObjects(%d)", force);
for (AnimObj *curObject = _animList; curObject; curObject = curObject->nextObject) {
if (!curObject->enabled)
continue;
if (!curObject->needRefresh && !force)
continue;
const int scale = (curObject->index == 0) ? _charScale : 0;
int x = curObject->xPos2 - curObject->width2;
if (scale)
x -= (0x100 - scale) >> 4;
if (x < 0)
x = 0;
if (x >= 320)
x = 319;
int y = curObject->yPos2 - curObject->height2;
if (scale)
y -= (0x100 - scale) >> 3;
if (y < 0)
y = 0;
if (y >= 187)
y = 186;
int width = curObject->width + curObject->width2 + 8;
int height = curObject->height + curObject->height2*2;
if (width + x > 320)
width -= width + x - 322;
const int maxY = _inventoryState ? 143 : 187;
if (height + y > maxY)
height -= height + y - (maxY + 1);
if (height > 0) {
_screen->hideMouse();
_screen->copyRegion(x, y, x, y, width, height, 2, 0, Screen::CR_NO_P_CHECK);
_screen->showMouse();
}
curObject->needRefresh = false;
}
}
void KyraEngine_v3::refreshAnimObjectsIfNeed() {
debugC(9, kDebugLevelAnimator, "KyraEngine_v3::refreshAnimObjectsIfNeed()");
for (AnimObj *curEntry = _animList; curEntry; curEntry = curEntry->nextObject) {
if (curEntry->enabled && curEntry->needRefresh) {
restorePage3();
drawAnimObjects();
refreshAnimObjects(0);
_screen->updateScreen();
return;
}
}
}
void KyraEngine_v3::flagAnimObjsForRefresh() {
debugC(9, kDebugLevelAnimator, "KyraEngine_v3::flagAnimObjsForRefresh()");
for (AnimObj *curEntry = _animList; curEntry; curEntry = curEntry->nextObject)
curEntry->needRefresh = true;
}
void KyraEngine_v3::updateCharacterAnim(int charId) {
debugC(9, kDebugLevelAnimator, "KyraEngine_v3::updateCharacterAnim(%d)", charId);
AnimObj *obj = &_animObjects[0];
obj->needRefresh = true;
obj->flags &= ~1;
obj->xPos1 = _mainCharacter.x1;
obj->yPos1 = _mainCharacter.y1;
obj->shapePtr = getShapePtr(_mainCharacter.animFrame);
obj->shapeIndex = obj->shapeIndex2 = _mainCharacter.animFrame;
int shapeOffsetX = 0, shapeOffsetY = 0;
if (_mainCharacter.animFrame >= 50 && _mainCharacter.animFrame <= 87) {
shapeOffsetX = _malcolmShapeXOffset;
shapeOffsetY = _malcolmShapeYOffset;
} else {
shapeOffsetX = _newShapeXAdd;
shapeOffsetY = _newShapeYAdd;
}
obj->xPos2 = _mainCharacter.x1;
obj->yPos2 = _mainCharacter.y1;
_charScale = getScale(_mainCharacter.x1, _mainCharacter.y1);
obj->xPos2 += (shapeOffsetX * _charScale) >> 8;
obj->yPos2 += (shapeOffsetY * _charScale) >> 8;
_mainCharacter.x3 = _mainCharacter.x1 - (_charScale >> 4) - 1;
_mainCharacter.y3 = _mainCharacter.y1 - (_charScale >> 6) - 1;
if (_charBackUpWidth2 == -1) {
obj->width2 = 4;
obj->height2 = 10;
}
for (int i = 1; i <= 16; ++i) {
if (_animObjects[i].enabled && _animObjects[i].unk8)
_animObjects[i].needRefresh = true;
}
_animList = deleteAnimListEntry(_animList, obj);
if (_animList)
_animList = addToAnimListSorted(_animList, obj);
else
_animList = initAnimList(_animList, obj);
if (!_loadingState)
updateCharPal(1);
}
void KyraEngine_v3::updateSceneAnim(int anim, int newFrame) {
debugC(9, kDebugLevelAnimator, "KyraEngine_v3::updateSceneAnim(%d, %d)", anim, newFrame);
AnimObj *animObject = &_animObjects[1+anim];
if (!animObject->enabled)
return;
animObject->needRefresh = true;
if (_sceneAnims[anim].flags & 2)
animObject->flags |= 1;
else
animObject->flags &= ~1;
if (_sceneAnims[anim].flags & 4) {
animObject->shapePtr = _sceneShapes[newFrame];
animObject->shapeIndex2 = 0xFFFF;
animObject->shapeIndex3 = 0xFFFF;
animObject->animNum = 0xFFFF;
} else {
animObject->shapePtr = 0;
animObject->shapeIndex3 = newFrame;
animObject->animNum = anim;
}
animObject->xPos1 = _sceneAnims[anim].x;
animObject->yPos1 = _sceneAnims[anim].y;
animObject->xPos2 = _sceneAnims[anim].x2;
animObject->yPos2 = _sceneAnims[anim].y2;
if (_sceneAnims[anim].flags & 0x20) {
_animList = deleteAnimListEntry(_animList, animObject);
if (!_animList)
_animList = initAnimList(_animList, animObject);
else
_animList = addToAnimListSorted(_animList, animObject);
}
}
void KyraEngine_v3::setupSceneAnimObject(int animId, uint16 flags, int x, int y, int x2, int y2, int w,
int h, int unk10, int specialSize, int unk14, int shape, const char *filename) {
debugC(9, kDebugLevelAnimator, "KyraEngine_v3::setupSceneAnimObject(%d, %d, %d, %d, %d, %d, %d, %d, %d, %d, %d, %d, '%s')",
animId, flags, x, y, x2, y2, w, h, unk10, specialSize, unk14, shape, filename);
restorePage3();
SceneAnim &anim = _sceneAnims[animId];
anim.flags = flags;
anim.x = x;
anim.y = y;
anim.x2 = x2;
anim.y2 = y2;
anim.width = w;
anim.height = h;
anim.unk10 = unk10;
anim.specialSize = specialSize;
anim.unk14 = unk14;
anim.shapeIndex = shape;
if (filename)
strcpy(anim.filename, filename);
if (flags & 8) {
_sceneAnimMovie[animId]->open(filename, 1, 0);
musicUpdate(0);
if (_sceneAnimMovie[animId]->opened()) {
anim.wsaFlag = 1;
if (x2 == -1)
x2 = _sceneAnimMovie[animId]->xAdd();
if (y2 == -1)
y2 = _sceneAnimMovie[animId]->yAdd();
if (w == -1)
w = _sceneAnimMovie[animId]->width();
if (h == -1)
h = _sceneAnimMovie[animId]->height();
if (x == -1)
x = (w >> 1) + x2;
if (y == -1)
y = y2 + h - 1;
anim.x = x;
anim.y = y;
anim.x2 = x2;
anim.y2 = y2;
anim.width = w;
anim.height = h;
}
}
AnimObj *obj = &_animObjects[1+animId];
obj->enabled = true;
obj->needRefresh = true;
obj->unk8 = (anim.flags & 0x20) ? 1 : 0;
obj->flags = (anim.flags & 0x10) ? 0x800 : 0;
if (anim.flags & 2)
obj->flags |= 1;
obj->xPos1 = anim.x;
obj->yPos1 = anim.y;
if ((anim.flags & 4) && anim.shapeIndex != 0xFFFF)
obj->shapePtr = _sceneShapes[anim.shapeIndex];
else
obj->shapePtr = 0;
if (anim.flags & 8) {
obj->shapeIndex3 = anim.shapeIndex;
obj->animNum = animId;
} else {
obj->shapeIndex3 = 0xFFFF;
obj->animNum = 0xFFFF;
}
obj->xPos3 = obj->xPos2 = anim.x2;
obj->yPos3 = obj->yPos2 = anim.y2;
obj->width = anim.width;
obj->height = anim.height;
obj->width2 = obj->height2 = anim.specialSize;
if (_animList)
_animList = addToAnimListSorted(_animList, obj);
else
_animList = initAnimList(_animList, obj);
}
void KyraEngine_v3::removeSceneAnimObject(int anim, int refresh) {
debugC(9, kDebugLevelAnimator, "KyraEngine_v3::removeSceneAnimObject(%d, %d)", anim, refresh);
AnimObj *obj = &_animObjects[anim+1];
restorePage3();
obj->shapeIndex3 = 0xFFFF;
obj->animNum = 0xFFFF;
obj->needRefresh = true;
if (refresh)
refreshAnimObjectsIfNeed();
obj->enabled = false;
_animList = deleteAnimListEntry(_animList, obj);
_sceneAnimMovie[anim]->close();
}
void KyraEngine_v3::setCharacterAnimDim(int w, int h) {
debugC(9, kDebugLevelAnimator, "KyraEngine_v3::setCharacterAnimDim(%d, %d)", w, h);
restorePage3();
_charBackUpWidth = _animObjects[0].width;
_charBackUpWidth2 = _animObjects[0].width2;
_charBackUpHeight = _animObjects[0].height;
_charBackUpHeight2 = _animObjects[0].height2;
_animObjects[0].width2 = (w - _charBackUpWidth) / 2;
_animObjects[0].height2 = h - _charBackUpHeight;
_animObjects[0].width = w;
_animObjects[0].height = h;
}
void KyraEngine_v3::resetCharacterAnimDim() {
debugC(9, kDebugLevelAnimator, "KyraEngine_v3::resetCharacterAnimDim()");
restorePage3();
_animObjects[0].width2 = _charBackUpWidth2;
_animObjects[0].height2 = _charBackUpHeight2;
_animObjects[0].width = _charBackUpWidth;
_animObjects[0].height = _charBackUpHeight;
_charBackUpWidth2 = _charBackUpHeight2 = -1;
_charBackUpWidth = _charBackUpHeight = -1;
}
void KyraEngine_v3::showIdleAnim() {
debugC(9, kDebugLevelMain | kDebugLevelAnimator, "KyraEngine_v3::showIdleAnim()");
if (_mainCharacter.sceneId == 20 || _mainCharacter.sceneId == 21
|| _mainCharacter.sceneId == 12 || _mainCharacter.sceneId == 11)
return;
if (_mainCharacter.animFrame == 87)
return;
if (!_nextIdleType && !talkObjectsInCurScene()) {
malcolmRandomChat();
} else {
static const char *facingTable[] = {
"A", "R", "R", "FR", "FX", "FL", "L", "L"
};
char filename[14];
snprintf(filename, 14, "MI0%s%.02d.EMC", facingTable[_mainCharacter.facing], _malcolmShapes);
if (_res->exists(filename))
runTemporaryScript(filename, 1, 1, 1, 1);
}
_nextIdleType = !_nextIdleType;
}
int KyraEngine_v3::initNewShapes(uint8 *filedata) {
debugC(9, kDebugLevelAnimator, "KyraEngine_v3::initNewShapes(%p)", (const void*)filedata);
const int lastEntry = MIN(_newShapeLastEntry, 41);
for (int i = 0; i < lastEntry; ++i)
_gameShapes[9+i] = _screen->getPtrToShape(filedata, i);
return lastEntry;
}
void KyraEngine_v3::processNewShapes(int allowSkip, int resetChar) {
debugC(9, kDebugLevelAnimator, "KyraEngine_v3::processNewShapes(%d, %d)", allowSkip, resetChar);
setCharacterAnimDim(_newShapeWidth, _newShapeHeight);
_scriptInterpreter->initScript(&_temporaryScriptState, &_temporaryScriptData);
_scriptInterpreter->startScript(&_temporaryScriptState, 1);
resetSkipFlag();
while (_scriptInterpreter->validScript(&_temporaryScriptState)) {
_temporaryScriptExecBit = false;
while (_scriptInterpreter->validScript(&_temporaryScriptState) && !_temporaryScriptExecBit)
_scriptInterpreter->runScript(&_temporaryScriptState);
if (_newShapeAnimFrame < 0)
continue;
_mainCharacter.animFrame = _newShapeAnimFrame + 9;
updateCharacterAnim(0);
if (_chatText)
updateWithText();
else
update();
uint32 delayEnd = _system->getMillis() + _newShapeDelay * _tickLength;
while ((!skipFlag() || !allowSkip) && _system->getMillis() < delayEnd) {
if (_chatText)
updateWithText();
else
update();
delay(10);
}
if (skipFlag())
resetSkipFlag();
}
if (resetChar) {
if (_newShapeFlag >= 0) {
_mainCharacter.animFrame = _newShapeFlag + 9;
updateCharacterAnim(0);
if (_chatText)
updateWithText();
else
update();
}
_mainCharacter.animFrame = _characterFrameTable[_mainCharacter.facing];
updateCharacterAnim(0);
}
_newShapeFlag = -1;
resetCharacterAnimDim();
}
void KyraEngine_v3::resetNewShapes(int count, uint8 *filedata) {
debugC(9, kDebugLevelAnimator, "KyraEngine_v3::resetNewShapes(%d, %p)", count, (const void*)filedata);
for (int i = 0; i < count; ++i)
_gameShapes[9+i] = 0;
delete [] filedata;
setNextIdleAnimTimer();
}
void KyraEngine_v3::addItemToAnimList(int item) {
debugC(9, kDebugLevelAnimator, "KyraEngine_v3::addItemToAnimList(%d)", item);
restorePage3();
AnimObj *animObj = &_animObjects[17+item];
animObj->enabled = 1;
animObj->needRefresh = 1;
int itemId = _itemList[item].id;
animObj->xPos2 = animObj->xPos1 = _itemList[item].x;
animObj->yPos2 = animObj->yPos1 = _itemList[item].y;
animObj->shapePtr = getShapePtr(248+itemId);
animSetupPaletteEntry(animObj);
animObj->shapeIndex2 = animObj->shapeIndex = 248+itemId;
int scaleY, scaleX;
scaleY = scaleX = getScale(animObj->xPos1, animObj->yPos1);
uint8 *shapePtr = getShapePtr(248+itemId);
animObj->xPos3 = (animObj->xPos2 -= (_screen->getShapeScaledWidth(shapePtr, scaleX) >> 1));
animObj->yPos3 = (animObj->yPos2 -= _screen->getShapeScaledHeight(shapePtr, scaleY));
animObj->width2 = animObj->height2 = 0;
_animList = addToAnimListSorted(_animList, animObj);
animObj->needRefresh = 1;
}
void KyraEngine_v3::deleteItemAnimEntry(int item) {
debugC(9, kDebugLevelAnimator, "KyraEngine_v3::deleteItemAnimEntry(%d)", item);
AnimObj *animObj = &_animObjects[17+item];
restorePage3();
animObj->shapePtr = 0;
animObj->shapeIndex = 0xFFFF;
animObj->shapeIndex2 = 0xFFFF;
animObj->needRefresh = 1;
refreshAnimObjectsIfNeed();
animObj->enabled = 0;
_animList = deleteAnimListEntry(_animList, animObj);
}
} // end of namespace Kyra