mirror of
https://github.com/libretro/scummvm.git
synced 2025-01-10 11:51:52 +00:00
1535 lines
45 KiB
C++
1535 lines
45 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 "illusions/illusions.h"
|
|
#include "illusions/actor.h"
|
|
#include "illusions/camera.h"
|
|
#include "illusions/cursor.h"
|
|
#include "illusions/dictionary.h"
|
|
#include "illusions/fixedpoint.h"
|
|
#include "illusions/input.h"
|
|
#include "illusions/screen.h"
|
|
#include "illusions/scriptopcodes.h"
|
|
#include "illusions/sequenceopcodes.h"
|
|
#include "illusions/thread.h"
|
|
#include "illusions/threads/talkthread.h"
|
|
|
|
namespace Illusions {
|
|
|
|
// DefaultSequences
|
|
|
|
uint32 DefaultSequences::use(uint32 sequenceId) {
|
|
ItemsIterator it = Common::find_if(_items.begin(), _items.end(), DefaultSequenceEqual(sequenceId));
|
|
return it != _items.end() ? (*it)._newSequenceId : sequenceId;
|
|
}
|
|
|
|
void DefaultSequences::set(uint32 sequenceId, uint32 newSequenceId) {
|
|
ItemsIterator it = Common::find_if(_items.begin(), _items.end(), DefaultSequenceEqual(sequenceId));
|
|
if (it == _items.end())
|
|
_items.push_back(DefaultSequence(sequenceId, newSequenceId));
|
|
else if (sequenceId == newSequenceId)
|
|
_items.remove_at(it - _items.begin());
|
|
else
|
|
(*it)._newSequenceId = newSequenceId;
|
|
}
|
|
|
|
// Actor
|
|
|
|
Actor::Actor(IllusionsEngine *vm)
|
|
: _vm(vm), _pauseCtr(0) {
|
|
_pauseCtr = 0;
|
|
_spriteFlags = 0;
|
|
_drawFlags = 0;
|
|
_flags = 0;
|
|
_scale = 100;
|
|
_frameIndex = 0;
|
|
_newFrameIndex = 0;
|
|
_surfInfo._pixelSize = 0;
|
|
_surfInfo._dimensions._width = 0;
|
|
_surfInfo._dimensions._height = 0;
|
|
_surface = 0;
|
|
_frames = 0;
|
|
_scaleLayer = 0;
|
|
_priorityLayer = 0;
|
|
_regionLayer = 0;
|
|
_pathWalkPoints = 0;
|
|
_pathWalkRects = 0;
|
|
_position.x = 0;
|
|
_position.y = 0;
|
|
_position2.x = 0;
|
|
_position2.y = 0;
|
|
_facing = 64;
|
|
_regionIndex = 0;
|
|
_fontId = 0;
|
|
_actorIndex = 0;
|
|
_parentObjectId = 0;
|
|
_linkIndex = 0;
|
|
_linkIndex2 = 0;
|
|
for (uint i = 0; i < kSubObjectsCount; ++i) {
|
|
_subobjects[i] = 0;
|
|
}
|
|
_notifyThreadId1 = 0;
|
|
_notifyThreadId2 = 0;
|
|
_entryTblPtr = 0;
|
|
_seqCodeIp = 0;
|
|
_sequenceId = 0;
|
|
_seqCodeValue1 = 0;
|
|
_seqCodeValue2 = 600;
|
|
_seqCodeValue3 = 0;
|
|
|
|
_notifyId3C = 0;
|
|
|
|
_controlRoutine = 0;
|
|
setControlRoutine(new Common::Functor2Mem<Control*, uint32, void, Controls>(_vm->_controls, &Controls::actorControlRoutine));
|
|
|
|
_walkCallerThreadId1 = 0;
|
|
_pathAngle = 0;
|
|
_pathFlag50 = false;
|
|
_pathCtrX = 0;
|
|
_pathCtrY = 0;
|
|
_pathInitialPosFlag = true;
|
|
_pathInitialPos.x = 0;
|
|
_pathInitialPos.y = 0;
|
|
_pathPoints = 0;
|
|
_pathPointIndex = 0;
|
|
_pathPointsCount = 0;
|
|
_pathNode = 0;
|
|
|
|
}
|
|
|
|
Actor::~Actor() {
|
|
delete _controlRoutine;
|
|
}
|
|
|
|
void Actor::pause() {
|
|
++_pauseCtr;
|
|
}
|
|
|
|
void Actor::unpause() {
|
|
--_pauseCtr;
|
|
}
|
|
|
|
void Actor::createSurface(SurfInfo &surfInfo) {
|
|
_surface = _vm->_screen->allocSurface(surfInfo);
|
|
if (_vm->getGameId() == kGameIdDuckman) {
|
|
if (_flags & Illusions::ACTOR_FLAG_IS_VISIBLE) {
|
|
if (_frameIndex) {
|
|
_flags |= Illusions::ACTOR_FLAG_2000;
|
|
}
|
|
_flags |= Illusions::ACTOR_FLAG_4000;
|
|
}
|
|
} else {
|
|
if (_frameIndex) {
|
|
_flags |= Illusions::ACTOR_FLAG_2000;
|
|
_flags |= Illusions::ACTOR_FLAG_4000;
|
|
}
|
|
}
|
|
}
|
|
|
|
void Actor::destroySurface() {
|
|
if (_surface) {
|
|
_surface->free();
|
|
delete _surface;
|
|
_surface = 0;
|
|
}
|
|
}
|
|
|
|
void Actor::initSequenceStack() {
|
|
_seqStackCount = 0;
|
|
}
|
|
|
|
void Actor::pushSequenceStack(int16 value) {
|
|
_seqStack[_seqStackCount++] = value;
|
|
}
|
|
|
|
int16 Actor::popSequenceStack() {
|
|
return _seqStack[--_seqStackCount];
|
|
}
|
|
|
|
void Actor::setControlRoutine(ActorControlRoutine *controlRoutine) {
|
|
delete _controlRoutine;
|
|
_controlRoutine = controlRoutine;
|
|
}
|
|
|
|
void Actor::runControlRoutine(Control *control, uint32 deltaTime) {
|
|
if (_controlRoutine)
|
|
(*_controlRoutine)(control, deltaTime);
|
|
}
|
|
|
|
bool Actor::findNamedPoint(uint32 namedPointId, Common::Point &pt) {
|
|
if (_namedPoints->findNamedPoint(namedPointId, pt)) {
|
|
pt.x += _position.x;
|
|
pt.y += _position.y;
|
|
return true;
|
|
}
|
|
return false;
|
|
}
|
|
|
|
// Control
|
|
|
|
Control::Control(IllusionsEngine *vm)
|
|
: _vm(vm) {
|
|
_flags = 0;
|
|
_pauseCtr = 0;
|
|
_priority = 0;
|
|
_objectId = 0;
|
|
_bounds._topLeft.x = 0;
|
|
_bounds._topLeft.y = 0;
|
|
_bounds._bottomRight.x = 0;
|
|
_bounds._bottomRight.y = 0;
|
|
_feetPt.x = 0;
|
|
_feetPt.y = 0;
|
|
_position.x = 0;
|
|
_position.y = 0;
|
|
_actorTypeId = 0;
|
|
_actor = 0;
|
|
_sceneId = _vm->getCurrentScene();
|
|
}
|
|
|
|
Control::~Control() {
|
|
}
|
|
|
|
void Control::pause() {
|
|
|
|
if (_vm->getGameId() == kGameIdBBDOU || !(_flags & 4)) {
|
|
_vm->_dict->setObjectControl(_objectId, 0);
|
|
if (_objectId == Illusions::CURSOR_OBJECT_ID)
|
|
_vm->setCursorControl(0);
|
|
}
|
|
|
|
if (_actor && !(_actor->_flags & Illusions::ACTOR_FLAG_200))
|
|
_actor->destroySurface();
|
|
|
|
}
|
|
|
|
void Control::unpause() {
|
|
|
|
if (_vm->getGameId() == kGameIdBBDOU || !(_flags & 4)) {
|
|
_vm->_dict->setObjectControl(_objectId, this);
|
|
if (_objectId == Illusions::CURSOR_OBJECT_ID)
|
|
_vm->setCursorControl(this);
|
|
}
|
|
|
|
if (_actor && !(_actor->_flags & Illusions::ACTOR_FLAG_200)) {
|
|
SurfInfo surfInfo;
|
|
ActorType *actorType = _vm->_dict->findActorType(_actorTypeId);
|
|
if (actorType)
|
|
surfInfo = actorType->_surfInfo;
|
|
else
|
|
surfInfo = _actor->_surfInfo;
|
|
_actor->createSurface(surfInfo);
|
|
}
|
|
}
|
|
|
|
void Control::appearActor() {
|
|
if (_vm->getGameId() == kGameIdDuckman) {
|
|
_flags |= 1;
|
|
_actor->_flags |= Illusions::ACTOR_FLAG_IS_VISIBLE;
|
|
if (_objectId == Illusions::CURSOR_OBJECT_ID) {
|
|
if (_actor->_frameIndex) {
|
|
_actor->_flags |= Illusions::ACTOR_FLAG_2000;
|
|
_actor->_flags |= Illusions::ACTOR_FLAG_4000;
|
|
}
|
|
_vm->_input->discardAllEvents();
|
|
}
|
|
} else {
|
|
if (_objectId == Illusions::CURSOR_OBJECT_ID) {
|
|
_vm->showCursor();
|
|
} else {
|
|
if (_actor->_frameIndex || _actorTypeId == 0x50004) {
|
|
_actor->_flags |= Illusions::ACTOR_FLAG_IS_VISIBLE;
|
|
} else {
|
|
_actor->_flags |= Illusions::ACTOR_FLAG_1000;
|
|
}
|
|
for (uint i = 0; i < kSubObjectsCount; ++i) {
|
|
if (_actor->_subobjects[i]) {
|
|
Control *subControl = _vm->_dict->getObjectControl(_actor->_subobjects[i]);
|
|
subControl->appearActor();
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
void Control::disappearActor() {
|
|
if (_vm->getGameId() == kGameIdDuckman) {
|
|
_flags &= ~1;
|
|
_actor->_flags &= ~Illusions::ACTOR_FLAG_IS_VISIBLE;
|
|
} else {
|
|
if (_objectId == Illusions::CURSOR_OBJECT_ID) {
|
|
_vm->hideCursor();
|
|
} else {
|
|
_actor->_flags &= ~Illusions::ACTOR_FLAG_IS_VISIBLE;
|
|
_actor->_flags &= ~Illusions::ACTOR_FLAG_1000;
|
|
for (uint i = 0; i < kSubObjectsCount; ++i) {
|
|
if (_actor->_subobjects[i]) {
|
|
Control *subControl = _vm->_dict->getObjectControl(_actor->_subobjects[i]);
|
|
subControl->disappearActor();
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
bool Control::isActorVisible() {
|
|
return (_actor->_flags & Illusions::ACTOR_FLAG_IS_VISIBLE) != 0;
|
|
}
|
|
|
|
void Control::activateObject() {
|
|
_flags |= 1;
|
|
if (_actor) {
|
|
for (uint i = 0; i < kSubObjectsCount; ++i) {
|
|
if (_actor->_subobjects[i]) {
|
|
Control *subControl = _vm->_dict->getObjectControl(_actor->_subobjects[i]);
|
|
subControl->activateObject();
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
void Control::deactivateObject() {
|
|
_flags &= ~1;
|
|
if (_actor) {
|
|
for (uint i = 0; i < kSubObjectsCount; ++i) {
|
|
if (_actor->_subobjects[i]) {
|
|
Control *subControl = _vm->_dict->getObjectControl(_actor->_subobjects[i]);
|
|
subControl->deactivateObject();
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
void Control::readPointsConfig(byte *pointsConfig) {
|
|
_bounds._topLeft.x = READ_LE_UINT16(pointsConfig + 0);
|
|
_bounds._topLeft.y = READ_LE_UINT16(pointsConfig + 2);
|
|
pointsConfig += 4;
|
|
_bounds._bottomRight.x = READ_LE_UINT16(pointsConfig + 0);
|
|
_bounds._bottomRight.y = READ_LE_UINT16(pointsConfig + 2);
|
|
pointsConfig += 4;
|
|
_feetPt.x = READ_LE_UINT16(pointsConfig + 0);
|
|
_feetPt.y = READ_LE_UINT16(pointsConfig + 2);
|
|
pointsConfig += 4;
|
|
_position.x = READ_LE_UINT16(pointsConfig + 0);
|
|
_position.y = READ_LE_UINT16(pointsConfig + 2);
|
|
pointsConfig += 4;
|
|
for (uint i = 0; i < kSubObjectsCount; ++i) {
|
|
_subobjectsPos[i].x = READ_LE_UINT16(pointsConfig + 0);
|
|
_subobjectsPos[i].y = READ_LE_UINT16(pointsConfig + 2);
|
|
pointsConfig += 4;
|
|
}
|
|
}
|
|
|
|
void Control::setActorPosition(Common::Point position) {
|
|
_actor->_position = position;
|
|
}
|
|
|
|
Common::Point Control::getActorPosition() {
|
|
if (_actor)
|
|
return _actor->_position;
|
|
return _position;
|
|
}
|
|
|
|
void Control::setActorScale(int scale) {
|
|
_actor->_scale = scale;
|
|
for (uint i = 0; i < kSubObjectsCount; ++i) {
|
|
if (_actor->_subobjects[i]) {
|
|
Control *subControl = _vm->_dict->getObjectControl(_actor->_subobjects[i]);
|
|
subControl->setActorScale(scale);
|
|
}
|
|
}
|
|
}
|
|
|
|
void Control::faceActor(uint facing) {
|
|
_actor->_facing = facing;
|
|
for (uint i = 0; i < kSubObjectsCount; ++i) {
|
|
if (_actor->_subobjects[i]) {
|
|
Control *subControl = _vm->_dict->getObjectControl(_actor->_subobjects[i]);
|
|
subControl->faceActor(facing);
|
|
}
|
|
}
|
|
}
|
|
|
|
void Control::linkToObject(uint32 parentObjectId, uint32 linkedObjectValue) {
|
|
_actor->_parentObjectId = parentObjectId;
|
|
_actor->_linkIndex = linkedObjectValue;
|
|
}
|
|
|
|
void Control::unlinkObject() {
|
|
_actor->_parentObjectId = 0;
|
|
_actor->_linkIndex = 0;
|
|
}
|
|
|
|
void Control::clearNotifyThreadId1() {
|
|
_actor->_notifyThreadId1 = 0;
|
|
}
|
|
|
|
void Control::clearNotifyThreadId2() {
|
|
for (uint i = 0; i < kSubObjectsCount; ++i) {
|
|
if (_actor->_subobjects[i]) {
|
|
Control *subControl = _vm->_dict->getObjectControl(_actor->_subobjects[i]);
|
|
subControl->_actor->_flags &= ~Illusions::ACTOR_FLAG_80;
|
|
subControl->_actor->_entryTblPtr = 0;
|
|
subControl->_actor->_notifyThreadId2 = 0;
|
|
}
|
|
}
|
|
_actor->_flags &= ~Illusions::ACTOR_FLAG_80;
|
|
_actor->_entryTblPtr = 0;
|
|
_actor->_notifyThreadId2 = 0;
|
|
}
|
|
|
|
void Control::setPriority(int16 priority) {
|
|
_priority = priority;
|
|
}
|
|
|
|
uint32 Control::getPriority() {
|
|
uint32 objectId;
|
|
int16 positionY, priority, priority1;
|
|
if (_actor) {
|
|
if (_actor->_parentObjectId && (_actor->_flags & Illusions::ACTOR_FLAG_40)) {
|
|
uint32 parentObjectId = getSubActorParent();
|
|
Control *parentControl = _vm->_dict->getObjectControl(parentObjectId);
|
|
objectId = parentControl->_objectId;
|
|
priority = parentControl->_priority;
|
|
positionY = parentControl->_actor->_position.y;
|
|
priority1 = _priority;
|
|
} else {
|
|
objectId = _objectId;
|
|
positionY = _actor->_position.y;
|
|
priority = _priority;
|
|
priority1 = 50;
|
|
}
|
|
} else {
|
|
positionY = _position.y;
|
|
objectId = _objectId;
|
|
priority = _priority;
|
|
priority1 = 1;
|
|
}
|
|
|
|
priority -= 1;
|
|
uint32 p = 50 * priority1 / 100;
|
|
if (p)
|
|
--p;
|
|
|
|
positionY = CLIP<int16>(positionY, -5000, 5000);
|
|
|
|
return p + 50 * ((objectId & 0x3F) + ((10000 * priority + positionY + 5000) << 6));
|
|
}
|
|
|
|
uint32 Control::getOverlapPriority() {
|
|
if (_vm->getGameId() == kGameIdBBDOU)
|
|
return getPriority();
|
|
return _priority;
|
|
}
|
|
|
|
uint32 Control::getDrawPriority() {
|
|
if (_vm->getGameId() == kGameIdBBDOU)
|
|
return getPriority();
|
|
return (_actor->_position.y + 32768) | (_priority << 16);
|
|
}
|
|
|
|
Common::Point Control::calcPosition(Common::Point posDelta) {
|
|
Common::Point pos;
|
|
if (_actor->_parentObjectId) {
|
|
int16 accuX = 0, accuY = 0;
|
|
Actor *actor = _actor;
|
|
while (actor->_parentObjectId) {
|
|
Control *parentControl = _vm->_dict->getObjectControl(actor->_parentObjectId);
|
|
accuX += parentControl->_subobjectsPos[actor->_linkIndex - 1].x;
|
|
accuY += parentControl->_subobjectsPos[actor->_linkIndex - 1].y;
|
|
actor = parentControl->_actor;
|
|
}
|
|
pos = actor->_position;
|
|
pos.x += accuX * actor->_scale / 100;
|
|
pos.y += accuY * actor->_scale / 100;
|
|
_actor->_position = pos;
|
|
if (!(_flags & 8)) {
|
|
pos.x -= posDelta.x;
|
|
pos.y -= posDelta.y;
|
|
}
|
|
} else {
|
|
pos = _actor->_position;
|
|
if (!(_flags & 8)) {
|
|
pos.x -= posDelta.x;
|
|
pos.y -= posDelta.y;
|
|
}
|
|
}
|
|
return pos;
|
|
}
|
|
|
|
uint32 Control::getSubActorParent() {
|
|
uint32 parentObjectId = _objectId;
|
|
while (1) {
|
|
Actor *actor = _vm->_dict->getObjectControl(parentObjectId)->_actor;
|
|
if (actor->_parentObjectId && (actor->_flags & Illusions::ACTOR_FLAG_40))
|
|
parentObjectId = actor->_parentObjectId;
|
|
else
|
|
break;
|
|
}
|
|
return parentObjectId;
|
|
}
|
|
|
|
void Control::getCollisionRectAccurate(Common::Rect &collisionRect) {
|
|
|
|
if (_actor && _actor->_frameIndex) {
|
|
collisionRect = Common::Rect(-_position.x, -_position.y,
|
|
-_position.x + _actor->_surfInfo._dimensions._width - 1,
|
|
-_position.y + _actor->_surfInfo._dimensions._height - 1);
|
|
} else {
|
|
collisionRect = Common::Rect(_bounds._topLeft.x, _bounds._topLeft.y, _bounds._bottomRight.x, _bounds._bottomRight.y);
|
|
}
|
|
|
|
if (_actor) {
|
|
if (_actor->_scale != 100) {
|
|
collisionRect.left = collisionRect.left * _actor->_scale / 100;
|
|
collisionRect.top = collisionRect.top * _actor->_scale / 100;
|
|
collisionRect.right = collisionRect.right * _actor->_scale / 100;
|
|
collisionRect.bottom = collisionRect.bottom * _actor->_scale / 100;
|
|
}
|
|
collisionRect.translate(_actor->_position.x, _actor->_position.y);
|
|
}
|
|
|
|
if (_flags & 8) {
|
|
Common::Point screenOffs = _vm->_camera->getScreenOffset();
|
|
collisionRect.translate(screenOffs.x, screenOffs.y);
|
|
}
|
|
|
|
}
|
|
|
|
void Control::getCollisionRect(Common::Rect &collisionRect) {
|
|
collisionRect = Common::Rect(_bounds._topLeft.x, _bounds._topLeft.y, _bounds._bottomRight.x, _bounds._bottomRight.y);
|
|
if (_actor) {
|
|
if (_actor->_scale != 100) {
|
|
collisionRect.left = collisionRect.left * _actor->_scale / 100;
|
|
collisionRect.top = collisionRect.top * _actor->_scale / 100;
|
|
collisionRect.right = collisionRect.right * _actor->_scale / 100;
|
|
collisionRect.bottom = collisionRect.bottom * _actor->_scale / 100;
|
|
}
|
|
collisionRect.translate(_actor->_position.x, _actor->_position.y);
|
|
}
|
|
if (_flags & 8) {
|
|
Common::Point screenOffs = _vm->_camera->getScreenOffset();
|
|
collisionRect.translate(screenOffs.x, screenOffs.y);
|
|
}
|
|
}
|
|
|
|
void Control::setActorUsePan(int usePan) {
|
|
if (usePan == 1)
|
|
_flags &= ~8;
|
|
else
|
|
_flags |= 8;
|
|
}
|
|
|
|
void Control::setActorFrameIndex(int16 frameIndex) {
|
|
if (frameIndex) {
|
|
_actor->_frameIndex = frameIndex;
|
|
const Frame &frame = (*_actor->_frames)[frameIndex - 1];
|
|
_actor->_surfInfo = frame._surfInfo;
|
|
readPointsConfig(frame._pointsConfig);
|
|
_actor->_flags |= Illusions::ACTOR_FLAG_2000;
|
|
_actor->_flags |= Illusions::ACTOR_FLAG_4000;
|
|
_actor->_newFrameIndex = 0;
|
|
}
|
|
}
|
|
|
|
void Control::stopActor() {
|
|
_actor->_seqCodeIp = 0;
|
|
if (_actor->_pathNode) {
|
|
if (_actor->_flags & Illusions::ACTOR_FLAG_400) {
|
|
delete _actor->_pathNode;
|
|
_actor->_flags &= ~Illusions::ACTOR_FLAG_400;
|
|
}
|
|
_actor->_pathNode = 0;
|
|
_actor->_pathPoints = 0;
|
|
_actor->_pathPointsCount = 0;
|
|
_actor->_pathPointIndex = 0;
|
|
_actor->_walkCallerThreadId1 = 0;
|
|
}
|
|
if (_vm->getGameId() == kGameIdBBDOU) {
|
|
_vm->notifyThreadId(_actor->_notifyId3C);
|
|
_vm->notifyThreadId(_actor->_notifyThreadId1);
|
|
}
|
|
}
|
|
|
|
void Control::startSequenceActor(uint32 sequenceId, int value, uint32 notifyThreadId) {
|
|
startSequenceActorIntern(sequenceId, value, 0, notifyThreadId);
|
|
}
|
|
|
|
void Control::stopSequenceActor() {
|
|
if (_actor->_flags & Illusions::ACTOR_FLAG_40) {
|
|
stopActor();
|
|
_actor->_frameIndex = 0;
|
|
if ((_actor->_flags & Illusions::ACTOR_FLAG_IS_VISIBLE) || (_actor->_flags & Illusions::ACTOR_FLAG_1000)) {
|
|
_actor->_flags &= ~Illusions::ACTOR_FLAG_IS_VISIBLE;
|
|
_actor->_flags |= Illusions::ACTOR_FLAG_1000;
|
|
}
|
|
}
|
|
for (uint i = 0; i < kSubObjectsCount; ++i) {
|
|
if (_actor->_subobjects[i]) {
|
|
Control *subControl = _vm->_dict->getObjectControl(_actor->_subobjects[i]);
|
|
subControl->stopSequenceActor();
|
|
}
|
|
}
|
|
}
|
|
|
|
void Control::startTalkActor(uint32 sequenceId, byte *entryTblPtr, uint32 threadId) {
|
|
bool doSeq = true;
|
|
if (_actor->_linkIndex2) {
|
|
Control *subControl = _vm->_dict->getObjectControl(_actor->_subobjects[_actor->_linkIndex2 - 1]);
|
|
if (subControl->_actor->_flags & Illusions::ACTOR_FLAG_IS_VISIBLE) {
|
|
if (_actor->_pathNode) {
|
|
doSeq = false;
|
|
subControl->_actor->_notifyThreadId2 = threadId;
|
|
subControl->_actor->_entryTblPtr = entryTblPtr;
|
|
subControl->_actor->_flags |= Illusions::ACTOR_FLAG_80;
|
|
Thread *thread = _vm->_threads->findThread(threadId);
|
|
thread->sendMessage(kMsgClearSequenceId2, 0);
|
|
}
|
|
}
|
|
}
|
|
if (doSeq)
|
|
startSequenceActorIntern(sequenceId, 2, entryTblPtr, threadId);
|
|
}
|
|
|
|
void Control::sequenceActor() {
|
|
if (_actor->_pauseCtr > 0)
|
|
return;
|
|
|
|
OpCall opCall;
|
|
bool sequenceFinished = false;
|
|
|
|
opCall._result = 0;
|
|
_actor->_seqCodeValue3 -= _actor->_seqCodeValue1;
|
|
|
|
while (_actor->_seqCodeValue3 <= 0 && !sequenceFinished) {
|
|
bool breakInner = false;
|
|
while (!breakInner) {
|
|
//debug(1, "[%08X] SEQ[%08X] op: %08X", _objectId, _actor->_sequenceId, _actor->_seqCodeIp[0]);
|
|
opCall._op = _actor->_seqCodeIp[0] & 0x7F;
|
|
opCall._opSize = _actor->_seqCodeIp[1];
|
|
opCall._code = _actor->_seqCodeIp + 2;
|
|
opCall._deltaOfs = opCall._opSize;
|
|
if (_actor->_seqCodeIp[0] & 0x80)
|
|
breakInner = true;
|
|
execSequenceOpcode(opCall);
|
|
if (opCall._result == 1) {
|
|
sequenceFinished = true;
|
|
breakInner = true;
|
|
} else if (opCall._result == 2) {
|
|
breakInner = true;
|
|
}
|
|
_actor->_seqCodeIp += opCall._deltaOfs;
|
|
}
|
|
_actor->_seqCodeValue3 += _actor->_seqCodeValue2;
|
|
}
|
|
|
|
if (_actor->_newFrameIndex != 0) {
|
|
//debug(1, "New frame %d", _actor->_newFrameIndex);
|
|
setActorFrameIndex(_actor->_newFrameIndex);
|
|
if (_vm->getGameId() == kGameIdBBDOU &&
|
|
!(_actor->_flags & Illusions::ACTOR_FLAG_IS_VISIBLE) && (_actor->_flags & Illusions::ACTOR_FLAG_1000) && (_objectId != Illusions::CURSOR_OBJECT_ID)) {
|
|
appearActor();
|
|
_actor->_flags &= ~Illusions::ACTOR_FLAG_1000;
|
|
}
|
|
//debug(1, "New frame OK");
|
|
}
|
|
|
|
if (sequenceFinished) {
|
|
//debug(1, "Sequence has finished");
|
|
_actor->_seqCodeIp = 0;
|
|
}
|
|
}
|
|
|
|
void Control::setActorIndex(int actorIndex) {
|
|
_actor->_actorIndex = actorIndex;
|
|
}
|
|
|
|
void Control::setActorIndexTo1() {
|
|
_actor->_actorIndex = 1;
|
|
}
|
|
|
|
void Control::setActorIndexTo2() {
|
|
_actor->_actorIndex = 2;
|
|
}
|
|
|
|
void Control::startSubSequence(int linkIndex, uint32 sequenceId) {
|
|
Control *linkedControl = _vm->_dict->getObjectControl(_actor->_subobjects[linkIndex - 1]);
|
|
Actor *linkedActor = linkedControl->_actor;
|
|
if (!linkedActor->_entryTblPtr)
|
|
linkedActor->_flags &= ~Illusions::ACTOR_FLAG_80;
|
|
linkedActor->_flags &= ~Illusions::ACTOR_FLAG_400;
|
|
linkedActor->_flags |= Illusions::ACTOR_FLAG_100;
|
|
linkedActor->_sequenceId = sequenceId;
|
|
linkedActor->_notifyThreadId1 = 0;
|
|
linkedActor->_notifyId3C = 0;
|
|
linkedActor->_walkCallerThreadId1 = 0;
|
|
|
|
Sequence *sequence = _vm->_dict->findSequence(sequenceId);
|
|
linkedActor->_seqCodeIp = sequence->_sequenceCode;
|
|
linkedActor->_frames = _vm->findActorSequenceFrames(sequence);
|
|
linkedActor->_seqCodeValue3 = 0;
|
|
linkedActor->_seqCodeValue1 = 0;
|
|
linkedActor->_seqCodeValue2 = 600;
|
|
linkedActor->initSequenceStack();
|
|
linkedControl->sequenceActor();
|
|
linkedControl->appearActor();
|
|
}
|
|
|
|
void Control::stopSubSequence(int linkIndex) {
|
|
Control *linkedControl = _vm->_dict->getObjectControl(_actor->_subobjects[linkIndex - 1]);
|
|
Actor *linkedActor = linkedControl->_actor;
|
|
uint32 notifyThreadId2 = _actor->_notifyThreadId2;
|
|
_actor->_linkIndex2 = linkIndex;
|
|
if (_actor->_entryTblPtr) {
|
|
linkedActor->_flags |= Illusions::ACTOR_FLAG_80;
|
|
linkedActor->_entryTblPtr = _actor->_entryTblPtr;
|
|
linkedActor->_notifyThreadId2 = _actor->_notifyThreadId2;
|
|
linkedActor->_seqCodeValue1 = _actor->_seqCodeValue1;
|
|
linkedActor->_seqCodeValue3 = _actor->_seqCodeValue3;
|
|
_actor->_flags &= ~Illusions::ACTOR_FLAG_80;
|
|
_actor->_entryTblPtr = 0;
|
|
_actor->_notifyThreadId1 = 0;
|
|
_actor->_notifyThreadId2 = 0;
|
|
}
|
|
if (notifyThreadId2) {
|
|
Thread *talkThread = _vm->_threads->findThread(notifyThreadId2);
|
|
talkThread->sendMessage(kMsgClearSequenceId2, 0);
|
|
}
|
|
}
|
|
|
|
void Control::startMoveActor(uint32 sequenceId, Common::Point destPt, uint32 callerThreadId1, uint32 callerThreadId2) {
|
|
|
|
PointArray *pathNode;
|
|
ActorType *actorType = _vm->_dict->findActorType(_actorTypeId);
|
|
|
|
_actor->_pathAngle = 0;
|
|
_actor->_pathFlag50 = false;
|
|
_actor->_seqCodeValue3 = 0;
|
|
_actor->_seqCodeValue1 = 0;
|
|
_actor->_pathInitialPosFlag = true;
|
|
|
|
uint newFacing;
|
|
if (_vm->calcPointDirection(_actor->_position, destPt, newFacing))
|
|
faceActor(newFacing);
|
|
|
|
if (actorType->_value1E)
|
|
_actor->_pathCtrY = actorType->_value1E;
|
|
else
|
|
_actor->_pathCtrY = 140;
|
|
|
|
pathNode = createPath(destPt);
|
|
|
|
if (pathNode->size() == 1 &&
|
|
_actor->_position.x == (*pathNode)[0].x &&
|
|
_actor->_position.y == (*pathNode)[0].y) {
|
|
delete pathNode;
|
|
_vm->notifyThreadId(callerThreadId2);
|
|
} else {
|
|
_actor->_posXShl = _actor->_position.x << 16;
|
|
_actor->_posYShl = _actor->_position.y << 16;
|
|
startSequenceActor(sequenceId, 1, 0);
|
|
_actor->_pathNode = pathNode;
|
|
_actor->_pathPointsCount = pathNode->size();
|
|
_actor->_pathPoints = pathNode->size();
|
|
_actor->_flags |= Illusions::ACTOR_FLAG_400;
|
|
_actor->_walkCallerThreadId1 = callerThreadId1;
|
|
_vm->notifyThreadId(_actor->_notifyId3C);
|
|
_actor->_notifyId3C = callerThreadId2;
|
|
_actor->_pathPointIndex = 0;
|
|
_vm->_input->discardEvent(kEventSkip);
|
|
}
|
|
|
|
}
|
|
|
|
PointArray *Control::createPath(Common::Point destPt) {
|
|
PointArray *walkPoints = (_actor->_flags & Illusions::ACTOR_FLAG_HAS_WALK_POINTS) ? _actor->_pathWalkPoints->_points : 0;
|
|
PathLines *walkRects = (_actor->_flags & Illusions::ACTOR_FLAG_HAS_WALK_RECTS) ? _actor->_pathWalkRects->_rects : 0;
|
|
PathFinder pathFinder;
|
|
WidthHeight bgDimensions = _vm->_backgroundInstances->getMasterBgDimensions();
|
|
PointArray *path = pathFinder.findPath(_vm->_camera, _actor->_position, destPt, walkPoints, walkRects, bgDimensions);
|
|
return path;
|
|
}
|
|
|
|
void Control::updateActorMovement(uint32 deltaTime) {
|
|
// TODO This needs some cleanup
|
|
|
|
static const int16 kAngleTbl[] = {60, 0, 120, 0, 60, 0, 120, 0};
|
|
bool fastWalked = false;
|
|
|
|
do {
|
|
|
|
if (!fastWalked && _vm->testMainActorFastWalk(this)) {
|
|
fastWalked = true;
|
|
disappearActor();
|
|
_actor->_flags |= Illusions::ACTOR_FLAG_8000;
|
|
_actor->_seqCodeIp = 0;
|
|
deltaTime = 2;
|
|
}
|
|
|
|
if (_vm->testMainActorCollision(this))
|
|
break;
|
|
|
|
Common::Point prevPt;
|
|
if (_actor->_pathPointIndex == 0) {
|
|
if (_actor->_pathInitialPosFlag) {
|
|
_actor->_pathCtrX = 0;
|
|
_actor->_pathInitialPos = _actor->_position;
|
|
_actor->_pathInitialPosFlag = false;
|
|
}
|
|
prevPt = _actor->_pathInitialPos;
|
|
} else {
|
|
prevPt = (*_actor->_pathNode)[_actor->_pathPointIndex - 1];
|
|
}
|
|
|
|
Common::Point currPt = (*_actor->_pathNode)[_actor->_pathPointIndex];
|
|
|
|
int16 deltaX = currPt.x - prevPt.x;
|
|
int16 deltaY = currPt.y - prevPt.y;
|
|
|
|
if (!_actor->_pathFlag50) {
|
|
|
|
FixedPoint16 angle;
|
|
if (currPt.x == prevPt.x) {
|
|
if (prevPt.y >= currPt.y)
|
|
angle = fixedMul(-0x5A0000, 0x478);
|
|
else
|
|
angle = fixedMul(0x5A0000, 0x478);
|
|
} else {
|
|
angle = fixedAtan(fixedDiv(deltaY << 16, deltaX << 16));
|
|
}
|
|
_actor->_pathAngle = angle;
|
|
|
|
int16 v13 = (fixedTrunc(fixedMul(angle, 0x394BB8)) + 360) % 360; // 0x394BB8 is 180 / pi
|
|
if (deltaX >= 0)
|
|
v13 += 180;
|
|
v13 = (v13 + 90) % 360;
|
|
int16 v15 = kAngleTbl[0] / -2;
|
|
uint newFacing = 1;
|
|
for (uint i = 0; i < 8; ++i) {
|
|
v15 += kAngleTbl[i];
|
|
if (v13 < v15) {
|
|
newFacing = 1 << i;
|
|
break;
|
|
}
|
|
}
|
|
if (newFacing != _actor->_facing) {
|
|
refreshSequenceCode();
|
|
faceActor(newFacing);
|
|
}
|
|
|
|
_actor->_pathFlag50 = true;
|
|
|
|
}
|
|
|
|
FixedPoint16 deltaX24, deltaY24;
|
|
|
|
if (_actor->_flags & Illusions::ACTOR_FLAG_400) {
|
|
|
|
FixedPoint16 v20 = fixedMul((deltaTime + _actor->_pathCtrX) << 16, _actor->_pathCtrY << 16);
|
|
FixedPoint16 v21 = fixedDiv(v20, 100 << 16);
|
|
FixedPoint16 v22 = fixedMul(v21, _actor->_scale << 16);
|
|
FixedPoint16 v23 = fixedDiv(v22, 100 << 16);
|
|
_actor->_seqCodeValue1 = 100 * _actor->_pathCtrY * deltaTime / 100;
|
|
if (v23) {
|
|
FixedPoint16 prevDistance = fixedDistance(prevPt.x << 16, prevPt.y << 16, _actor->_posXShl, _actor->_posYShl);
|
|
FixedPoint16 distance = prevDistance + v23;
|
|
if (prevPt.x > currPt.x)
|
|
distance = -distance;
|
|
deltaX24 = fixedMul(fixedCos(_actor->_pathAngle), distance);
|
|
deltaY24 = fixedMul(fixedSin(_actor->_pathAngle), distance);
|
|
} else {
|
|
deltaX24 = _actor->_posXShl - (prevPt.x << 16);
|
|
deltaY24 = _actor->_posYShl - (prevPt.y << 16);
|
|
}
|
|
} else {
|
|
if (100 * (int)deltaTime <= _actor->_seqCodeValue2)
|
|
break;
|
|
deltaX24 = deltaX << 16;
|
|
deltaY24 = deltaY << 16;
|
|
}
|
|
|
|
if (ABS(deltaX24) < ABS(deltaX << 16) ||
|
|
ABS(deltaY24) < ABS(deltaY << 16)) {
|
|
FixedPoint16 newX = (prevPt.x << 16) + deltaX24;
|
|
FixedPoint16 newY = (prevPt.y << 16) + deltaY24;
|
|
if (newX == _actor->_posXShl && newY == _actor->_posYShl) {
|
|
_actor->_pathCtrX += deltaTime;
|
|
} else {
|
|
_actor->_pathCtrX = 0;
|
|
_actor->_posXShl = newX;
|
|
_actor->_posYShl = newY;
|
|
_actor->_position.x = fixedTrunc(_actor->_posXShl);
|
|
_actor->_position.y = fixedTrunc(_actor->_posYShl);
|
|
}
|
|
} else {
|
|
_actor->_position = currPt;
|
|
_actor->_posXShl = _actor->_position.x << 16;
|
|
_actor->_posYShl = _actor->_position.y << 16;
|
|
--_actor->_pathPointsCount;
|
|
++_actor->_pathPointIndex;
|
|
++_actor->_pathPoints;
|
|
_actor->_pathInitialPosFlag = true;
|
|
if (_actor->_pathPointsCount == 0) {
|
|
if (_actor->_flags & Illusions::ACTOR_FLAG_400) {
|
|
delete _actor->_pathNode;
|
|
_actor->_flags &= ~Illusions::ACTOR_FLAG_400;
|
|
}
|
|
_actor->_pathNode = 0;
|
|
_actor->_pathPoints = 0;
|
|
_actor->_pathPointsCount = 0;
|
|
_actor->_pathPointIndex = 0;
|
|
if (_actor->_notifyId3C) {
|
|
_vm->notifyThreadId(_actor->_notifyId3C);
|
|
_actor->_walkCallerThreadId1 = 0;
|
|
}
|
|
fastWalked = false;
|
|
}
|
|
_actor->_pathFlag50 = false;
|
|
}
|
|
|
|
} while (fastWalked);
|
|
|
|
}
|
|
|
|
void Control::refreshSequenceCode() {
|
|
Sequence *sequence = _vm->_dict->findSequence(_actor->_sequenceId);
|
|
_actor->_seqCodeIp = sequence->_sequenceCode;
|
|
}
|
|
|
|
void Control::getActorFrameDimensions(WidthHeight &dimensions) {
|
|
dimensions._width = _actor->_surface->w;
|
|
dimensions._height = _actor->_surface->h;
|
|
}
|
|
|
|
void Control::drawActorRect(const Common::Rect r, byte color) {
|
|
_vm->_screen->fillSurfaceRect(_actor->_surface, r, color);
|
|
_actor->_flags |= Illusions::ACTOR_FLAG_4000;
|
|
}
|
|
|
|
void Control::fillActor(byte color) {
|
|
_vm->_screen->fillSurface(_actor->_surface, color);
|
|
_actor->_flags |= Illusions::ACTOR_FLAG_4000;
|
|
}
|
|
|
|
bool Control::isPixelCollision(Common::Point &pt) {
|
|
Frame *frame = &(*_actor->_frames)[_actor->_frameIndex - 1];
|
|
return _vm->_screen->isSpritePixelSolid(pt, _position, _actor->_position,
|
|
_actor->_surfInfo, _actor->_scale, frame->_flags, frame->_compressedPixels);
|
|
}
|
|
|
|
void Control::startSequenceActorIntern(uint32 sequenceId, int value, byte *entryTblPtr, uint32 notifyThreadId) {
|
|
stopActor();
|
|
|
|
_actor->_flags &= ~Illusions::ACTOR_FLAG_80;
|
|
_actor->_flags &= ~Illusions::ACTOR_FLAG_400;
|
|
_actor->_flags |= Illusions::ACTOR_FLAG_100;
|
|
|
|
sequenceId = _actor->_defaultSequences.use(sequenceId);
|
|
|
|
_actor->_sequenceId = sequenceId;
|
|
_actor->_notifyThreadId1 = notifyThreadId;
|
|
_actor->_notifyId3C = 0;
|
|
_actor->_walkCallerThreadId1 = 0;
|
|
_actor->_entryTblPtr = 0;
|
|
|
|
Sequence *sequence = _vm->_dict->findSequence(sequenceId);
|
|
|
|
if (!sequence && _vm->getGameId() == kGameIdDuckman) {
|
|
//debug(1, "Load external sequence...");
|
|
_vm->_resSys->loadResource(0x00060000 | (sequenceId & 0xFFFF), _vm->getCurrentScene(), 0);
|
|
sequence = _vm->_dict->findSequence(sequenceId);
|
|
_actor->_flags |= Illusions::ACTOR_FLAG_800;
|
|
}
|
|
|
|
_actor->_seqCodeIp = sequence->_sequenceCode;
|
|
_actor->_frames = _vm->findActorSequenceFrames(sequence);
|
|
|
|
_actor->_seqCodeValue3 = 0;
|
|
_actor->_seqCodeValue1 = 0;
|
|
|
|
if (_vm->getGameId() == kGameIdBBDOU) {
|
|
_actor->_seqCodeValue2 = value == 1 ? 350 : 600;
|
|
} else if (_vm->getGameId() == kGameIdDuckman) {
|
|
_actor->_seqCodeValue2 = value == 1 ? 350 : 750;
|
|
}
|
|
|
|
_actor->initSequenceStack();
|
|
|
|
if (_vm->getGameId() == kGameIdBBDOU)
|
|
stopSequenceActor();
|
|
|
|
_actor->_linkIndex2 = 0;
|
|
|
|
if (entryTblPtr) {
|
|
_actor->_flags |= Illusions::ACTOR_FLAG_80;
|
|
_actor->_entryTblPtr = entryTblPtr;
|
|
if (_vm->getGameId() == kGameIdBBDOU) {
|
|
_actor->_notifyThreadId1 = 0;
|
|
_actor->_notifyThreadId2 = notifyThreadId;
|
|
}
|
|
}
|
|
|
|
if (_vm->getGameId() == kGameIdBBDOU)
|
|
sequenceActor();
|
|
|
|
}
|
|
|
|
void Control::execSequenceOpcode(OpCall &opCall) {
|
|
_vm->_controls->_sequenceOpcodes->execOpcode(this, opCall);
|
|
}
|
|
|
|
// Controls
|
|
|
|
Controls::Controls(IllusionsEngine *vm)
|
|
: _vm(vm) {
|
|
_sequenceOpcodes = new SequenceOpcodes(_vm);
|
|
_nextTempObjectId = 0;
|
|
}
|
|
|
|
Controls::~Controls() {
|
|
delete _sequenceOpcodes;
|
|
destroyControls();
|
|
}
|
|
|
|
void Controls::placeBackgroundObject(BackgroundObject *backgroundObject) {
|
|
Control *control = newControl();
|
|
control->_objectId = backgroundObject->_objectId;
|
|
control->_flags = backgroundObject->_flags;
|
|
control->_priority = backgroundObject->_priority;
|
|
control->readPointsConfig(backgroundObject->_pointsConfig);
|
|
control->activateObject();
|
|
_controls.push_front(control);
|
|
_vm->_dict->setObjectControl(control->_objectId, control);
|
|
debug(0, "Added background control. objectId: %08X", control->_objectId);
|
|
}
|
|
|
|
void Controls::placeActor(uint32 actorTypeId, Common::Point placePt, uint32 sequenceId, uint32 objectId, uint32 notifyThreadId) {
|
|
Control *control = newControl();
|
|
Actor *actor = newActor();
|
|
ActorType *actorType = _vm->_dict->findActorType(actorTypeId);
|
|
|
|
control->_objectId = objectId;
|
|
control->_flags = actorType->_flags;
|
|
control->_priority = actorType->_priority;
|
|
control->readPointsConfig(actorType->_pointsConfig);
|
|
control->_actorTypeId = actorTypeId;
|
|
control->_actor = actor;
|
|
|
|
if (_vm->isCursorObject(actorTypeId, objectId))
|
|
_vm->setCursorControlRoutine(control);
|
|
|
|
if (actorType->_surfInfo._dimensions._width > 0 || actorType->_surfInfo._dimensions._height > 0) {
|
|
actor->createSurface(actorType->_surfInfo);
|
|
} else {
|
|
actor->_flags |= Illusions::ACTOR_FLAG_200;
|
|
}
|
|
|
|
actor->_position = placePt;
|
|
actor->_position2 = placePt;
|
|
Common::Point currPan = _vm->_camera->getCurrentPan();
|
|
if (!_vm->calcPointDirection(placePt, currPan, actor->_facing))
|
|
actor->_facing = 64;
|
|
actor->_scale = actorType->_scale;
|
|
actor->_namedPoints = &actorType->_namedPoints;
|
|
|
|
BackgroundResource *bgRes = _vm->_backgroundInstances->getActiveBgResource();
|
|
if (actorType->_pathWalkPointsIndex) {
|
|
actor->_pathWalkPoints = bgRes->getPathWalkPoints(actorType->_pathWalkPointsIndex - 1);
|
|
actor->_flags |= Illusions::ACTOR_FLAG_HAS_WALK_POINTS;
|
|
}
|
|
|
|
if (actorType->_scaleLayerIndex) {
|
|
actor->_scaleLayer = bgRes->getScaleLayer(actorType->_scaleLayerIndex - 1);
|
|
actor->_flags |= Illusions::ACTOR_FLAG_SCALED;
|
|
}
|
|
|
|
if (actorType->_pathWalkRectIndex) {
|
|
actor->_pathWalkRects = bgRes->getPathWalkRects(actorType->_pathWalkRectIndex - 1);
|
|
actor->_flags |= Illusions::ACTOR_FLAG_HAS_WALK_RECTS;
|
|
}
|
|
|
|
if (actorType->_priorityLayerIndex) {
|
|
actor->_priorityLayer = bgRes->getPriorityLayer(actorType->_priorityLayerIndex - 1);
|
|
actor->_flags |= Illusions::ACTOR_FLAG_PRIORITY;
|
|
}
|
|
|
|
if (actorType->_regionLayerIndex) {
|
|
actor->_regionLayer = bgRes->getRegionLayer(actorType->_regionLayerIndex - 1);
|
|
actor->_flags |= Illusions::ACTOR_FLAG_REGION;
|
|
}
|
|
|
|
actor->_pathCtrY = 140;
|
|
|
|
_controls.push_front(control);
|
|
_vm->_dict->setObjectControl(objectId, control);
|
|
|
|
if (_vm->getGameId() == kGameIdDuckman) {
|
|
control->appearActor();
|
|
} else if (_vm->getGameId() == kGameIdBBDOU) {
|
|
control->_flags |= 0x01;
|
|
actor->_flags |= Illusions::ACTOR_FLAG_1000;
|
|
}
|
|
|
|
if (_vm->isCursorObject(actorTypeId, objectId))
|
|
_vm->placeCursorControl(control, sequenceId);
|
|
|
|
// TODO HACK at least we should restrict this to the sequenceId
|
|
control->setActorIndex(1);
|
|
|
|
control->startSequenceActor(sequenceId, 2, notifyThreadId);
|
|
}
|
|
|
|
void Controls::placeSequenceLessActor(uint32 objectId, Common::Point placePt, WidthHeight dimensions, int16 priority) {
|
|
Control *control = newControl();
|
|
Actor *actor = newActor();
|
|
control->_flags = 0;
|
|
control->_priority = priority;
|
|
control->_objectId = objectId;
|
|
control->_bounds._topLeft.x = 0;
|
|
control->_bounds._topLeft.y = 0;
|
|
control->_bounds._bottomRight.x = dimensions._width - 1;
|
|
control->_bounds._bottomRight.y = dimensions._height - 1;
|
|
control->_feetPt.x = dimensions._width / 2;
|
|
control->_feetPt.y = dimensions._height / 2;
|
|
control->_position.x = 0;
|
|
control->_position.y = 0;
|
|
control->_actorTypeId = 0x50004;
|
|
control->_actor = actor;
|
|
actor->setControlRoutine(0);
|
|
actor->_surfInfo._pixelSize = dimensions._width * dimensions._height;
|
|
actor->_surfInfo._dimensions = dimensions;
|
|
actor->createSurface(actor->_surfInfo);
|
|
actor->_position = placePt;
|
|
actor->_position2 = placePt;
|
|
actor->_facing = 64;
|
|
actor->_scale = 100;
|
|
actor->_namedPoints = 0;
|
|
actor->_pathCtrY = 140;
|
|
|
|
_controls.push_front(control);
|
|
_vm->_dict->setObjectControl(objectId, control);
|
|
control->appearActor();
|
|
}
|
|
|
|
void Controls::placeActorLessObject(uint32 objectId, Common::Point feetPt, Common::Point pt, int16 priority, uint flags) {
|
|
Control *control = newControl();
|
|
control->_flags = flags;
|
|
control->_feetPt = feetPt;
|
|
control->_priority = priority;
|
|
control->_objectId = objectId;
|
|
control->_bounds._topLeft = feetPt;
|
|
control->_bounds._bottomRight = pt;
|
|
control->_position.x = 0;
|
|
control->_position.y = 0;
|
|
control->_actorTypeId = 0;
|
|
control->_actor = 0;
|
|
_controls.push_front(control);
|
|
_vm->_dict->setObjectControl(objectId, control);
|
|
}
|
|
|
|
void Controls::placeSubActor(uint32 objectId, int linkIndex, uint32 actorTypeId, uint32 sequenceId) {
|
|
Control *parentControl = _vm->_dict->getObjectControl(objectId);
|
|
uint32 tempObjectId = newTempObjectId();
|
|
placeActor(actorTypeId, Common::Point(0, 0), sequenceId, tempObjectId, 0);
|
|
parentControl->_actor->_subobjects[linkIndex - 1] = tempObjectId;
|
|
Actor *subActor = _vm->_dict->getObjectControl(tempObjectId)->_actor;
|
|
subActor->_flags |= Illusions::ACTOR_FLAG_40;
|
|
subActor->_parentObjectId = parentControl->_objectId;
|
|
subActor->_linkIndex = linkIndex;
|
|
}
|
|
|
|
void Controls::placeDialogItem(uint16 objectNum, uint32 actorTypeId, uint32 sequenceId, Common::Point placePt, int16 choiceJumpOffs) {
|
|
Control *control = newControl();
|
|
Actor *actor = newActor();
|
|
ActorType *actorType = _vm->_dict->findActorType(actorTypeId);
|
|
control->_flags = 0xC;
|
|
control->_priority = actorType->_priority;
|
|
control->_objectId = objectNum | 0x40000;
|
|
control->readPointsConfig(actorType->_pointsConfig);
|
|
control->_actorTypeId = actorTypeId;
|
|
control->_actor = actor;
|
|
actor->setControlRoutine(new Common::Functor2Mem<Control*, uint32, void, Controls>(this, &Controls::dialogItemControlRoutine));
|
|
actor->_choiceJumpOffs = choiceJumpOffs;
|
|
actor->createSurface(actorType->_surfInfo);
|
|
actor->_position = placePt;
|
|
actor->_position2 = placePt;
|
|
actor->_scale = actorType->_scale;
|
|
actor->_color = actorType->_color;
|
|
_controls.push_front(control);
|
|
control->appearActor();
|
|
control->startSequenceActor(sequenceId, 2, 0);
|
|
control->setActorIndex(1);
|
|
}
|
|
|
|
void Controls::destroyControls() {
|
|
ItemsIterator it = _controls.begin();
|
|
while (it != _controls.end()) {
|
|
destroyControlInternal(*it);
|
|
it = _controls.erase(it);
|
|
}
|
|
}
|
|
|
|
void Controls::destroyActiveControls() {
|
|
ItemsIterator it = _controls.begin();
|
|
while (it != _controls.end()) {
|
|
if ((*it)->_pauseCtr <= 0) {
|
|
destroyControlInternal(*it);
|
|
it = _controls.erase(it);
|
|
} else
|
|
++it;
|
|
}
|
|
}
|
|
|
|
void Controls::destroyControlsBySceneId(uint32 sceneId) {
|
|
ItemsIterator it = _controls.begin();
|
|
while (it != _controls.end()) {
|
|
if ((*it)->_sceneId == sceneId) {
|
|
destroyControlInternal(*it);
|
|
it = _controls.erase(it);
|
|
} else
|
|
++it;
|
|
}
|
|
}
|
|
|
|
void Controls::destroyDialogItems() {
|
|
ItemsIterator it = _controls.begin();
|
|
while (it != _controls.end()) {
|
|
if (((*it)->_pauseCtr == 0) && ((*it)->_flags & 4)) {
|
|
destroyControlInternal(*it);
|
|
it = _controls.erase(it);
|
|
} else
|
|
++it;
|
|
}
|
|
}
|
|
|
|
void Controls::threadIsDead(uint32 threadId) {
|
|
for (ItemsIterator it = _controls.begin(); it != _controls.end(); ++it) {
|
|
Control *control = *it;
|
|
if (control->_actor &&
|
|
(control->_actor->_notifyThreadId1 == threadId || control->_actor->_notifyId3C == threadId)) {
|
|
control->_actor->_notifyThreadId1 = 0;
|
|
control->_actor->_notifyId3C = 0;
|
|
}
|
|
}
|
|
}
|
|
|
|
void Controls::pauseControls() {
|
|
for (ItemsIterator it = _controls.begin(); it != _controls.end(); ++it) {
|
|
Control *control = *it;
|
|
++control->_pauseCtr;
|
|
if (control->_pauseCtr == 1)
|
|
control->pause();
|
|
}
|
|
}
|
|
|
|
void Controls::unpauseControls() {
|
|
for (ItemsIterator it = _controls.begin(); it != _controls.end(); ++it) {
|
|
Control *control = *it;
|
|
--control->_pauseCtr;
|
|
if (control->_pauseCtr == 0)
|
|
control->unpause();
|
|
}
|
|
}
|
|
|
|
void Controls::pauseControlsBySceneId(uint32 sceneId) {
|
|
for (ItemsIterator it = _controls.begin(); it != _controls.end(); ++it) {
|
|
Control *control = *it;
|
|
if (control->_sceneId == sceneId) {
|
|
++control->_pauseCtr;
|
|
if (control->_pauseCtr == 1)
|
|
control->pause();
|
|
}
|
|
}
|
|
}
|
|
|
|
void Controls::unpauseControlsBySceneId(uint32 sceneId) {
|
|
for (ItemsIterator it = _controls.begin(); it != _controls.end(); ++it) {
|
|
Control *control = *it;
|
|
if (control->_sceneId == sceneId) {
|
|
--control->_pauseCtr;
|
|
if (control->_pauseCtr == 0)
|
|
control->unpause();
|
|
}
|
|
}
|
|
}
|
|
|
|
bool Controls::getOverlappedObject(Control *control, Common::Point pt, Control **outOverlappedControl, int minPriority) {
|
|
Control *foundControl = 0;
|
|
uint32 foundPriority = 0;
|
|
uint32 minPriorityExt = _vm->getPriorityFromBase(minPriority);
|
|
|
|
for (ItemsIterator it = _controls.begin(); it != _controls.end(); ++it) {
|
|
Control *testControl = *it;
|
|
if (testControl != control && testControl->_pauseCtr == 0 &&
|
|
(testControl->_flags & 1) && !(testControl->_flags & 0x10) &&
|
|
(!testControl->_actor || (testControl->_actor->_flags & Illusions::ACTOR_FLAG_IS_VISIBLE))) {
|
|
Common::Rect collisionRect;
|
|
testControl->getCollisionRect(collisionRect);
|
|
if (!collisionRect.isEmpty() && collisionRect.contains(pt)) {
|
|
uint32 testPriority = testControl->getOverlapPriority();
|
|
if ((!foundControl || foundPriority < testPriority) &&
|
|
testPriority >= minPriorityExt) {
|
|
foundControl = testControl;
|
|
foundPriority = testPriority;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
if (foundControl) {
|
|
if (foundControl->_actor && foundControl->_actor->_parentObjectId && (foundControl->_actor->_flags & Illusions::ACTOR_FLAG_40)) {
|
|
uint32 parentObjectId = foundControl->getSubActorParent();
|
|
foundControl = _vm->_dict->getObjectControl(parentObjectId);
|
|
}
|
|
*outOverlappedControl = foundControl;
|
|
}
|
|
|
|
return foundControl != 0;
|
|
}
|
|
|
|
bool Controls::getOverlappedObjectAccurate(Control *control, Common::Point pt, Control **outOverlappedControl, int minPriority) {
|
|
Control *foundControl = 0;
|
|
uint32 foundPriority = 0;
|
|
uint32 minPriorityExt = _vm->getPriorityFromBase(minPriority);
|
|
|
|
for (ItemsIterator it = _controls.begin(); it != _controls.end(); ++it) {
|
|
Control *testControl = *it;
|
|
if (testControl != control && testControl->_pauseCtr == 0 &&
|
|
(testControl->_flags & 1) && !(testControl->_flags & 0x10) &&
|
|
(!testControl->_actor || (testControl->_actor->_flags & Illusions::ACTOR_FLAG_IS_VISIBLE))) {
|
|
Common::Rect collisionRect;
|
|
testControl->getCollisionRectAccurate(collisionRect);
|
|
if (!collisionRect.isEmpty() && collisionRect.contains(pt) &&
|
|
(!testControl->_actor || testControl->isPixelCollision(pt))) {
|
|
uint32 testPriority = testControl->getOverlapPriority();
|
|
if ((!foundControl || foundPriority < testPriority) &&
|
|
testPriority >= minPriorityExt) {
|
|
foundControl = testControl;
|
|
foundPriority = testPriority;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
if (foundControl) {
|
|
if (foundControl->_actor && foundControl->_actor->_parentObjectId && (foundControl->_actor->_flags & Illusions::ACTOR_FLAG_40)) {
|
|
uint32 parentObjectId = foundControl->getSubActorParent();
|
|
foundControl = _vm->_dict->getObjectControl(parentObjectId);
|
|
}
|
|
*outOverlappedControl = foundControl;
|
|
}
|
|
|
|
return foundControl != 0;
|
|
}
|
|
|
|
bool Controls::getDialogItemAtPos(Control *control, Common::Point pt, Control **outOverlappedControl) {
|
|
Control *foundControl = 0;
|
|
for (ItemsIterator it = _controls.begin(); it != _controls.end(); ++it) {
|
|
Control *testControl = *it;
|
|
if (testControl != control && testControl->_pauseCtr == 0 &&
|
|
(testControl->_flags & 1) && (testControl->_flags & 4)) {
|
|
Common::Rect collisionRect;
|
|
testControl->getCollisionRect(collisionRect);
|
|
if (!collisionRect.isEmpty() && collisionRect.contains(pt) &&
|
|
(!foundControl || foundControl->_priority < testControl->_priority))
|
|
foundControl = testControl;
|
|
}
|
|
}
|
|
*outOverlappedControl = foundControl;
|
|
return foundControl != 0;
|
|
}
|
|
|
|
bool Controls::getOverlappedWalkObject(Control *control, Common::Point pt, Control **outOverlappedControl) {
|
|
Control *foundControl = 0;
|
|
for (ItemsIterator it = _controls.begin(); it != _controls.end(); ++it) {
|
|
Control *testControl = *it;
|
|
if (testControl != control && testControl->_pauseCtr == 0 &&
|
|
(testControl->_flags & 1)) {
|
|
Common::Rect collisionRect;
|
|
testControl->getCollisionRect(collisionRect);
|
|
if (!collisionRect.isEmpty() && collisionRect.contains(pt) &&
|
|
(!foundControl || foundControl->_priority < testControl->_priority))
|
|
foundControl = testControl;
|
|
}
|
|
}
|
|
if (foundControl)
|
|
*outOverlappedControl = foundControl;
|
|
return foundControl != 0;
|
|
}
|
|
|
|
void Controls::destroyControl(Control *control) {
|
|
_controls.remove(control);
|
|
destroyControlInternal(control);
|
|
}
|
|
|
|
bool Controls::findNamedPoint(uint32 namedPointId, Common::Point &pt) {
|
|
for (ItemsIterator it = _controls.begin(); it != _controls.end(); ++it) {
|
|
Control *control = *it;
|
|
if (control->_pauseCtr == 0 && control->_actor && control->_actor->findNamedPoint(namedPointId, pt))
|
|
return true;
|
|
}
|
|
return false;
|
|
}
|
|
|
|
void Controls::actorControlRoutine(Control *control, uint32 deltaTime) {
|
|
|
|
Actor *actor = control->_actor;
|
|
|
|
if (actor->_pauseCtr > 0)
|
|
return;
|
|
|
|
if (control->_actor->_pathNode) {
|
|
control->updateActorMovement(deltaTime);
|
|
} else {
|
|
actor->_seqCodeValue1 = 100 * deltaTime;
|
|
}
|
|
|
|
if (actor->_flags & Illusions::ACTOR_FLAG_SCALED) {
|
|
int scale = actor->_scaleLayer->getScale(actor->_position);
|
|
control->setActorScale(scale);
|
|
}
|
|
|
|
if (actor->_flags & Illusions::ACTOR_FLAG_PRIORITY) {
|
|
int16 priority = actor->_priorityLayer->getPriority(actor->_position);
|
|
if (priority)
|
|
control->setPriority(priority + 1);
|
|
}
|
|
|
|
if (actor->_flags & Illusions::ACTOR_FLAG_REGION) {
|
|
// Update transition sequence
|
|
int regionIndex = actor->_regionLayer->getRegionIndex(actor->_position);
|
|
if (actor->_regionIndex != regionIndex) {
|
|
if (regionIndex) {
|
|
uint32 savedSequenceId = actor->_sequenceId;
|
|
byte *savedSeqCodeIp = actor->_seqCodeIp;
|
|
int savedSeqCodeValue1 = actor->_seqCodeValue1;
|
|
int savedSeqCodeValue3 = actor->_seqCodeValue3;
|
|
uint32 regionSequenceId = actor->_regionLayer->getRegionSequenceId(regionIndex);
|
|
//debug(1, "Running transition sequence %08X", regionSequenceId);
|
|
Sequence *sequence = _vm->_dict->findSequence(regionSequenceId);
|
|
actor->_sequenceId = regionSequenceId;
|
|
actor->_seqCodeIp = sequence->_sequenceCode;
|
|
actor->_seqCodeValue3 = 0;
|
|
control->sequenceActor();
|
|
actor->_sequenceId = savedSequenceId;
|
|
actor->_seqCodeIp = savedSeqCodeIp;
|
|
actor->_seqCodeValue3 = savedSeqCodeValue3;
|
|
actor->_seqCodeValue1 = savedSeqCodeValue1;
|
|
}
|
|
actor->_regionIndex = regionIndex;
|
|
}
|
|
}
|
|
|
|
}
|
|
|
|
void Controls::dialogItemControlRoutine(Control *control, uint32 deltaTime) {
|
|
Actor *actor = control->_actor;
|
|
if (actor->_pauseCtr <= 0)
|
|
actor->_seqCodeValue1 = 100 * deltaTime;
|
|
}
|
|
|
|
Actor *Controls::newActor() {
|
|
return new Actor(_vm);
|
|
}
|
|
|
|
Control *Controls::newControl() {
|
|
return new Control(_vm);
|
|
}
|
|
|
|
uint32 Controls::newTempObjectId() {
|
|
uint32 nextTempObjectId1 = _nextTempObjectId;
|
|
uint32 nextTempObjectId2 = _nextTempObjectId + 0x1000;
|
|
if (nextTempObjectId2 > 0xFFFF) {
|
|
nextTempObjectId1 = 0;
|
|
nextTempObjectId2 = 0x1000;
|
|
}
|
|
_nextTempObjectId = nextTempObjectId1 + 1;
|
|
return nextTempObjectId2 | 0x40000;
|
|
}
|
|
|
|
void Controls::destroyControlInternal(Control *control) {
|
|
|
|
if ((_vm->getGameId() == kGameIdBBDOU || !(control->_flags & 4)) && control->_pauseCtr <= 0)
|
|
_vm->_dict->setObjectControl(control->_objectId, 0);
|
|
|
|
if ((_vm->getGameId() == kGameIdBBDOU || !(control->_flags & 4)) && control->_objectId == Illusions::CURSOR_OBJECT_ID && control->_pauseCtr <= 0)
|
|
_vm->setCursorControl(0);
|
|
|
|
if (control->_actor) {
|
|
if (control->_actor->_pathNode && (control->_actor->_flags & Illusions::ACTOR_FLAG_400))
|
|
delete control->_actor->_pathNode;
|
|
if (!(control->_actor->_flags & Illusions::ACTOR_FLAG_200))
|
|
control->_actor->destroySurface();
|
|
delete control->_actor;
|
|
control->_actor = 0;
|
|
}
|
|
|
|
delete control;
|
|
}
|
|
|
|
void Controls::disappearActors() {
|
|
for (ItemsIterator it = _controls.begin(); it != _controls.end(); ++it) {
|
|
Control *control = *it;
|
|
if (control->_flags & 4 && control->_pauseCtr == 0) {
|
|
control->disappearActor();
|
|
}
|
|
}
|
|
Control *control = _vm->_dict->getObjectControl(0x40148);
|
|
if (control) {
|
|
control->disappearActor();
|
|
}
|
|
}
|
|
|
|
void Controls::appearActors() {
|
|
for (ItemsIterator it = _controls.begin(); it != _controls.end(); ++it) {
|
|
Control *control = *it;
|
|
if (control->_flags & 4 && control->_pauseCtr == 0) {
|
|
control->appearActor();
|
|
}
|
|
}
|
|
Control *control = _vm->_dict->getObjectControl(0x40148);
|
|
if (control) {
|
|
control->appearActor();
|
|
}
|
|
}
|
|
|
|
void Controls::pauseActors(uint32 objectId) {
|
|
for (ItemsIterator it = _controls.begin(); it != _controls.end(); ++it) {
|
|
Control *control = *it;
|
|
if (control->_actor && control->_objectId != objectId) {
|
|
control->_actor->pause();
|
|
}
|
|
}
|
|
}
|
|
|
|
void Controls::unpauseActors(uint32 objectId) {
|
|
for (ItemsIterator it = _controls.begin(); it != _controls.end(); ++it) {
|
|
Control *control = *it;
|
|
if (control->_actor && control->_objectId != objectId) {
|
|
control->_actor->unpause();
|
|
}
|
|
}
|
|
_vm->_unpauseControlActorFlag = true;
|
|
}
|
|
|
|
|
|
} // End of namespace Illusions
|