diff options
Diffstat (limited to 'engines/illusions/camera.cpp')
-rw-r--r-- | engines/illusions/camera.cpp | 306 |
1 files changed, 285 insertions, 21 deletions
diff --git a/engines/illusions/camera.cpp b/engines/illusions/camera.cpp index df5de4b083..d81c746274 100644 --- a/engines/illusions/camera.cpp +++ b/engines/illusions/camera.cpp @@ -23,6 +23,7 @@ #include "illusions/illusions.h" #include "illusions/camera.h" #include "illusions/backgroundresource.h" +#include "illusions/fixedpoint.h" #include "illusions/time.h" namespace Illusions { @@ -47,8 +48,8 @@ Camera::Camera(IllusionsEngine *vm) _activeState._panNotifyId = 0; _activeState._trackingLimits.x = 0; _activeState._trackingLimits.y = 0; - _activeState._pt.x = 320; - _activeState._pt.y = 240; + _activeState._centerPt.x = 320; + _activeState._centerPt.y = 240; _activeState._pointFlags = 0; } @@ -66,7 +67,7 @@ void Camera::set(Common::Point &panPoint, WidthHeight &dimensions) { _activeState._bounds._bottomRight.x = MAX(0, dimensions._width - 640) + 320; _activeState._bounds._bottomRight.y = MAX(0, dimensions._height - 480) + 240; _activeState._panTargetPoint = panPoint; - // TODO camera_clipPanTargetPoint(); + clipPanTargetPoint(); _activeState._currPan = _activeState._panTargetPoint; _activeState._panXShl = _activeState._currPan.x << 16; _activeState._panYShl = _activeState._currPan.y << 16; @@ -77,8 +78,98 @@ void Camera::set(Common::Point &panPoint, WidthHeight &dimensions) { _activeState._trackingLimits.x = 0; _activeState._trackingLimits.y = 0; _activeState._pointFlags = 0; - _activeState._pt.x = 320; - _activeState._pt.y = 240; + _activeState._centerPt.x = 320; + _activeState._centerPt.y = 240; +} + +void Camera::panCenterObject(uint32 objectId, int16 panSpeed) { + Common::Point *actorPosition = _vm->getObjectActorPositionPtr(objectId); + _activeState._cameraMode = 1; + _activeState._panSpeed = panSpeed; + _activeState._trackingLimits.x = 8; + _activeState._trackingLimits.y = 8; + _activeState._pointFlags = 0; + _activeState._panToPositionPtr = actorPosition; + _activeState._panObjectId = objectId; + _activeState._panTargetPoint = *actorPosition; + _activeState._panNotifyId = 0; + clipPanTargetPoint(); + _activeState._panStartTime = getCurrentTime(); + recalcPan(_activeState._panStartTime); +} + +void Camera::panTrackObject(uint32 objectId) { + Common::Point *actorPosition = _vm->getObjectActorPositionPtr(objectId); + _activeState._cameraMode = 3; + _activeState._panObjectId = objectId; + _activeState._trackingLimits.x = 160; + _activeState._trackingLimits.y = 120; + _activeState._pointFlags = 0; + _activeState._panSpeed = 710; + _activeState._panToPositionPtr = actorPosition; + _activeState._panNotifyId = 0; + _activeState._panTargetPoint = *actorPosition; + clipPanTargetPoint(); + _activeState._panStartTime = getCurrentTime(); + recalcPan(_activeState._panStartTime); +} + +void Camera::panToPoint(Common::Point pt, int16 panSpeed, uint32 panNotifyId) { + + if (_activeState._panNotifyId) { + // TODO scrmgrNotifyID(_activeState._panNotifyId); + _activeState._panNotifyId = 0; + } + + _activeState._panTargetPoint = Camera::getPtOffset(pt); + clipPanTargetPoint(); + + if (panSpeed) { + _activeState._cameraMode = 5; + _activeState._panSpeed = panSpeed; + _activeState._trackingLimits.x = 0; + _activeState._trackingLimits.y = 0; + _activeState._pointFlags = 0; + _activeState._panToPositionPtr = 0; + _activeState._panNotifyId = panNotifyId; + _activeState._panStartTime = getCurrentTime(); + recalcPan(_activeState._panStartTime); + } else { + _activeState._currPan = _activeState._panTargetPoint; + // TODO stopPan(); + if (panNotifyId) { + // TODO scrmgrNotifyID(panNotifyId); + } + } +} + +void Camera::panEdgeFollow(uint32 objectId, int16 panSpeed) { + Common::Point *actorPosition = _vm->getObjectActorPositionPtr(objectId); + _activeState._cameraMode = 2; + _activeState._trackingLimits.x = 318; + _activeState._trackingLimits.y = 238; + _activeState._pointFlags = 0; + _activeState._panSpeed = panSpeed; + _activeState._panToPositionPtr = actorPosition; + _activeState._panObjectId = objectId; + _activeState._panTargetPoint = _activeState._currPan; + _activeState._panNotifyId = 0; + clipPanTargetPoint(); + _activeState._panStartTime = getCurrentTime(); + recalcPan(_activeState._panStartTime); +} + +void Camera::stopPan() { + _activeState._cameraMode = 6; + _activeState._panTargetPoint = _activeState._currPan; + _activeState._panSpeed = 1; + _activeState._panXShl = _activeState._currPan.x << 16; + _activeState._panYShl = _activeState._currPan.y << 16; + _activeState._panToPositionPtr = 0; + _activeState._panObjectId = 0; + _activeState._panNotifyId = 0; + _activeState._pointFlags = 0; + _vm->_backgroundItems->refreshPan(); } void Camera::pause() { @@ -93,6 +184,65 @@ void Camera::unpause() { _activeState._panStartTime += pauseDuration; } +void Camera::pushCameraMode() { + CameraModeStackItem item; + item._cameraMode = _activeState._cameraMode; + item._panSpeed = _activeState._panSpeed; + item._panObjectId = 0; + item._panNotifyId = 0; + item._panTargetPoint.x = 0; + item._panTargetPoint.y = 0; + switch (_activeState._cameraMode) { + case 4: + item._cameraMode = 3; + item._panObjectId = _activeState._panObjectId; + break; + case 1: + case 2: + case 3: + item._panObjectId = _activeState._panObjectId; + break; + case 5: + item._panTargetPoint = _activeState._panTargetPoint; + item._panNotifyId = _activeState._panNotifyId; + break; + } + _stack.push(item); +} + +void Camera::popCameraMode() { + CameraModeStackItem item = _stack.pop(); + + if (item._panObjectId && !_vm->getObjectActorPositionPtr(item._panObjectId)) { + // Tracking object doesn't exist any longer + stopPan(); + return; + } + + switch (item._cameraMode) { + case 1: + panCenterObject(item._panObjectId, item._panSpeed); + break; + case 2: + panEdgeFollow(item._panObjectId, item._panSpeed); + break; + case 3: + panTrackObject(item._panObjectId); + break; + case 5: + panToPoint(item._panTargetPoint, item._panSpeed, item._panNotifyId); + break; + case 6: + stopPan(); + break; + } + +} + +void Camera::clearCameraModeStack() { + _stack.clear(); +} + void Camera::update(uint32 currTime) { if (_activeState._paused) @@ -148,7 +298,7 @@ void Camera::setBoundsToDimensions(WidthHeight &dimensions) { _activeState._bounds._topLeft.y = 240; _activeState._bounds._bottomRight.x = MAX(0, dimensions._width - 640) + 320; _activeState._bounds._bottomRight.y = MAX(0, dimensions._height - 480) + 240; - // TODO camera_clipPanTargetPoint(); + clipPanTargetPoint(); } Common::Point Camera::getCurrentPan() { @@ -164,43 +314,106 @@ Common::Point Camera::getScreenOffset() { void Camera::updateMode1(uint32 currTime) { Common::Point ptOffs = getPtOffset(*_activeState._panToPositionPtr); - int deltaX = ptOffs.x - _activeState._currPan.x + 320 - _activeState._pt.x; - int deltaY = ptOffs.y - _activeState._currPan.y + 240 - _activeState._pt.y; + int deltaX = ptOffs.x - _activeState._currPan.x + 320 - _activeState._centerPt.x; + int deltaY = ptOffs.y - _activeState._currPan.y + 240 - _activeState._centerPt.y; int deltaXAbs = ABS(deltaX); int deltaYAbs = ABS(deltaY); - if (deltaXAbs > _activeState._trackingLimits.x) { + if (deltaXAbs > _activeState._trackingLimits.x) _activeState._panTargetPoint.x = _activeState._currPan.x + ABS(deltaXAbs - _activeState._trackingLimits.x) * (deltaX >= 0 ? 1 : -1); - } else { + else _activeState._panTargetPoint.x = _activeState._currPan.x; - } - if (deltaYAbs > _activeState._trackingLimits.y) { + if (deltaYAbs > _activeState._trackingLimits.y) _activeState._panTargetPoint.y = _activeState._currPan.y + ABS(deltaYAbs - _activeState._trackingLimits.y) * (deltaY >= 0 ? 1 : -1); - } else { + else _activeState._panTargetPoint.y = _activeState._currPan.y; - } - // TODO Camera_clipPanTargetPoint(); + clipPanTargetPoint(); if (!isPanFinished()) { uint32 oldPanTime = _activeState._panStartTime; _activeState._panStartTime = _activeState._time28; - // TODO Camera_recalcPan(oldPanTime); + recalcPan(oldPanTime); } } void Camera::updateMode2(uint32 currTime) { - // TODO + Common::Point panToPosition = *_activeState._panToPositionPtr; + uint pointFlags = 0; + WRect rect; + + rect._topLeft.x = 320 - _activeState._trackingLimits.x; + rect._topLeft.y = 240 - _activeState._trackingLimits.y; + rect._bottomRight.x = 320 + _activeState._trackingLimits.x; + rect._bottomRight.y = 240 + _activeState._trackingLimits.y; + + if (calcPointFlags(panToPosition, rect, pointFlags)) { + if (pointFlags != _activeState._pointFlags) { + _activeState._pointFlags = pointFlags; + if (pointFlags & 1) + _activeState._panTargetPoint.x = _activeState._bounds._topLeft.x; + else if (pointFlags & 2) + _activeState._panTargetPoint.x = _activeState._bounds._bottomRight.x; + else + _activeState._panTargetPoint.x = _activeState._currPan.x; + if (pointFlags & 4) + _activeState._panTargetPoint.y = _activeState._bounds._topLeft.y; + else if (pointFlags & 8) + _activeState._panTargetPoint.y = _activeState._bounds._bottomRight.y; + else + _activeState._panTargetPoint.y = _activeState._currPan.y; + clipPanTargetPoint(); + _activeState._panStartTime = currTime; + recalcPan(currTime); + } + } else if (_activeState._pointFlags) { + _activeState._pointFlags = 0; + _activeState._panTargetPoint = _activeState._currPan; + } + } void Camera::updateMode3(uint32 currTime) { - // TODO + Common::Point panToPosition = *_activeState._panToPositionPtr; + int deltaX = panToPosition.x - _activeState._currPan.x; + int deltaY = panToPosition.y - _activeState._currPan.y; + + if (ABS(deltaX) > _activeState._trackingLimits.x) + _activeState._panTargetPoint.x = _activeState._currPan.x + 2 * _activeState._trackingLimits.x * (deltaX >= 0 ? 1 : -1); + else + _activeState._panTargetPoint.x = _activeState._currPan.x; + + if (ABS(deltaY) > _activeState._trackingLimits.y) + _activeState._panTargetPoint.y = _activeState._currPan.y + 2 * _activeState._trackingLimits.y * (deltaY >= 0 ? 1 : -1); + else + _activeState._panTargetPoint.y = _activeState._currPan.y; + + clipPanTargetPoint(); + + if (!isPanFinished()) { + _activeState._panStartTime = currTime; + recalcPan(currTime); + _activeState._cameraMode = 4; + } + } bool Camera::updatePan(uint32 currTime) { - // TODO + if (currTime - _activeState._time28 >= _activeState._time2E) { + _activeState._panXShl = _activeState._panTargetPoint.x << 16; + _activeState._panYShl = _activeState._panTargetPoint.y << 16; + } else { + _activeState._panXShl += fixedMul(_activeState._someX, (currTime - _activeState._panStartTime) << 16); + _activeState._panYShl += fixedMul(_activeState._someY, (currTime - _activeState._panStartTime) << 16); + } + _activeState._panStartTime = currTime; + Common::Point newPan(_activeState._panXShl >> 16, _activeState._panYShl >> 16); + if (_activeState._currPan.x != newPan.x || _activeState._currPan.y != newPan.y) { + _activeState._currPan = newPan; + return true; + } return false; } @@ -209,9 +422,60 @@ bool Camera::isPanFinished() { } Common::Point Camera::getPtOffset(Common::Point pt) { - pt.x = pt.x - _activeState._pt.x + 320; - pt.y = pt.y - _activeState._pt.y + 240; + pt.x = pt.x - _activeState._centerPt.x + 320; + pt.y = pt.y - _activeState._centerPt.y + 240; return pt; } +void Camera::recalcPan(uint32 currTime) { + _activeState._currPan2 = getCurrentPan(); + _activeState._time28 = currTime; + + if (_activeState._panSpeed == 0) { + _activeState._time2E = 0; + } else { + FP16 x1 = _activeState._currPan2.x << 16; + FP16 y1 = _activeState._currPan2.y << 16; + FP16 x2 = _activeState._panTargetPoint.x << 16; + FP16 y2 = _activeState._panTargetPoint.y << 16; + FP16 distance = fixedDistance(x1, y1, x2, y2); + _activeState._time2E = 60 * fixedTrunc(distance) / _activeState._panSpeed; + } + + if (_activeState._time2E != 0) { + _activeState._someX = fixedDiv((_activeState._panTargetPoint.x - _activeState._currPan2.x) << 16, _activeState._time2E << 16); + _activeState._someY = fixedDiv((_activeState._panTargetPoint.y - _activeState._currPan2.y) << 16, _activeState._time2E << 16); + } else { + _activeState._someX = (_activeState._panTargetPoint.x - _activeState._currPan2.x) << 16; + _activeState._someY = (_activeState._panTargetPoint.y - _activeState._currPan2.y) << 16; + } + +} + +bool Camera::calcPointFlags(Common::Point &pt, WRect &rect, uint &outFlags) { + bool result = false; + if (pt.x < rect._topLeft.x) { + outFlags |= 1; + result = true; + } else if (pt.x > rect._bottomRight.x) { + outFlags |= 2; + result = true; + } + if (pt.y < rect._topLeft.y) { + outFlags |= 4; + result = true; + } else if (pt.y > rect._bottomRight.y) { + outFlags |= 8; + result = true; + } + return result; +} + +void Camera::clipPanTargetPoint() { + _activeState._panTargetPoint.x = CLIP(_activeState._panTargetPoint.x, + _activeState._bounds._topLeft.x, _activeState._bounds._bottomRight.x); + _activeState._panTargetPoint.y = CLIP(_activeState._panTargetPoint.y, + _activeState._bounds._topLeft.y, _activeState._bounds._bottomRight.y); +} + } // End of namespace Illusions |