diff options
-rw-r--r-- | engines/illusions/bbdou/bbdou_specialcode.cpp | 217 | ||||
-rw-r--r-- | engines/illusions/bbdou/bbdou_specialcode.h | 6 |
2 files changed, 212 insertions, 11 deletions
diff --git a/engines/illusions/bbdou/bbdou_specialcode.cpp b/engines/illusions/bbdou/bbdou_specialcode.cpp index d7ba8ceb41..9278a41813 100644 --- a/engines/illusions/bbdou/bbdou_specialcode.cpp +++ b/engines/illusions/bbdou/bbdou_specialcode.cpp @@ -377,7 +377,7 @@ void BbdouSpecialCode::setCursorControlRoutine(uint32 objectId, int num) { new Common::Functor2Mem<Control*, uint32, void, BbdouSpecialCode>(this, &BbdouSpecialCode::cursorInteractControlRoutine)); else control->_actor->setControlRoutine( - new Common::Functor2Mem<Control*, uint32, void, BbdouSpecialCode>(this, &BbdouSpecialCode::cursorControlRoutine2)); + new Common::Functor2Mem<Control*, uint32, void, BbdouSpecialCode>(this, &BbdouSpecialCode::cursorCrosshairControlRoutine)); } Common::Point BbdouSpecialCode::getBackgroundCursorPos(Common::Point cursorPos) { @@ -605,7 +605,7 @@ void BbdouSpecialCode::cursorInteractControlRoutine(Control *cursorControl, uint } -void BbdouSpecialCode::cursorControlRoutine2(Control *cursorControl, uint32 deltaTime) { +void BbdouSpecialCode::cursorCrosshairControlRoutine(Control *cursorControl, uint32 deltaTime) { static const struct ShooterAnim { uint32 objectId; @@ -620,12 +620,14 @@ void BbdouSpecialCode::cursorControlRoutine2(Control *cursorControl, uint32 delt {0x60652, 0x60653, 0x60654, 0x60655, 0x60656, 0x60657, 0x60658, 0x60659}} }; + static const uint32 kShooterObjectIds[] = { + 0x401DF, 0x401E0, 0x401E1, 0x401E2, + 0x401E3, 0x401E4, 0x401E5, 0x401E6, + }; + Actor *actor = cursorControl->_actor; CursorData &cursorData = _cursor->_data; - // TODO - //debug("BbdouSpecialCode::cursorControlRoutine2()"); - if (cursorData._visibleCtr <= 0) { if (cursorData._currOverlappedObjectId || cursorData._mode == 3) { if (cursorData._mode == 3) @@ -636,11 +638,11 @@ void BbdouSpecialCode::cursorControlRoutine2(Control *cursorControl, uint32 delt return; } - Common::Point cursorPos = _vm->_input->getCursorPosition(); + Common::Point screenCursorPos = _vm->_input->getCursorPosition(); - if (cursorPos != actor->_position) { - actor->_position = cursorPos; - int16 gridX = 8 * cursorPos.x / 640; + if (screenCursorPos != actor->_position) { + actor->_position = screenCursorPos; + int16 gridX = 8 * screenCursorPos.x / 640; if (gridX >= 8) gridX = 4; @@ -658,7 +660,146 @@ void BbdouSpecialCode::cursorControlRoutine2(Control *cursorControl, uint32 delt } - // TODO A lot of stuff + Common::Point cursorPos = getBackgroundCursorPos(cursorPos); + bool foundOverlapped = false; + Control *overlappedControl = 0; + + if (cursorData._flags & 1) + foundOverlapped = false; + else { + /* TODO Implement getOverlappedObjectAccurate + foundOverlapped = _vm->_controls->getOverlappedObjectAccurate(cursorControl, cursorPos, + &overlappedControl, cursorData._item10._field58); + */ + } + + if (foundOverlapped) { + if (overlappedControl->_objectId != cursorData._currOverlappedObjectId) { + resetItem10(cursorControl->_objectId, &cursorData._item10); + int value = _cursor->findStruct8bsValue(overlappedControl->_objectId); + if (value == 2 || value == 3 || value == 4 || value == 5 || value == 6 || value == 7) { + if (cursorData._mode != 3) { + _cursor->saveInfo(); + cursorData._mode = 3; + cursorData._item10._verbId = 0x1B0006; + cursorData._holdingObjectId = 0; + } + switch (value) { + case 2: + cursorData._sequenceId = 0x60010; + break; + case 3: + cursorData._sequenceId = 0x60011; + break; + case 4: + cursorData._sequenceId = 0x60012; + break; + case 5: + cursorData._sequenceId = 0x60013; + break; + case 6: + cursorData._sequenceId = 0x60015; + break; + case 7: + cursorData._sequenceId = 0x60014; + break; + } + _cursor->show(cursorControl); + cursorData._currOverlappedObjectId = overlappedControl->_objectId; + } else { + if (cursorData._mode == 3) + _cursor->restoreInfo(); + _cursor->show(cursorControl); + cursorControl->setActorIndexTo2(); + if (overlappedControl->_objectId) { + cursorData._item10._verbId = 0x1B0003; + cursorData._currOverlappedObjectId = overlappedControl->_objectId; + } else { + cursorData._item10._verbId = 0x1B0002; + cursorData._currOverlappedObjectId = overlappedControl->_objectId; + } + } + } + } else { + if (cursorData._currOverlappedObjectId || cursorData._mode == 3) { + if (cursorData._mode == 3) + _cursor->restoreInfo(); + _cursor->show(cursorControl); + cursorControl->setActorIndexTo1(); + resetItem10(cursorControl->_objectId, &cursorData._item10); + } + cursorData._currOverlappedObjectId = 0; + } + + actor->_seqCodeValue1 = 100 * deltaTime; + + if (cursorData._currOverlappedObjectId) { + + if (_vm->_input->pollEvent(kEventLeftClick)) { + + uint32 outSceneId, outVerbId, outObjectId2, outObjectId; + bool success = getShooterCause(_vm->getCurrentScene(), + cursorData._item10._verbId, cursorData._holdingObjectId, cursorData._currOverlappedObjectId, + outSceneId, outVerbId, outObjectId2, outObjectId); + + uint index = (uint)_vm->getRandom(2); + const ShooterAnim &anim = kShooterAnims[index]; + uint32 objectId = anim.objectId; + int gridX = _shooterStatus[index].gridX; + Control *gunControl = _vm->getObjectControl(objectId); + if (gunControl) { + _shooterStatus[index].flag = true; + gunControl->startSequenceActor(anim.sequenceIds2[gridX], 2, 0); + } + Control *hitControl = _vm->getObjectControl(kShooterObjectIds[_shooterObjectIdIndex]); + if (hitControl) { + hitControl->setActorPosition(actor->_position); + hitControl->startSequenceActor(0x6068D, 2, 0); + } + ++_shooterObjectIdIndex; + if (_shooterObjectIdIndex >= ARRAYSIZE(kShooterObjectIds)) + _shooterObjectIdIndex = 0; + + if (success) { + _cursor->hide(cursorControl->_objectId); + uint32 threadId = startCauseThread(cursorControl->_objectId, _vm->getCurrentScene(), outVerbId, outObjectId2, outObjectId); + if (cursorData._field90) { + _vm->_threads->killThread(cursorData._causeThreadId2); + cursorData._field90 = 0; + } + cursorData._causeThreadId1 = _vm->causeTrigger(outSceneId, outVerbId, outObjectId2, outObjectId, threadId); + cursorData._causeThreadId2 = cursorData._causeThreadId1; + resetItem10(cursorControl->_objectId, &cursorData._item10); + cursorData._currOverlappedObjectId = 0; + cursorControl->setActorIndexTo1(); + } + + } else if (_vm->_input->pollEvent(kEventRightClick) && cursorData._item10._playSound48 == 1 && !cursorData._item10._flag56) { + // TODO I don't think this is used; _playSound48 seems to be always 0 here + debug("Cursor_sub_10004DD0 TODO"); + // TODO Cursor_sub_10004DD0(controla->objectId, cursorData->currOverlappedObjectId, cursorData->holdingObjectId, &cursorData->item10); + } + + } else if (_vm->_input->pollEvent(kEventLeftClick)) { + uint index = (uint)_vm->getRandom(2); + const ShooterAnim &anim = kShooterAnims[index]; + uint32 objectId = anim.objectId; + int gridX = _shooterStatus[index].gridX; + Control *gunControl = _vm->getObjectControl(objectId); + if (gunControl) { + _shooterStatus[index].flag = true; + gunControl->startSequenceActor(anim.sequenceIds2[gridX], 2, 0); + } + Control *hitControl = _vm->getObjectControl(kShooterObjectIds[_shooterObjectIdIndex]); + if (hitControl) { + hitControl->setActorPosition(actor->_position); + hitControl->startSequenceActor(0x6068D, 2, 0); + } + ++_shooterObjectIdIndex; + if (_shooterObjectIdIndex >= ARRAYSIZE(kShooterObjectIds)) + _shooterObjectIdIndex = 0; + + } } @@ -832,4 +973,60 @@ void BbdouSpecialCode::addSalad(uint32 sequenceId) { control->deactivateObject(); } +bool BbdouSpecialCode::getShooterCause(uint32 sceneId, uint32 verbId, uint32 objectId2, uint32 objectId, + uint32 &outSceneId, uint32 &outVerbId, uint32 &outObjectId2, uint32 &outObjectId) { + bool success = false; + objectId2 = verbId != 0x1B0003 ? 0 : objectId2; + if (_vm->causeIsDeclared(sceneId, verbId, objectId2, objectId)) { + outSceneId = sceneId; + outVerbId = verbId; + outObjectId2 = objectId2; + outObjectId = objectId; + success = true; + } else if (verbId == 0x1B0003 && _vm->causeIsDeclared(sceneId, 0x1B0008, 0, objectId)) { + outSceneId = sceneId; + outVerbId = 0x1B0003; + outObjectId2 = 0; + outObjectId = objectId; + success = true; + } else if (_vm->causeIsDeclared(sceneId, verbId, objectId2, 0x40001)) { + outSceneId = sceneId; + outVerbId = verbId; + outObjectId2 = objectId2; + outObjectId = 0x40001; + success = true; + } else if (verbId == 0x1B0003 && _vm->causeIsDeclared(sceneId, 0x1B0008, 0, 0x40001)) { + outSceneId = sceneId; + outVerbId = 0x1B0008; + outObjectId2 = 0; + outObjectId = 0x40001; + success = true; + } else if (_vm->causeIsDeclared(0x10003, verbId, objectId2, objectId)) { + outSceneId = 0x10003; + outVerbId = verbId; + outObjectId2 = objectId2; + outObjectId = objectId; + success = true; + } else if (verbId == 0x1B0003 && _vm->causeIsDeclared(0x10003, 0x1B0008, 0, objectId)) { + outSceneId = 0x10003; + outVerbId = verbId; + outObjectId2 = 0; + outObjectId = objectId; + success = true; + } else if (_vm->causeIsDeclared(0x10003, verbId, objectId2, 0x40001)) { + outSceneId = 0x10003; + outVerbId = verbId; + outObjectId2 = objectId2; + outObjectId = 0x40001; + success = true; + } else if (verbId == 0x1B0003 && _vm->causeIsDeclared(0x10003, 0x1B0008, 0, 0x40001)) { + outSceneId = 0x10003; + outVerbId = verbId; + outObjectId2 = 0; + outObjectId = 0x40001; + success = true; + } + return success; +} + } // End of namespace Illusions diff --git a/engines/illusions/bbdou/bbdou_specialcode.h b/engines/illusions/bbdou/bbdou_specialcode.h index 22fdd12cbc..0ee5fae4a1 100644 --- a/engines/illusions/bbdou/bbdou_specialcode.h +++ b/engines/illusions/bbdou/bbdou_specialcode.h @@ -108,6 +108,7 @@ public: // Shooter ShooterStatus _shooterStatus[2]; + uint _shooterObjectIdIndex; // Special code interface functions void spcInitCursor(OpCall &opCall); @@ -150,7 +151,7 @@ protected: Item10 *item10, uint32 progResKeywordId); bool findVerbId(Item10 *item10, uint32 currOverlappedObjectId, int always0, uint32 &outVerbId); void cursorInteractControlRoutine(Control *cursorControl, uint32 deltaTime); - void cursorControlRoutine2(Control *cursorControl, uint32 deltaTime); + void cursorCrosshairControlRoutine(Control *cursorControl, uint32 deltaTime); bool testVerbId(uint32 verbId, uint32 holdingObjectId, uint32 overlappedObjectId); bool getCause(uint32 sceneId, uint32 verbId, uint32 objectId2, uint32 objectId, uint32 &outVerbId, uint32 &outObjectId2, uint32 &outObjectId); @@ -160,6 +161,9 @@ protected: // Salad void initSalad(); void addSalad(uint32 sequenceId); + // Shooter + bool getShooterCause(uint32 sceneId, uint32 verbId, uint32 objectId2, uint32 objectId, + uint32 &outSceneId, uint32 &outVerbId, uint32 &outObjectId2, uint32 &outObjectId); }; } // End of namespace Illusions |