2013-06-16 12:11:18 +00:00
|
|
|
/* 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 "fullpipe/fullpipe.h"
|
|
|
|
|
|
|
|
#include "common/file.h"
|
|
|
|
#include "common/array.h"
|
|
|
|
#include "common/list.h"
|
|
|
|
|
|
|
|
#include "fullpipe/objects.h"
|
2013-09-28 11:15:46 +00:00
|
|
|
#include "fullpipe/statics.h"
|
2013-06-16 12:11:18 +00:00
|
|
|
#include "fullpipe/motion.h"
|
2013-09-17 20:00:33 +00:00
|
|
|
#include "fullpipe/messages.h"
|
2013-09-18 18:30:06 +00:00
|
|
|
#include "fullpipe/gameloader.h"
|
2013-06-16 12:11:18 +00:00
|
|
|
|
|
|
|
namespace Fullpipe {
|
|
|
|
|
2013-09-18 15:04:36 +00:00
|
|
|
bool MotionController::load(MfcArchive &file) {
|
2013-06-16 12:11:18 +00:00
|
|
|
// Is originally empty file.readClass();
|
|
|
|
|
2013-09-18 15:04:36 +00:00
|
|
|
debug(5, "MotionController::load()");
|
2013-06-16 12:11:18 +00:00
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2013-09-18 15:04:36 +00:00
|
|
|
bool MctlCompound::load(MfcArchive &file) {
|
|
|
|
debug(5, "MctlCompound::load()");
|
2013-07-12 06:03:02 +00:00
|
|
|
|
2013-06-16 12:11:18 +00:00
|
|
|
int count = file.readUint32LE();
|
|
|
|
|
2013-09-18 15:04:36 +00:00
|
|
|
debug(6, "MctlCompound::count = %d", count);
|
2013-06-16 12:11:18 +00:00
|
|
|
|
|
|
|
for (int i = 0; i < count; i++) {
|
2013-07-06 19:56:11 +00:00
|
|
|
debug(6, "CompoundArray[%d]", i);
|
2013-09-28 08:21:13 +00:00
|
|
|
MctlCompoundArrayItem *obj = new MctlCompoundArrayItem();
|
|
|
|
|
|
|
|
obj->_motionControllerObj = (MotionController *)file.readClass();
|
2013-06-16 12:11:18 +00:00
|
|
|
|
2013-07-06 19:56:11 +00:00
|
|
|
int count1 = file.readUint32LE();
|
2013-06-18 21:07:28 +00:00
|
|
|
|
2013-07-06 19:56:11 +00:00
|
|
|
debug(6, "ConnectionPoint::count: %d", count1);
|
|
|
|
for (int j = 0; j < count1; j++) {
|
|
|
|
debug(6, "ConnectionPoint[%d]", j);
|
2013-09-18 15:04:36 +00:00
|
|
|
MctlConnectionPoint *obj1 = (MctlConnectionPoint *)file.readClass();
|
2013-06-18 21:07:28 +00:00
|
|
|
|
2013-10-04 20:14:09 +00:00
|
|
|
obj->_connectionPoints.push_back(obj1);
|
2013-07-06 19:56:11 +00:00
|
|
|
}
|
2013-06-18 21:07:28 +00:00
|
|
|
|
2013-07-06 19:56:11 +00:00
|
|
|
obj->_field_20 = file.readUint32LE();
|
|
|
|
obj->_field_24 = file.readUint32LE();
|
2013-06-18 21:07:28 +00:00
|
|
|
|
2013-07-06 19:56:11 +00:00
|
|
|
debug(6, "graphReact");
|
2013-09-18 15:04:36 +00:00
|
|
|
obj->_movGraphReactObj = (MovGraphReact *)file.readClass();
|
2013-06-18 21:07:28 +00:00
|
|
|
|
2013-09-27 19:16:15 +00:00
|
|
|
_motionControllers.push_back(obj);
|
2013-06-16 12:11:18 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2013-09-28 10:06:06 +00:00
|
|
|
void MctlCompound::addObject(StaticANIObject *obj) {
|
|
|
|
for (uint i = 0; i < _motionControllers.size(); i++)
|
|
|
|
_motionControllers[i]->_motionControllerObj->addObject(obj);
|
2013-09-24 20:24:33 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
int MctlCompound::removeObject(StaticANIObject *obj) {
|
|
|
|
warning("STUB: MctlCompound::removeObject()");
|
|
|
|
|
|
|
|
return 0;
|
2013-07-22 15:46:03 +00:00
|
|
|
}
|
|
|
|
|
2013-09-18 15:04:36 +00:00
|
|
|
void MctlCompound::initMovGraph2() {
|
2013-09-25 21:07:31 +00:00
|
|
|
if (_objtype != kObjTypeMctlCompound)
|
|
|
|
return;
|
|
|
|
|
|
|
|
for (uint i = 0; i < _motionControllers.size(); i++) {
|
|
|
|
if (_motionControllers[i]->_motionControllerObj->_objtype != kObjTypeMovGraph)
|
|
|
|
continue;
|
|
|
|
|
|
|
|
MovGraph *gr = (MovGraph *)_motionControllers[i]->_motionControllerObj;
|
|
|
|
|
2013-09-27 19:16:15 +00:00
|
|
|
MovGraph2 *newgr = new MovGraph2();
|
2013-09-25 21:07:31 +00:00
|
|
|
|
2013-09-27 19:16:15 +00:00
|
|
|
newgr->_links = gr->_links;
|
|
|
|
newgr->_nodes = gr->_nodes;
|
2013-09-25 21:07:31 +00:00
|
|
|
|
|
|
|
gr->_links.clear();
|
|
|
|
gr->_nodes.clear();
|
|
|
|
|
|
|
|
delete gr;
|
|
|
|
|
|
|
|
_motionControllers[i]->_motionControllerObj = newgr;
|
|
|
|
}
|
2013-07-24 21:34:15 +00:00
|
|
|
}
|
|
|
|
|
2013-09-24 20:24:33 +00:00
|
|
|
void MctlCompound::freeItems() {
|
2013-10-30 20:24:37 +00:00
|
|
|
for (uint i = 0; i < _motionControllers.size(); i++)
|
|
|
|
_motionControllers[i]->_motionControllerObj->freeItems();
|
2013-09-24 20:24:33 +00:00
|
|
|
}
|
|
|
|
|
2013-10-15 06:10:57 +00:00
|
|
|
MessageQueue *MctlCompound::method34(StaticANIObject *subj, int xpos, int ypos, int fuzzyMatch, int staticsId) {
|
2013-09-18 15:04:36 +00:00
|
|
|
warning("STUB: MctlCompound::method34()");
|
2013-09-08 20:53:02 +00:00
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2013-10-16 21:37:04 +00:00
|
|
|
MessageQueue *MctlCompound::doWalkTo(StaticANIObject *subj, int xpos, int ypos, int fuzzyMatch, int staticsId) {
|
2013-10-03 22:25:22 +00:00
|
|
|
int match1 = -1;
|
|
|
|
int match2 = -1;
|
|
|
|
|
|
|
|
if (!subj)
|
|
|
|
return 0;
|
|
|
|
|
|
|
|
for (uint i = 0; i < _motionControllers.size(); i++) {
|
|
|
|
if (_motionControllers[i]->_movGraphReactObj) {
|
|
|
|
if (_motionControllers[i]->_movGraphReactObj->pointInRegion(subj->_ox, subj->_oy)) {
|
|
|
|
match1 = i;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (match1 == -1)
|
|
|
|
return 0;
|
|
|
|
|
|
|
|
for (uint i = 0; i < _motionControllers.size(); i++) {
|
|
|
|
if (_motionControllers[i]->_movGraphReactObj) {
|
|
|
|
if (_motionControllers[i]->_movGraphReactObj->pointInRegion(xpos, ypos)) {
|
|
|
|
match2 = i;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (match2 == -1)
|
|
|
|
return 0;
|
|
|
|
|
|
|
|
if (match1 == match2)
|
2013-10-16 21:37:04 +00:00
|
|
|
return _motionControllers[match1]->_motionControllerObj->doWalkTo(subj, xpos, ypos, fuzzyMatch, staticsId);
|
2013-10-03 22:25:22 +00:00
|
|
|
|
|
|
|
MctlConnectionPoint *closestP = findClosestConnectionPoint(subj->_ox, subj->_oy, match1, xpos, ypos, match2, &match2);
|
|
|
|
|
|
|
|
if (!closestP)
|
|
|
|
return 0;
|
|
|
|
|
2013-10-16 21:37:04 +00:00
|
|
|
MessageQueue *mq = _motionControllers[match1]->_motionControllerObj->doWalkTo(subj, closestP->_connectionX, closestP->_connectionY, 1, closestP->_field_14);
|
2013-10-03 22:25:22 +00:00
|
|
|
|
|
|
|
ExCommand *ex;
|
|
|
|
|
2013-10-04 20:14:09 +00:00
|
|
|
if (mq) {
|
2013-10-03 22:25:22 +00:00
|
|
|
for (uint i = 0; i < closestP->_messageQueueObj->getCount(); i++) {
|
|
|
|
ex = new ExCommand(closestP->_messageQueueObj->getExCommandByIndex(i));
|
2013-10-04 20:14:09 +00:00
|
|
|
ex->_excFlags |= 2;
|
|
|
|
mq->_exCommands.push_back(ex);
|
2013-10-03 22:25:22 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
ex = new ExCommand(subj->_id, 51, 0, xpos, ypos, 0, 1, 0, 0, 0);
|
|
|
|
|
2013-10-15 06:10:57 +00:00
|
|
|
ex->_field_20 = fuzzyMatch;
|
2013-10-03 22:25:22 +00:00
|
|
|
ex->_keyCode = subj->_okeyCode;
|
|
|
|
ex->_excFlags |= 2;
|
|
|
|
|
2013-10-04 20:14:09 +00:00
|
|
|
mq->_exCommands.push_back(ex);
|
2013-10-03 22:25:22 +00:00
|
|
|
}
|
|
|
|
|
2013-10-04 20:14:09 +00:00
|
|
|
return mq;
|
|
|
|
}
|
|
|
|
|
|
|
|
MctlConnectionPoint *MctlCompound::findClosestConnectionPoint(int ox, int oy, int destIndex, int connectionX, int connectionY, int sourceIndex, int *minDistancePtr) {
|
|
|
|
warning("STUB: MctlCompound::findClosestConnectionPoint()");
|
|
|
|
|
|
|
|
return 0;
|
2013-09-08 20:53:02 +00:00
|
|
|
}
|
|
|
|
|
2013-09-18 15:04:36 +00:00
|
|
|
bool MctlCompoundArray::load(MfcArchive &file) {
|
|
|
|
debug(5, "MctlCompoundArray::load()");
|
2013-07-12 06:03:02 +00:00
|
|
|
|
2013-06-16 12:11:18 +00:00
|
|
|
int count = file.readUint32LE();
|
|
|
|
|
2013-09-18 15:04:36 +00:00
|
|
|
debug(0, "MctlCompoundArray::count = %d", count);
|
2013-06-16 12:11:18 +00:00
|
|
|
|
2013-06-18 21:27:17 +00:00
|
|
|
assert(0);
|
|
|
|
|
2013-06-16 12:11:18 +00:00
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2013-10-01 19:56:50 +00:00
|
|
|
MovGraphItem::MovGraphItem() {
|
|
|
|
ani = 0;
|
|
|
|
field_4 = 0;
|
|
|
|
field_8 = 0;
|
|
|
|
field_C = 0;
|
|
|
|
field_10 = 0;
|
|
|
|
field_14 = 0;
|
|
|
|
field_18 = 0;
|
|
|
|
field_1C = 0;
|
|
|
|
field_20 = 0;
|
|
|
|
field_24 = 0;
|
|
|
|
items = 0;
|
|
|
|
count = 0;
|
|
|
|
field_30 = 0;
|
|
|
|
field_34 = 0;
|
|
|
|
field_38 = 0;
|
|
|
|
field_3C = 0;
|
|
|
|
}
|
|
|
|
|
2013-09-18 15:04:36 +00:00
|
|
|
int MovGraph_messageHandler(ExCommand *cmd);
|
2013-09-17 20:00:33 +00:00
|
|
|
|
2013-09-18 18:19:37 +00:00
|
|
|
int MovGraphCallback(int a1, int a2, int a3) {
|
|
|
|
warning("STUB: MovgraphCallback");
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2013-09-18 15:04:36 +00:00
|
|
|
MovGraph::MovGraph() {
|
2013-09-18 18:19:37 +00:00
|
|
|
_callback1 = MovGraphCallback;
|
2013-07-06 19:56:11 +00:00
|
|
|
_field_44 = 0;
|
2013-09-18 15:04:36 +00:00
|
|
|
insertMessageHandler(MovGraph_messageHandler, getMessageHandlersCount() - 1, 129);
|
2013-09-17 20:00:33 +00:00
|
|
|
|
|
|
|
_objtype = kObjTypeMovGraph;
|
2013-06-16 12:11:18 +00:00
|
|
|
}
|
|
|
|
|
2013-09-18 15:04:36 +00:00
|
|
|
bool MovGraph::load(MfcArchive &file) {
|
|
|
|
debug(5, "MovGraph::load()");
|
2013-07-12 06:03:02 +00:00
|
|
|
|
2013-07-06 19:56:11 +00:00
|
|
|
_links.load(file);
|
|
|
|
_nodes.load(file);
|
2013-06-16 12:11:18 +00:00
|
|
|
|
2013-07-06 19:56:11 +00:00
|
|
|
return true;
|
2013-06-16 12:11:18 +00:00
|
|
|
}
|
|
|
|
|
2013-09-28 10:06:06 +00:00
|
|
|
void MovGraph::addObject(StaticANIObject *obj) {
|
2013-10-01 19:56:50 +00:00
|
|
|
_mgm.clear();
|
|
|
|
_mgm.addItem(obj->_id);
|
|
|
|
|
|
|
|
for (uint i = 0; i < _items.size(); i++)
|
|
|
|
if (_items[i]->ani == obj)
|
|
|
|
return;
|
|
|
|
|
|
|
|
MovGraphItem *item = new MovGraphItem();
|
|
|
|
|
|
|
|
item->ani = obj;
|
|
|
|
|
|
|
|
_items.push_back(item);
|
|
|
|
|
|
|
|
_mgm.addItem(obj->_id); // FIXME: Is it really needed?
|
2013-07-22 15:46:03 +00:00
|
|
|
}
|
|
|
|
|
2013-09-27 19:16:15 +00:00
|
|
|
int MovGraph::removeObject(StaticANIObject *obj) {
|
|
|
|
warning("STUB: MovGraph::removeObject()");
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
void MovGraph::freeItems() {
|
|
|
|
warning("STUB: MovGraph::freeItems()");
|
|
|
|
}
|
|
|
|
|
|
|
|
int MovGraph::method28() {
|
|
|
|
warning("STUB: MovGraph::method28()");
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
int MovGraph::method2C() {
|
|
|
|
warning("STUB: MovGraph::method2C()");
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2013-10-15 06:10:57 +00:00
|
|
|
MessageQueue *MovGraph::method34(StaticANIObject *subj, int xpos, int ypos, int fuzzyMatch, int staticsId) {
|
2013-09-27 19:16:15 +00:00
|
|
|
warning("STUB: MovGraph::method34()");
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
int MovGraph::changeCallback() {
|
|
|
|
warning("STUB: MovGraph::changeCallback()");
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
int MovGraph::method3C() {
|
|
|
|
warning("STUB: MovGraph::method3C()");
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
int MovGraph::method44() {
|
|
|
|
warning("STUB: MovGraph::method44()");
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2013-10-16 21:37:04 +00:00
|
|
|
MessageQueue *MovGraph::doWalkTo(StaticANIObject *subj, int xpos, int ypos, int fuzzyMatch, int staticsId) {
|
|
|
|
warning("STUB: MovGraph::doWalkTo()");
|
2013-09-27 19:16:15 +00:00
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
int MovGraph::method50() {
|
|
|
|
warning("STUB: MovGraph::method50()");
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2013-10-15 06:10:57 +00:00
|
|
|
double MovGraph::calcDistance(Common::Point *point, MovGraphLink *link, int fuzzyMatch) {
|
2013-09-29 20:08:45 +00:00
|
|
|
int n1x = link->_movGraphNode1->_x;
|
|
|
|
int n1y = link->_movGraphNode1->_y;
|
|
|
|
int n2x = link->_movGraphNode2->_x;
|
|
|
|
int n2y = link->_movGraphNode2->_y;
|
|
|
|
double dist1x = (double)(point->x - n1x);
|
|
|
|
double dist1y = (double)(n1y - point->y);
|
|
|
|
double dist2x = (double)(n2x - n1x);
|
|
|
|
double dist2y = (double)(n2y - n1y);
|
|
|
|
double dist1 = sqrt(dist1y * dist1y + dist1x * dist1x);
|
|
|
|
double dist2 = ((double)(n1y - n2y) * dist1y + dist2x * dist1x) / link->_distance / dist1;
|
|
|
|
double distm = dist2 * dist1;
|
|
|
|
double res = sqrt(1.0 - dist2 * dist2) * dist1;
|
|
|
|
|
|
|
|
if (dist2 <= 0.0 || distm >= link->_distance) {
|
2013-10-15 06:10:57 +00:00
|
|
|
if (fuzzyMatch) {
|
2013-09-29 20:08:45 +00:00
|
|
|
if (dist2 > 0.0) {
|
|
|
|
if (distm >= link->_distance) {
|
|
|
|
point->x = n2x;
|
|
|
|
point->y = n2y;
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
point->x = n1x;
|
|
|
|
point->y = n1y;
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
return -1.0;
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
point->x = n1x + (dist2x * distm / link->_distance);
|
|
|
|
point->y = n1y + (dist2y * distm / link->_distance);
|
|
|
|
}
|
2013-09-17 20:00:33 +00:00
|
|
|
|
2013-09-29 20:08:45 +00:00
|
|
|
return res;
|
2013-09-17 20:00:33 +00:00
|
|
|
}
|
|
|
|
|
2013-10-22 21:49:55 +00:00
|
|
|
void MovGraph::calcNodeDistancesAndAngles() {
|
|
|
|
for (ObList::iterator i = _links.begin(); i != _links.end(); ++i) {
|
|
|
|
assert(((CObject *)*i)->_objtype == kObjTypeMovGraphLink);
|
|
|
|
|
|
|
|
MovGraphLink *lnk = (MovGraphLink *)*i;
|
|
|
|
|
|
|
|
lnk->_flags &= 0x7FFFFFFF;
|
|
|
|
|
|
|
|
lnk->calcNodeDistanceAndAngle();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2013-09-28 11:15:46 +00:00
|
|
|
int MovGraph2::getItemIndexByGameObjectId(int objectId) {
|
|
|
|
for (uint i = 0; i < _items.size(); i++)
|
|
|
|
if (_items[i]->_objectId == objectId)
|
|
|
|
return i;
|
|
|
|
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
2013-10-14 21:48:49 +00:00
|
|
|
int MovGraph2::getItemSubIndexByStaticsId(int idx, int staticsId) {
|
|
|
|
for (int i = 0; i < 4; i++)
|
|
|
|
if (_items[idx]->_subItems[i]._staticsId1 == staticsId || _items[idx]->_subItems[i]._staticsId2 == staticsId)
|
|
|
|
return i;
|
|
|
|
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
int MovGraph2::getItemSubIndexByMovementId(int idx, int movId) {
|
|
|
|
for (int i = 0; i < 4; i++)
|
|
|
|
if (_items[idx]->_subItems[i]._walk[0]._movementId == movId || _items[idx]->_subItems[i]._turn[0]._movementId == movId ||
|
|
|
|
_items[idx]->_subItems[i]._turnS[0]._movementId == movId)
|
|
|
|
return i;
|
|
|
|
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
int MovGraph2::getItemSubIndexByMGM(int idx, StaticANIObject *ani) {
|
|
|
|
warning("STUB: MovGraph2::getItemSubIndexByMGM()");
|
|
|
|
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
2013-09-28 11:15:46 +00:00
|
|
|
bool MovGraph2::initDirections(StaticANIObject *obj, MovGraph2Item *item) {
|
2013-09-28 12:59:46 +00:00
|
|
|
item->_obj = obj;
|
|
|
|
item->_objectId = obj->_id;
|
|
|
|
|
|
|
|
GameVar *var = g_fullpipe->getGameLoaderGameVar()->getSubVarByName(obj->_objectName);
|
|
|
|
if (!var)
|
|
|
|
return false;
|
|
|
|
|
|
|
|
var = var->getSubVarByName("Test_walk");
|
|
|
|
|
|
|
|
if (!var)
|
|
|
|
return false;
|
|
|
|
|
|
|
|
GameVar *varD = 0;
|
|
|
|
Common::Point point;
|
|
|
|
|
|
|
|
for (int dir = 0; dir < 4; dir++) {
|
|
|
|
switch (dir) {
|
|
|
|
case 0:
|
|
|
|
varD = var->getSubVarByName("Right");
|
|
|
|
break;
|
|
|
|
case 1:
|
|
|
|
varD = var->getSubVarByName("Left");
|
|
|
|
break;
|
|
|
|
case 2:
|
|
|
|
varD = var->getSubVarByName("Up");
|
|
|
|
break;
|
|
|
|
case 3:
|
|
|
|
varD = var->getSubVarByName("Down");
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!varD)
|
|
|
|
return false;
|
|
|
|
|
|
|
|
for (int act = 0; act < 3; act++) {
|
2013-10-05 18:32:16 +00:00
|
|
|
int idx = 0;
|
2013-09-28 12:59:46 +00:00
|
|
|
|
|
|
|
switch(act) {
|
|
|
|
case 0:
|
|
|
|
idx = varD->getSubVarAsInt("Start");
|
|
|
|
break;
|
|
|
|
case 1:
|
|
|
|
idx = varD->getSubVarAsInt("Go");
|
|
|
|
break;
|
|
|
|
case 2:
|
|
|
|
idx = varD->getSubVarAsInt("Stop");
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
item->_subItems[dir]._walk[act]._movementId = idx;
|
|
|
|
|
|
|
|
Movement *mov = obj->getMovementById(idx);
|
|
|
|
|
|
|
|
item->_subItems[dir]._walk[act]._mov = mov;
|
|
|
|
if (mov) {
|
|
|
|
mov->calcSomeXY(point, 0);
|
|
|
|
item->_subItems[dir]._walk[act]._mx = point.x;
|
|
|
|
item->_subItems[dir]._walk[act]._my = point.y;
|
|
|
|
}
|
|
|
|
}
|
2013-09-28 11:15:46 +00:00
|
|
|
|
2013-09-28 12:59:46 +00:00
|
|
|
for (int act = 0; act < 4; act++) {
|
2013-10-05 18:32:16 +00:00
|
|
|
int idx = 0;
|
2013-09-28 12:59:46 +00:00
|
|
|
|
|
|
|
switch(act) {
|
|
|
|
case 0:
|
|
|
|
idx = varD->getSubVarAsInt("TurnR");
|
|
|
|
break;
|
|
|
|
case 1:
|
|
|
|
idx = varD->getSubVarAsInt("TurnL");
|
|
|
|
break;
|
|
|
|
case 2:
|
|
|
|
idx = varD->getSubVarAsInt("TurnU");
|
|
|
|
break;
|
|
|
|
case 3:
|
|
|
|
idx = varD->getSubVarAsInt("TurnD");
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
item->_subItems[dir]._turn[act]._movementId = idx;
|
|
|
|
|
|
|
|
Movement *mov = obj->getMovementById(idx);
|
|
|
|
|
|
|
|
item->_subItems[dir]._turn[act]._mov = mov;
|
|
|
|
if (mov) {
|
|
|
|
mov->calcSomeXY(point, 0);
|
|
|
|
item->_subItems[dir]._turn[act]._mx = point.x;
|
|
|
|
item->_subItems[dir]._turn[act]._my = point.y;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
for (int act = 0; act < 4; act++) {
|
2013-10-05 18:32:16 +00:00
|
|
|
int idx = 0;
|
2013-09-28 12:59:46 +00:00
|
|
|
|
|
|
|
switch(act) {
|
|
|
|
case 0:
|
|
|
|
idx = varD->getSubVarAsInt("TurnSR");
|
|
|
|
break;
|
|
|
|
case 1:
|
|
|
|
idx = varD->getSubVarAsInt("TurnSL");
|
|
|
|
break;
|
|
|
|
case 2:
|
|
|
|
idx = varD->getSubVarAsInt("TurnSU");
|
|
|
|
break;
|
|
|
|
case 3:
|
|
|
|
idx = varD->getSubVarAsInt("TurnSD");
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
item->_subItems[dir]._turnS[act]._movementId = idx;
|
|
|
|
|
|
|
|
Movement *mov = obj->getMovementById(idx);
|
|
|
|
|
|
|
|
item->_subItems[dir]._turnS[act]._mov = mov;
|
|
|
|
if (mov) {
|
|
|
|
mov->calcSomeXY(point, 0);
|
|
|
|
item->_subItems[dir]._turnS[act]._mx = point.x;
|
|
|
|
item->_subItems[dir]._turnS[act]._my = point.y;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
item->_subItems[dir]._staticsId1 = item->_subItems[dir]._walk[0]._mov->_staticsObj1->_staticsId;
|
|
|
|
item->_subItems[dir]._staticsId2 = item->_subItems[dir]._walk[0]._mov->_staticsObj2->_staticsId;
|
|
|
|
|
|
|
|
}
|
|
|
|
return true;
|
2013-09-28 11:15:46 +00:00
|
|
|
}
|
|
|
|
|
2013-09-28 10:06:06 +00:00
|
|
|
void MovGraph2::addObject(StaticANIObject *obj) {
|
2013-09-28 11:15:46 +00:00
|
|
|
MovGraph::addObject(obj);
|
|
|
|
|
|
|
|
int id = getItemIndexByGameObjectId(obj->_id);
|
|
|
|
|
|
|
|
if (id >= 0) {
|
|
|
|
_items[id]->_obj = obj;
|
|
|
|
} else {
|
|
|
|
MovGraph2Item *item = new MovGraph2Item;
|
|
|
|
|
|
|
|
if (initDirections(obj, item)) {
|
|
|
|
_items.push_back(item);
|
|
|
|
} else {
|
|
|
|
delete item;
|
|
|
|
}
|
|
|
|
}
|
2013-09-27 19:16:15 +00:00
|
|
|
}
|
|
|
|
|
2013-10-16 21:35:02 +00:00
|
|
|
void MovGraph2::buildMovInfo1SubItems(MovInfo1 *movinfo, Common::Array<MovGraphLink *> *linkList, LinkInfo *lnkSrc, LinkInfo *lnkDst) {
|
2013-10-19 12:54:17 +00:00
|
|
|
MovInfo1Sub *elem;
|
|
|
|
Common::Point point;
|
|
|
|
Common::Rect rect;
|
|
|
|
|
|
|
|
int subIndex = movinfo->subIndex;
|
|
|
|
|
|
|
|
movinfo->items.clear();
|
|
|
|
|
|
|
|
elem = new MovInfo1Sub;
|
|
|
|
elem->subIndex = subIndex;
|
|
|
|
elem->x = movinfo->pt1.x;
|
|
|
|
elem->y = movinfo->pt1.y;
|
|
|
|
elem->distance = -1;
|
|
|
|
|
|
|
|
movinfo->items.push_back(elem);
|
|
|
|
|
|
|
|
int prevSubIndex = movinfo->subIndex;
|
|
|
|
|
2013-10-19 14:30:57 +00:00
|
|
|
for (uint i = 0; i < linkList->size(); i++) {
|
2013-10-19 12:54:17 +00:00
|
|
|
int idx1;
|
|
|
|
|
|
|
|
if (linkList->size() <= 1) {
|
|
|
|
if (linkList->size() == 1)
|
|
|
|
idx1 = getShortSide((*linkList)[0], movinfo->pt2.x - movinfo->pt1.x, movinfo->pt2.y - movinfo->pt1.y);
|
|
|
|
else
|
|
|
|
idx1 = getShortSide(0, movinfo->pt2.x - movinfo->pt1.x, movinfo->pt2.y - movinfo->pt1.y);
|
|
|
|
|
|
|
|
point.y = -1;
|
|
|
|
rect.bottom = -1;
|
|
|
|
rect.right = -1;
|
|
|
|
rect.top = -1;
|
|
|
|
rect.left = -1;
|
|
|
|
} else {
|
2013-10-19 14:30:57 +00:00
|
|
|
idx1 = findLink(linkList, i, &rect, &point);
|
2013-10-19 12:54:17 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
if (idx1 != prevSubIndex) {
|
|
|
|
prevSubIndex = idx1;
|
|
|
|
subIndex = idx1;
|
|
|
|
|
|
|
|
elem = new MovInfo1Sub;
|
|
|
|
elem->subIndex = subIndex;
|
|
|
|
elem->x = rect.left;
|
|
|
|
elem->y = rect.top;
|
|
|
|
elem->distance = -1;
|
|
|
|
|
|
|
|
movinfo->items.push_back(elem);
|
|
|
|
}
|
|
|
|
|
2013-10-19 19:45:30 +00:00
|
|
|
if (i != linkList->size() - 1) {
|
2013-10-19 12:54:17 +00:00
|
|
|
while (1) {
|
2013-10-19 14:30:57 +00:00
|
|
|
i++;
|
|
|
|
if (findLink(linkList, i, &rect, 0) != prevSubIndex) {
|
|
|
|
i--;
|
|
|
|
findLink(linkList, i, &rect, &point);
|
2013-10-19 12:54:17 +00:00
|
|
|
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
2013-10-19 19:45:30 +00:00
|
|
|
if (i == linkList->size() - 1)
|
2013-10-19 12:54:17 +00:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (movinfo->items.back()->subIndex != 10) {
|
|
|
|
subIndex = prevSubIndex;
|
|
|
|
|
|
|
|
elem = new MovInfo1Sub;
|
|
|
|
elem->subIndex = 10;
|
|
|
|
elem->x = -1;
|
|
|
|
elem->y = -1;
|
|
|
|
elem->distance = -1;
|
|
|
|
|
|
|
|
movinfo->items.push_back(elem);
|
|
|
|
|
2013-10-19 14:30:57 +00:00
|
|
|
if (i == linkList->size()) {
|
2013-10-19 12:54:17 +00:00
|
|
|
elem = new MovInfo1Sub;
|
|
|
|
elem->subIndex = prevSubIndex;
|
|
|
|
elem->x = movinfo->pt2.x;
|
|
|
|
elem->y = movinfo->pt2.y;
|
|
|
|
elem->distance = movinfo->distance2;
|
|
|
|
|
|
|
|
movinfo->items.push_back(elem);
|
|
|
|
} else {
|
|
|
|
elem = new MovInfo1Sub;
|
|
|
|
elem->subIndex = prevSubIndex;
|
|
|
|
elem->x = rect.right;
|
|
|
|
elem->y = rect.bottom;
|
|
|
|
elem->distance = point.y;
|
|
|
|
|
|
|
|
movinfo->items.push_back(elem);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (subIndex != movinfo->item1Index) {
|
|
|
|
elem = new MovInfo1Sub;
|
|
|
|
elem->subIndex = movinfo->item1Index;
|
|
|
|
elem->x = movinfo->pt2.x;
|
|
|
|
elem->y = movinfo->pt2.y;
|
|
|
|
elem->distance = movinfo->distance2;
|
|
|
|
|
|
|
|
movinfo->items.push_back(elem);
|
|
|
|
}
|
|
|
|
|
|
|
|
movinfo->itemsCount = movinfo->items.size();
|
2013-10-16 21:35:02 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
MessageQueue *MovGraph2::buildMovInfo1MessageQueue(MovInfo1 *movInfo) {
|
2013-11-07 20:47:48 +00:00
|
|
|
#if 0
|
|
|
|
MovInfo1 movinfo;
|
|
|
|
|
|
|
|
memcpy(movinfo, movInfo, sizeof(movinfo));
|
|
|
|
|
|
|
|
curX = movInfo->pt1.x;
|
|
|
|
curY = movInfo->pt1.y;
|
|
|
|
curDistance = movInfo->distance1;
|
|
|
|
|
|
|
|
mq = new MessageQueue(g_fullpipe->globalMessageQueueList->compact());
|
|
|
|
|
|
|
|
for (int i = 0; i < movInfo->_itemsCount - 1; i++) {
|
|
|
|
v9 = (MovInfo1Sub *)movInfo->items;
|
|
|
|
v10 = v9[i + 1].subIndex;
|
|
|
|
|
|
|
|
if (v10 != 10) {
|
|
|
|
if (v40 >= movInfo->_itemsCount - 2 || v9[i + 2].subIndex != 10) {
|
|
|
|
v16 = v9[i].subIndex;
|
|
|
|
v17 = (char *)this->items[1] + 16 * (v10 + 8);
|
|
|
|
subidx = 93 * movInfo->field_0;
|
|
|
|
movinfo.flags = 0;
|
|
|
|
v14 = 8 * subidx;
|
|
|
|
v15 = (MovGraph2Item *)(&v17[184 * v16] + v14);
|
|
|
|
} else {
|
|
|
|
v11 = v9[i].subIndex;
|
|
|
|
v12 = (char *)this->items[1] + 16 * (v10 + 4);
|
|
|
|
v13 = 93 * movInfo->field_0;
|
|
|
|
movinfo.flags = 2;
|
|
|
|
v14 = 8 * v13;
|
|
|
|
v15 = (MovGraph2Item *)(&v12[184 * v11] + v14);
|
|
|
|
}
|
|
|
|
if (v40 < movInfo->_itemsCount - 2
|
|
|
|
|| (v19 = v9[i + 1].x, v20 = (char *)&v9[i].x, v47 = (int *)v20, v21 = *(_DWORD *)v20, v21 == v19)
|
|
|
|
&& v9[i].y == v9[i + 1].y
|
|
|
|
|| v21 == -1
|
|
|
|
|| v9[i].y == -1
|
|
|
|
|| v19 == -1
|
|
|
|
|| v9[i + 1].y == -1) {
|
|
|
|
|
2013-11-08 23:45:31 +00:00
|
|
|
ExCommand *ex = new ExCommand(_items[1][movInfo->field_0].objectId, 1, v15->objectId, 0, 0, 0, 1, 0, 0, 0);
|
2013-11-07 20:47:48 +00:00
|
|
|
|
2013-11-08 23:45:31 +00:00
|
|
|
ex->_excFlags |= 2u;
|
|
|
|
ex->_keyCode = _items[1][movInfo->field_0].obj->GameObject.okeyCode;
|
|
|
|
ex->_field_24 = 1;
|
|
|
|
ex->_field_14 = -1;
|
|
|
|
mq->_exCommands.push_back(ex);
|
2013-11-07 20:47:48 +00:00
|
|
|
|
|
|
|
curX += v15->subItems[0].staticsId2;
|
|
|
|
curY += v15->subItems[0].staticsId1;
|
|
|
|
} else {
|
|
|
|
memset(mgminfo, 0, sizeof(mgminfo));
|
|
|
|
|
|
|
|
HIWORD(v22) = 0a;
|
|
|
|
v23 = v15->obj;
|
|
|
|
mgminfo.ani = *(StaticANIObject **)((char *)&this->items[1]->obj + v14);
|
|
|
|
LOWORD(v22) = *(_WORD *)(v23->callback1 + 132);
|
|
|
|
mgminfo.staticsId2 = v22;
|
|
|
|
mgminfo.x1 = v9[i + 1].x;
|
|
|
|
mgminfo.y1 = v9[i + 1].y;
|
|
|
|
mgminfo.field_1C = v9[i + 1].field_C;
|
|
|
|
mgminfo.staticsId1 = *(_WORD *)(v23->initialCounter + 132);
|
|
|
|
mgminfo.x2 = *v47;
|
|
|
|
v24 = v15->objectId;
|
|
|
|
mgminfo.y2 = v9[i].y;
|
|
|
|
mgminfo.field_10 = 1;
|
|
|
|
mgminfo.flags = 127;
|
|
|
|
mgminfo.movementId = v24;
|
|
|
|
|
|
|
|
v25 = (MessageQueue *)MGM_sub_445330((MGM *)&this->movGraph.mgm, &mgminfo);
|
|
|
|
MessageQueue_transferExCommands(mq, v25);
|
|
|
|
|
|
|
|
if (v25)
|
|
|
|
(*(void (__thiscall **)(MessageQueue *, signed int))(v25->CObject.vmt + 4))(v25, 1);
|
|
|
|
|
|
|
|
v26 = (MovInfo1Sub *)movInfo->items;
|
|
|
|
curX = v26[i + 1].x;
|
|
|
|
curY = v26[i + 1].y;
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
movinfo.item1Index = v9[i].subIndex;
|
|
|
|
movinfo.subIndex = movinfo.item1Index;
|
|
|
|
movinfo.pt1.y = curY;
|
|
|
|
movinfo.pt1.x = curX;
|
|
|
|
movinfo.distance1 = curDistance;
|
|
|
|
v29 = v9[i + 2].x;
|
|
|
|
movinfo.pt2.x = v9[i + 2].x;
|
|
|
|
movinfo.pt2.y = v9[i + 2].y;
|
|
|
|
movinfo.distance2 = v9[i + 2].field_C;
|
|
|
|
|
|
|
|
if (v40 >= movInfo->_itemsCount - 4
|
|
|
|
|| (v30 = v9[i + 2].subIndex, v30 == 10)
|
|
|
|
|| (v31 = v9[i + 3].subIndex, v31 == 10)
|
|
|
|
|| v30 == v31
|
|
|
|
|| v9[i + 4].subIndex != 10) {
|
|
|
|
if (v40 >= movInfo->itemsCount - 3
|
|
|
|
|| (v33 = v9[i + 2].subIndex, v33 == 10)
|
|
|
|
|| (v34 = v9[i + 3].subIndex, v34 == 10)
|
|
|
|
|| v33 == v34) {
|
|
|
|
movinfo.flags = movinfo.flags & 2 | movInfo->flags & 1;
|
|
|
|
} else {
|
|
|
|
v35 = (MovInfo1 *)((char *)&this->items[1][movInfo->field_0] + 184 * v33 + 16 * (v34 + 8));
|
|
|
|
movinfo.pt2.x = v29 - v35->pt1.y;
|
|
|
|
movinfo.pt2.y -= v35->pt2.x;
|
|
|
|
movinfo.flags = movinfo.flags & 2 | movInfo->flags & 1;
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
v32 = (MovInfo1 *)((char *)&this->items[1][movInfo->field_0] + 184 * v30 + 16 * (v31 + 4));
|
|
|
|
|
|
|
|
if (movinfo.item1Index && movinfo.item1Index != 1) {
|
|
|
|
movinfo.pt2.y -= v32->pt2.x;
|
|
|
|
movinfo.flags = movinfo.flags & 2 | 1;
|
|
|
|
} else {
|
|
|
|
movinfo.pt2.x = v29 - v32->pt1.y;
|
|
|
|
movinfo.flags = movinfo.flags & 2 | 1;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
i++;
|
|
|
|
|
|
|
|
v36 = MovGraph2_sub_454CD0(this, &movinfo);
|
|
|
|
|
|
|
|
if (!v36) {
|
|
|
|
delete mq;
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
MessageQueue_transferExCommands(mq, v36);
|
|
|
|
|
|
|
|
delete v36;
|
|
|
|
|
|
|
|
curX = movinfo.pt2.x;
|
|
|
|
curY = movinfo.pt2.y;
|
|
|
|
curDistance = movinfo.distance2;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
movInfo->pt2.x = movinfo.pt2.x;
|
|
|
|
movInfo->pt2.y = movinfo.pt2.y;
|
|
|
|
|
|
|
|
return mq;
|
|
|
|
|
|
|
|
#endif
|
2013-10-16 21:35:02 +00:00
|
|
|
warning("STUB: MovGraph2::buildMovInfo1MessageQueue()");
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2013-09-27 19:16:15 +00:00
|
|
|
int MovGraph2::removeObject(StaticANIObject *obj) {
|
|
|
|
warning("STUB: MovGraph2::removeObject()");
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
void MovGraph2::freeItems() {
|
|
|
|
warning("STUB: MovGraph2::freeItems()");
|
|
|
|
}
|
|
|
|
|
2013-10-25 21:20:58 +00:00
|
|
|
MessageQueue *MovGraph2::method34(StaticANIObject *ani, int xpos, int ypos, int fuzzyMatch, int staticsId) {
|
|
|
|
if (!ani->isIdle())
|
|
|
|
return 0;
|
2013-09-27 19:16:15 +00:00
|
|
|
|
2013-10-25 21:20:58 +00:00
|
|
|
if (ani->_flags & 0x100)
|
|
|
|
return 0;
|
|
|
|
|
|
|
|
MessageQueue *mq = doWalkTo(ani, xpos, ypos, fuzzyMatch, staticsId);
|
|
|
|
|
|
|
|
if (!mq)
|
|
|
|
return 0;
|
|
|
|
|
|
|
|
if (ani->_movement) {
|
|
|
|
if (mq->getCount() <= 1 || mq->getExCommandByIndex(0)->_messageKind != 22) {
|
|
|
|
PicAniInfo picAniInfo;
|
|
|
|
|
|
|
|
ani->getPicAniInfo(&picAniInfo);
|
|
|
|
ani->updateStepPos();
|
|
|
|
MessageQueue *mq1 = doWalkTo(ani, xpos, ypos, fuzzyMatch, staticsId);
|
|
|
|
|
|
|
|
ani->setPicAniInfo(&picAniInfo);
|
|
|
|
|
|
|
|
if (mq1) {
|
|
|
|
delete mq;
|
|
|
|
|
|
|
|
mq = mq1;
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
ani->_movement = 0;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!mq->chain(ani)) {
|
|
|
|
delete mq;
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
return mq;
|
2013-09-27 19:16:15 +00:00
|
|
|
}
|
|
|
|
|
2013-10-16 21:37:04 +00:00
|
|
|
MessageQueue *MovGraph2::doWalkTo(StaticANIObject *obj, int xpos, int ypos, int fuzzyMatch, int staticsId) {
|
2013-10-06 20:39:49 +00:00
|
|
|
LinkInfo linkInfoDest;
|
|
|
|
LinkInfo linkInfoSource;
|
|
|
|
MovInfo1 movInfo1;
|
|
|
|
PicAniInfo picAniInfo;
|
2013-10-14 21:48:49 +00:00
|
|
|
Common::Point point;
|
2013-10-06 20:39:49 +00:00
|
|
|
|
|
|
|
int idx = getItemIndexByGameObjectId(obj->_id);
|
|
|
|
|
|
|
|
if (idx < 0)
|
|
|
|
return 0;
|
|
|
|
|
|
|
|
linkInfoSource.link = 0;
|
|
|
|
linkInfoSource.node = 0;
|
|
|
|
|
|
|
|
linkInfoDest.link = 0;
|
|
|
|
linkInfoDest.node = 0;
|
|
|
|
|
2013-10-08 20:16:51 +00:00
|
|
|
point.x = 0;
|
|
|
|
|
2013-10-14 21:48:49 +00:00
|
|
|
obj->getPicAniInfo(&picAniInfo);
|
2013-10-06 20:39:49 +00:00
|
|
|
|
|
|
|
int idxsub;
|
|
|
|
|
|
|
|
if (obj->_movement)
|
|
|
|
idxsub = getItemSubIndexByMovementId(idx, obj->_movement->_id);
|
|
|
|
else
|
|
|
|
idxsub = getItemSubIndexByStaticsId(idx, obj->_statics->_staticsId);
|
|
|
|
|
|
|
|
bool subMgm = false;
|
|
|
|
|
|
|
|
if (idxsub == -1) {
|
|
|
|
idxsub = getItemSubIndexByMGM(idx, obj);
|
|
|
|
subMgm = true;
|
|
|
|
|
|
|
|
if (idxsub == -1)
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (obj->_movement) {
|
2013-10-10 20:41:04 +00:00
|
|
|
int newx, newy;
|
|
|
|
|
2013-10-06 20:39:49 +00:00
|
|
|
if (subMgm) {
|
|
|
|
obj->_messageQueueId = 0;
|
2013-10-14 21:48:49 +00:00
|
|
|
obj->changeStatics2(_items[idx]->_subItems[idxsub]._staticsId1);
|
2013-10-10 20:41:04 +00:00
|
|
|
newx = obj->_ox;
|
|
|
|
newy = obj->_oy;
|
2013-10-06 20:39:49 +00:00
|
|
|
} else {
|
2013-10-10 20:41:04 +00:00
|
|
|
obj->_movement->calcSomeXY(point, 0);
|
|
|
|
newx = obj->_movement->_ox - point.x;
|
|
|
|
newy = obj->_movement->_oy - point.y;
|
2013-10-06 20:39:49 +00:00
|
|
|
if (idxsub != 1 && idxsub) {
|
|
|
|
if (idxsub == 2 || idxsub == 3) {
|
2013-10-10 20:41:04 +00:00
|
|
|
newy = obj->_movement->_oy;
|
2013-10-06 20:39:49 +00:00
|
|
|
}
|
|
|
|
} else {
|
2013-10-10 20:41:04 +00:00
|
|
|
newx = obj->_movement->_ox;
|
2013-10-06 20:39:49 +00:00
|
|
|
}
|
|
|
|
}
|
2013-10-10 20:41:04 +00:00
|
|
|
|
|
|
|
obj->_movement = 0;
|
|
|
|
obj->setOXY(newx, newy);
|
2013-10-06 20:39:49 +00:00
|
|
|
}
|
2013-10-10 20:41:04 +00:00
|
|
|
|
|
|
|
if (obj->_ox == xpos && obj->_oy == ypos) {
|
2013-10-14 21:48:49 +00:00
|
|
|
g_fullpipe->_globalMessageQueueList->compact();
|
|
|
|
|
|
|
|
MessageQueue *mq = new MessageQueue();
|
2013-10-10 20:41:04 +00:00
|
|
|
|
|
|
|
if (staticsId && obj->_statics->_staticsId != staticsId) {
|
2013-10-14 21:48:49 +00:00
|
|
|
int idxwalk = getItemSubIndexByStaticsId(idx, staticsId);
|
|
|
|
if (idxwalk == -1) {
|
|
|
|
obj->setPicAniInfo(&picAniInfo);
|
|
|
|
|
2013-10-18 12:54:59 +00:00
|
|
|
delete mq;
|
|
|
|
|
2013-10-06 20:39:49 +00:00
|
|
|
return 0;
|
|
|
|
}
|
2013-10-14 21:48:49 +00:00
|
|
|
|
|
|
|
ExCommand *ex = new ExCommand(picAniInfo.objectId, 1, _items[idx]->_subItems[idxsub]._walk[idxwalk]._movementId, 0, 0, 0, 1, 0, 0, 0);
|
|
|
|
|
|
|
|
ex->_field_24 = 1;
|
|
|
|
ex->_keyCode = picAniInfo.field_8;
|
|
|
|
ex->_excFlags |= 2;
|
|
|
|
|
|
|
|
mq->_exCommands.push_back(ex);
|
2013-10-06 20:39:49 +00:00
|
|
|
} else {
|
2013-10-14 21:48:49 +00:00
|
|
|
ExCommand *ex = new ExCommand(picAniInfo.objectId, 22, obj->_statics->_staticsId, 0, 0, 0, 1, 0, 0, 0);
|
|
|
|
|
|
|
|
ex->_keyCode = picAniInfo.field_8;
|
|
|
|
ex->_excFlags |= 3;
|
|
|
|
mq->_exCommands.push_back(ex);
|
|
|
|
|
|
|
|
ex = new ExCommand(picAniInfo.objectId, 5, -1, obj->_ox, obj->_oy, 0, 1, 0, 0, 0);
|
|
|
|
|
|
|
|
ex->_field_14 = -1;
|
|
|
|
ex->_keyCode = picAniInfo.field_8;
|
|
|
|
ex->_excFlags |= 3;
|
|
|
|
mq->_exCommands.push_back(ex);
|
2013-10-06 20:39:49 +00:00
|
|
|
}
|
2013-10-14 21:48:49 +00:00
|
|
|
|
|
|
|
obj->setPicAniInfo(&picAniInfo);
|
|
|
|
|
|
|
|
return mq;
|
2013-10-06 20:39:49 +00:00
|
|
|
}
|
2013-10-14 21:48:49 +00:00
|
|
|
|
2013-10-10 20:41:04 +00:00
|
|
|
linkInfoSource.node = findNode(obj->_ox, obj->_oy, 0);
|
|
|
|
|
2013-10-06 20:39:49 +00:00
|
|
|
if (!linkInfoSource.node) {
|
2013-10-10 20:41:04 +00:00
|
|
|
linkInfoSource.link = findLink1(obj->_ox, obj->_oy, idxsub, 0);
|
|
|
|
|
|
|
|
if (!linkInfoSource.link) {
|
|
|
|
linkInfoSource.link = findLink2(obj->_ox, obj->_oy);
|
|
|
|
|
|
|
|
if (!linkInfoSource.link) {
|
2013-10-15 06:10:57 +00:00
|
|
|
obj->setPicAniInfo(&picAniInfo);
|
2013-10-10 20:41:04 +00:00
|
|
|
|
2013-10-06 20:39:49 +00:00
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2013-10-10 20:41:04 +00:00
|
|
|
|
|
|
|
linkInfoDest.node = findNode(xpos, ypos, fuzzyMatch);
|
|
|
|
|
2013-10-06 20:39:49 +00:00
|
|
|
if (!linkInfoDest.node) {
|
2013-10-10 20:41:04 +00:00
|
|
|
linkInfoDest.link = findLink1(xpos, ypos, idxsub, fuzzyMatch);
|
2013-10-17 19:44:30 +00:00
|
|
|
|
2013-10-10 20:41:04 +00:00
|
|
|
if (!linkInfoDest.link) {
|
2013-10-15 06:10:57 +00:00
|
|
|
obj->setPicAniInfo(&picAniInfo);
|
2013-10-10 20:41:04 +00:00
|
|
|
|
2013-10-06 20:39:49 +00:00
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
}
|
2013-10-10 20:41:04 +00:00
|
|
|
|
2013-10-15 06:10:57 +00:00
|
|
|
Common::Array<MovGraphLink *> tempLinkList;
|
2013-10-18 21:44:10 +00:00
|
|
|
double minPath = findMinPath(&linkInfoSource, &linkInfoDest, &tempLinkList);
|
2013-10-10 20:41:04 +00:00
|
|
|
|
2013-11-02 23:55:58 +00:00
|
|
|
debug(0, "MovGraph2::doWalkTo(): path: %g parts: %d", minPath, tempLinkList.size());
|
2013-10-18 21:44:10 +00:00
|
|
|
|
|
|
|
if (minPath < 0.0 || ((linkInfoSource.node != linkInfoDest.node || !linkInfoSource.node) && !tempLinkList.size()))
|
2013-10-06 20:39:49 +00:00
|
|
|
return 0;
|
2013-10-15 06:10:57 +00:00
|
|
|
|
2013-10-06 20:39:49 +00:00
|
|
|
movInfo1.subIndex = idxsub;
|
2013-10-15 06:10:57 +00:00
|
|
|
movInfo1.pt1.x = obj->_ox;
|
|
|
|
movInfo1.pt1.y = obj->_oy;
|
|
|
|
|
|
|
|
int dx1 = obj->_ox;
|
|
|
|
int dy1 = obj->_oy;
|
|
|
|
int dx2, dy2;
|
2013-10-10 20:41:04 +00:00
|
|
|
|
2013-10-06 20:39:49 +00:00
|
|
|
if (linkInfoSource.node)
|
2013-10-15 06:10:57 +00:00
|
|
|
movInfo1.distance1 = linkInfoSource.node->_distance;
|
2013-10-06 20:39:49 +00:00
|
|
|
else
|
2013-10-15 06:10:57 +00:00
|
|
|
movInfo1.distance1 = linkInfoSource.link->_movGraphNode1->_distance;
|
2013-10-10 20:41:04 +00:00
|
|
|
|
2013-10-06 20:39:49 +00:00
|
|
|
if (linkInfoDest.node) {
|
2013-10-15 06:10:57 +00:00
|
|
|
dx2 = linkInfoDest.node->_x;
|
|
|
|
dy2 = linkInfoDest.node->_y;
|
|
|
|
|
|
|
|
movInfo1.pt2.x = linkInfoDest.node->_x;
|
|
|
|
movInfo1.pt2.y = linkInfoDest.node->_y;
|
|
|
|
|
|
|
|
movInfo1.distance2 = linkInfoDest.node->_distance;
|
2013-10-06 20:39:49 +00:00
|
|
|
} else {
|
|
|
|
movInfo1.pt2.x = xpos;
|
|
|
|
movInfo1.pt2.y = ypos;
|
2013-10-15 06:10:57 +00:00
|
|
|
|
|
|
|
MovGraphNode *nod = linkInfoDest.link->_movGraphNode1;
|
2013-10-16 22:08:42 +00:00
|
|
|
double dst1 = sqrt((double)((ypos - nod->_y) * (ypos - nod->_y) + (xpos - nod->_x) * (xpos - nod->_x)));
|
2013-10-15 06:10:57 +00:00
|
|
|
int dst = linkInfoDest.link->_movGraphNode2->_distance - nod->_distance;
|
|
|
|
|
|
|
|
movInfo1.distance2 = nod->_distance + (dst1 * (double)dst / linkInfoDest.link->_distance);
|
|
|
|
|
|
|
|
calcDistance(&movInfo1.pt2, linkInfoDest.link, 1);
|
|
|
|
|
|
|
|
dx1 = movInfo1.pt1.x;
|
|
|
|
dy1 = movInfo1.pt1.y;
|
|
|
|
dx2 = movInfo1.pt2.x;
|
|
|
|
dy2 = movInfo1.pt2.y;
|
2013-10-06 20:39:49 +00:00
|
|
|
}
|
2013-10-10 20:41:04 +00:00
|
|
|
|
2013-10-06 20:39:49 +00:00
|
|
|
if (staticsId) {
|
2013-10-16 21:35:02 +00:00
|
|
|
movInfo1.item1Index = getItemSubIndexByStaticsId(idx, staticsId);
|
|
|
|
} else if (tempLinkList.size() <= 1) {
|
|
|
|
if (tempLinkList.size() == 1)
|
|
|
|
movInfo1.item1Index = getShortSide(tempLinkList[0], dx2 - dx1, dy2 - dy1);
|
2013-10-06 20:39:49 +00:00
|
|
|
else
|
2013-10-16 21:35:02 +00:00
|
|
|
movInfo1.item1Index = getShortSide(0, dx2 - dx1, dy2 - dy1);
|
2013-10-06 20:39:49 +00:00
|
|
|
} else {
|
2013-10-19 14:30:57 +00:00
|
|
|
movInfo1.item1Index = findLink(&tempLinkList, tempLinkList.size() - 1, 0, 0);
|
2013-10-06 20:39:49 +00:00
|
|
|
}
|
2013-10-16 21:35:02 +00:00
|
|
|
|
2013-10-06 20:39:49 +00:00
|
|
|
movInfo1.flags = fuzzyMatch != 0;
|
2013-10-16 21:35:02 +00:00
|
|
|
|
|
|
|
if (_items[idx]->_subItems[idxsub]._staticsId1 != obj->_statics->_staticsId)
|
|
|
|
movInfo1.flags |= 2;
|
|
|
|
|
|
|
|
buildMovInfo1SubItems(&movInfo1, &tempLinkList, &linkInfoSource, &linkInfoDest);
|
|
|
|
|
|
|
|
MessageQueue *mq = buildMovInfo1MessageQueue(&movInfo1);
|
|
|
|
|
|
|
|
linkInfoDest.node = findNode(movInfo1.pt2.x, movInfo1.pt2.y, fuzzyMatch);
|
|
|
|
|
|
|
|
if (!linkInfoDest.node)
|
|
|
|
linkInfoDest.link = findLink1(movInfo1.pt2.x, movInfo1.pt2.y, movInfo1.item1Index, fuzzyMatch);
|
|
|
|
|
|
|
|
if (fuzzyMatch || linkInfoDest.link || linkInfoDest.node) {
|
|
|
|
if (mq && mq->getCount() > 0 && picAniInfo.movementId) {
|
|
|
|
ExCommand *ex = mq->getExCommandByIndex(0);
|
|
|
|
|
|
|
|
if (ex && (ex->_messageKind == 1 || ex->_messageKind == 20)
|
|
|
|
&& picAniInfo.movementId == ex->_messageNum
|
|
|
|
&& picAniInfo.someDynamicPhaseIndex == ex->_field_14) {
|
|
|
|
mq->deleteExCommandByIndex(0, 1);
|
2013-10-06 20:39:49 +00:00
|
|
|
} else {
|
2013-10-16 21:35:02 +00:00
|
|
|
ex = new ExCommand(picAniInfo.objectId, 5, ex->_messageNum, obj->_ox, obj->_oy, 0, 1, 0, 0, 0);
|
2013-10-16 06:03:43 +00:00
|
|
|
ex->_field_14 = -1;
|
|
|
|
ex->_keyCode = picAniInfo.field_8;
|
|
|
|
ex->_excFlags |= 2;
|
2013-10-16 21:35:02 +00:00
|
|
|
mq->addExCommand(ex);
|
|
|
|
|
|
|
|
ex = new ExCommand(picAniInfo.objectId, 22, _items[idx]->_subItems[idxsub]._staticsId1, 0, 0, 0, 1, 0, 0, 0);
|
|
|
|
|
|
|
|
ex->_keyCode = picAniInfo.field_8;
|
|
|
|
ex->_excFlags |= 3;
|
|
|
|
mq->addExCommand(ex);
|
2013-10-06 20:39:49 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
} else {
|
2013-10-16 21:35:02 +00:00
|
|
|
if (mq)
|
|
|
|
delete mq;
|
|
|
|
mq = 0;
|
2013-10-06 20:39:49 +00:00
|
|
|
}
|
2013-09-27 19:16:15 +00:00
|
|
|
|
2013-10-16 21:35:02 +00:00
|
|
|
obj->setPicAniInfo(&picAniInfo);
|
|
|
|
|
|
|
|
return mq;
|
2013-09-27 19:16:15 +00:00
|
|
|
}
|
|
|
|
|
2013-10-15 06:10:57 +00:00
|
|
|
MovGraphNode *MovGraph2::findNode(int x, int y, int fuzzyMatch) {
|
2013-10-16 21:55:34 +00:00
|
|
|
for (ObList::iterator i = _nodes.begin(); i != _nodes.end(); ++i) {
|
|
|
|
assert(((CObject *)*i)->_objtype == kObjTypeMovGraphNode);
|
|
|
|
|
|
|
|
MovGraphNode *node = (MovGraphNode *)*i;
|
|
|
|
|
|
|
|
if (fuzzyMatch) {
|
|
|
|
if (abs(node->_x - x) < 15 && abs(node->_y - y) < 15)
|
|
|
|
return node;
|
|
|
|
} else {
|
|
|
|
if (node->_x == x && node->_y == y)
|
|
|
|
return node;
|
|
|
|
}
|
|
|
|
}
|
2013-10-15 06:10:57 +00:00
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2013-10-16 21:35:02 +00:00
|
|
|
int MovGraph2::getShortSide(MovGraphLink *lnk, int x, int y) {
|
2013-10-21 21:22:19 +00:00
|
|
|
bool cond;
|
2013-10-16 21:35:02 +00:00
|
|
|
|
2013-10-21 21:22:19 +00:00
|
|
|
if (lnk)
|
|
|
|
cond = abs(lnk->_movGraphNode2->_x - lnk->_movGraphNode1->_x) > abs(lnk->_movGraphNode2->_y - lnk->_movGraphNode1->_y);
|
|
|
|
else
|
|
|
|
cond = abs(x) > abs(y);
|
|
|
|
|
|
|
|
if (cond)
|
|
|
|
return x <= 0;
|
|
|
|
else
|
|
|
|
return ((y > 0) + 2);
|
2013-10-16 21:35:02 +00:00
|
|
|
}
|
|
|
|
|
2013-10-19 14:30:57 +00:00
|
|
|
int MovGraph2::findLink(Common::Array<MovGraphLink *> *linkList, int idx, Common::Rect *rect, Common::Point *point) {
|
|
|
|
MovGraphNode *node1 = (*linkList)[idx]->_movGraphNode1;
|
|
|
|
MovGraphNode *node2 = (*linkList)[idx]->_movGraphNode2;
|
|
|
|
MovGraphNode *node3 = node1;
|
2013-10-16 21:35:02 +00:00
|
|
|
|
2013-10-19 14:30:57 +00:00
|
|
|
if (idx != 0) {
|
|
|
|
MovGraphLink *lnk = (*linkList)[idx - 1];
|
|
|
|
|
|
|
|
if (lnk->_movGraphNode2 != node1) {
|
|
|
|
if (lnk->_movGraphNode1 != node1) {
|
|
|
|
if (lnk->_movGraphNode2 == node2 || lnk->_movGraphNode1 == node2) {
|
|
|
|
node3 = node2;
|
|
|
|
node2 = node1;
|
|
|
|
}
|
|
|
|
goto LABEL_7;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
node3 = node1;
|
2013-11-02 23:55:58 +00:00
|
|
|
} else if (idx != (int)(linkList->size() - 1)) {
|
2013-10-19 14:30:57 +00:00
|
|
|
MovGraphLink *lnk = (*linkList)[idx + 1];
|
|
|
|
|
|
|
|
if (lnk->_movGraphNode1 == node1 || lnk->_movGraphNode1 == node1) {
|
|
|
|
node3 = node2;
|
|
|
|
node2 = node1;
|
|
|
|
} else if (lnk->_movGraphNode2 == node2 || lnk->_movGraphNode1 == node2) {
|
|
|
|
node3 = node1;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
LABEL_7:
|
|
|
|
if (rect) {
|
|
|
|
rect->left = node3->_x;
|
|
|
|
rect->top = node3->_y;
|
|
|
|
rect->right = node2->_x;
|
|
|
|
rect->bottom = node2->_y;
|
|
|
|
}
|
|
|
|
if (point) {
|
|
|
|
point->x = node3->_distance;
|
|
|
|
point->y = node2->_distance;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (abs(node3->_x - node2->_x) <= abs(node3->_y - node2->_y))
|
|
|
|
return (node3->_y < node2->_x) + 2;
|
|
|
|
else
|
|
|
|
return node3->_x >= node2->_x;
|
2013-10-16 21:35:02 +00:00
|
|
|
}
|
|
|
|
|
2013-10-15 06:10:57 +00:00
|
|
|
MovGraphLink *MovGraph2::findLink1(int x, int y, int idx, int fuzzyMatch) {
|
2013-10-17 19:44:30 +00:00
|
|
|
Common::Point point;
|
|
|
|
MovGraphLink *res = 0;
|
2013-10-15 06:10:57 +00:00
|
|
|
|
2013-10-17 19:44:30 +00:00
|
|
|
for (ObList::iterator i = _links.begin(); i != _links.end(); ++i) {
|
|
|
|
assert(((CObject *)*i)->_objtype == kObjTypeMovGraphLink);
|
|
|
|
|
|
|
|
MovGraphLink *lnk = (MovGraphLink *)*i;
|
|
|
|
|
|
|
|
if (fuzzyMatch) {
|
|
|
|
point.x = x;
|
|
|
|
point.y = y;
|
|
|
|
double dst = calcDistance(&point, lnk, 0);
|
|
|
|
|
|
|
|
if (dst >= 0.0 && dst < 2.0)
|
|
|
|
return lnk;
|
|
|
|
} else if (!(lnk->_flags & 0x20000000)) {
|
|
|
|
if (lnk->_movGraphReact->pointInRegion(x, y)) {
|
|
|
|
if (abs(lnk->_movGraphNode1->_x - lnk->_movGraphNode2->_x) <= abs(lnk->_movGraphNode1->_y - lnk->_movGraphNode2->_y)) {
|
|
|
|
if (idx == 2 || idx == 3)
|
|
|
|
return lnk;
|
|
|
|
res = lnk;
|
|
|
|
} else {
|
|
|
|
if (idx == 1 || !idx)
|
|
|
|
return lnk;
|
|
|
|
res = lnk;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return res;
|
2013-10-15 06:10:57 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
MovGraphLink *MovGraph2::findLink2(int x, int y) {
|
2013-10-17 20:09:26 +00:00
|
|
|
double mindist = 1.0e20;
|
2013-11-02 23:15:49 +00:00
|
|
|
MovGraphLink *res = 0;
|
2013-10-15 06:10:57 +00:00
|
|
|
|
2013-10-17 20:09:26 +00:00
|
|
|
for (ObList::iterator i = _links.begin(); i != _links.end(); ++i) {
|
|
|
|
assert(((CObject *)*i)->_objtype == kObjTypeMovGraphLink);
|
|
|
|
|
|
|
|
MovGraphLink *lnk = (MovGraphLink *)*i;
|
|
|
|
|
|
|
|
if (!(lnk->_flags & 0x20000000)) {
|
|
|
|
double n1x = lnk->_movGraphNode1->_x;
|
|
|
|
double n1y = lnk->_movGraphNode1->_y;
|
|
|
|
double n2x = lnk->_movGraphNode2->_x;
|
|
|
|
double n2y = lnk->_movGraphNode2->_y;
|
|
|
|
double n1dx = n1x - x;
|
|
|
|
double n1dy = n1y - y;
|
|
|
|
double dst1 = sqrt(n1dy * n1dy + n1dx * n1dx);
|
|
|
|
double coeff1 = ((n1y - n2y) * n1dy + (n2x - n1x) * n1dx) / lnk->_distance / dst1;
|
|
|
|
double dst3 = coeff1 * dst1;
|
|
|
|
double dst2 = sqrt(1.0 - coeff1 * coeff1) * dst1;
|
|
|
|
|
|
|
|
if (coeff1 * dst1 < 0.0) {
|
|
|
|
dst3 = 0.0;
|
|
|
|
dst2 = sqrt(n1dy * n1dy + n1dx * n1dx);
|
|
|
|
}
|
|
|
|
if (dst3 > lnk->_distance) {
|
|
|
|
dst3 = lnk->_distance;
|
|
|
|
dst2 = sqrt((n2x - x) * (n2x - x) + (n2y - y) * (n2y - y));
|
|
|
|
}
|
|
|
|
if (dst3 >= 0.0 && dst3 <= lnk->_distance && dst2 < mindist) {
|
|
|
|
mindist = dst2;
|
|
|
|
res = lnk;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (mindist < 1.0e20)
|
|
|
|
return res;
|
|
|
|
else
|
|
|
|
return 0;
|
2013-10-15 06:10:57 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
double MovGraph2::findMinPath(LinkInfo *linkInfoSource, LinkInfo *linkInfoDest, Common::Array<MovGraphLink *> *listObj) {
|
2013-10-18 21:44:10 +00:00
|
|
|
LinkInfo linkInfoWorkSource;
|
|
|
|
|
|
|
|
if (linkInfoSource->link != linkInfoDest->link || linkInfoSource->node != linkInfoDest->node) {
|
|
|
|
double minDistance = -1.0;
|
|
|
|
|
|
|
|
if (linkInfoSource->node) {
|
|
|
|
for (ObList::iterator i = _links.begin(); i != _links.end(); ++i) {
|
|
|
|
MovGraphLink *lnk = (MovGraphLink *)*i;
|
|
|
|
|
|
|
|
if ((lnk->_movGraphNode1 == linkInfoSource->node || lnk->_movGraphNode2 == linkInfoSource->node) && !(lnk->_flags & 0xA0000000)) {
|
|
|
|
linkInfoWorkSource.node = 0;
|
|
|
|
linkInfoWorkSource.link = lnk;
|
|
|
|
|
|
|
|
Common::Array<MovGraphLink *> tmpList;
|
|
|
|
|
|
|
|
lnk->_flags |= 0x80000000;
|
|
|
|
|
|
|
|
double newDistance = findMinPath(&linkInfoWorkSource, linkInfoDest, &tmpList);
|
|
|
|
|
|
|
|
if (newDistance >= 0.0 && (minDistance < 0.0 || newDistance + lnk->_distance < minDistance)) {
|
|
|
|
listObj->clear();
|
|
|
|
listObj->push_back(tmpList);
|
|
|
|
|
|
|
|
minDistance = newDistance + lnk->_distance;
|
|
|
|
}
|
|
|
|
|
|
|
|
lnk->_flags &= 0x7FFFFFFF;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
} else if (linkInfoSource->link) {
|
|
|
|
linkInfoWorkSource.node = linkInfoSource->link->_movGraphNode1;
|
|
|
|
linkInfoWorkSource.link = 0;
|
|
|
|
|
|
|
|
Common::Array<MovGraphLink *> tmpList;
|
|
|
|
|
|
|
|
double newDistance = findMinPath(&linkInfoWorkSource, linkInfoDest, &tmpList);
|
2013-10-15 06:10:57 +00:00
|
|
|
|
2013-10-18 21:44:10 +00:00
|
|
|
if (newDistance >= 0.0) {
|
|
|
|
listObj->clear();
|
|
|
|
|
|
|
|
listObj->push_back(linkInfoSource->link);
|
|
|
|
listObj->push_back(tmpList);
|
|
|
|
|
|
|
|
minDistance = newDistance;
|
|
|
|
}
|
|
|
|
|
|
|
|
linkInfoWorkSource.link = 0;
|
|
|
|
linkInfoWorkSource.node = linkInfoSource->link->_movGraphNode2;
|
|
|
|
|
|
|
|
tmpList.clear();
|
|
|
|
|
|
|
|
newDistance = findMinPath(&linkInfoWorkSource, linkInfoDest, &tmpList);
|
|
|
|
|
|
|
|
if (newDistance >= 0 && (minDistance < 0.0 || newDistance < minDistance)) {
|
|
|
|
listObj->push_back(linkInfoSource->link);
|
|
|
|
listObj->push_back(tmpList);
|
|
|
|
|
|
|
|
minDistance = newDistance;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return minDistance;
|
|
|
|
} else {
|
|
|
|
if (linkInfoSource->link)
|
|
|
|
listObj->push_back(linkInfoSource->link);
|
|
|
|
|
|
|
|
return 0.0;
|
|
|
|
}
|
2013-10-15 06:10:57 +00:00
|
|
|
}
|
|
|
|
|
2013-09-18 15:04:36 +00:00
|
|
|
MovGraphNode *MovGraph::calcOffset(int ox, int oy) {
|
2013-10-22 21:36:43 +00:00
|
|
|
MovGraphNode *res = 0;
|
|
|
|
double mindist = 1.0e10;
|
2013-09-17 20:00:33 +00:00
|
|
|
|
2013-10-22 21:36:43 +00:00
|
|
|
for (ObList::iterator i = _nodes.begin(); i != _nodes.end(); ++i) {
|
|
|
|
assert(((CObject *)*i)->_objtype == kObjTypeMovGraphNode);
|
|
|
|
|
|
|
|
MovGraphNode *node = (MovGraphNode *)*i;
|
|
|
|
|
|
|
|
double dist = sqrt((double)((node->_x - oy) * (node->_x - oy) + (node->_x - ox) * (node->_x - ox)));
|
|
|
|
if (dist < mindist) {
|
|
|
|
mindist = dist;
|
|
|
|
res = node;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return res;
|
2013-09-17 20:00:33 +00:00
|
|
|
}
|
|
|
|
|
2013-10-01 19:56:50 +00:00
|
|
|
void MGM::clear() {
|
2013-10-02 06:03:08 +00:00
|
|
|
_items.clear();
|
2013-10-01 19:56:50 +00:00
|
|
|
}
|
|
|
|
|
2013-10-02 05:20:04 +00:00
|
|
|
MGMItem::MGMItem() {
|
|
|
|
objId = 0;
|
|
|
|
}
|
|
|
|
|
2013-10-02 06:24:23 +00:00
|
|
|
MGMSubItem::MGMSubItem() {
|
|
|
|
movement = 0;
|
|
|
|
staticsIndex = 0;
|
|
|
|
field_8 = 0;
|
|
|
|
field_C = 0;
|
|
|
|
x = 0;
|
|
|
|
y = 0;
|
|
|
|
}
|
|
|
|
|
2013-10-01 19:56:50 +00:00
|
|
|
void MGM::addItem(int objId) {
|
2013-10-02 05:20:04 +00:00
|
|
|
if (getItemIndexById(objId) == -1) {
|
|
|
|
MGMItem *item = new MGMItem();
|
|
|
|
|
|
|
|
item->objId = objId;
|
|
|
|
_items.push_back(item);
|
|
|
|
}
|
|
|
|
rebuildTables(objId);
|
|
|
|
}
|
|
|
|
|
|
|
|
void MGM::rebuildTables(int objId) {
|
2013-10-02 06:24:23 +00:00
|
|
|
int idx = getItemIndexById(objId);
|
|
|
|
|
|
|
|
if (idx == -1)
|
|
|
|
return;
|
|
|
|
|
|
|
|
_items[idx]->subItems.clear();
|
|
|
|
_items[idx]->statics.clear();
|
|
|
|
_items[idx]->movements1.clear();
|
|
|
|
_items[idx]->movements2.clear();
|
|
|
|
|
|
|
|
StaticANIObject *obj = g_fullpipe->_currentScene->getStaticANIObject1ById(objId, -1);
|
|
|
|
|
|
|
|
if (!obj)
|
|
|
|
return;
|
|
|
|
|
2013-10-02 19:39:11 +00:00
|
|
|
for (uint i = 0; i < obj->_staticsList.size(); i++)
|
|
|
|
_items[idx]->statics.push_back((Statics *)obj->_staticsList[i]);
|
|
|
|
|
|
|
|
for (uint i = 0; i < obj->_movements.size(); i++)
|
|
|
|
_items[idx]->movements1.push_back((Movement *)obj->_movements[i]);
|
|
|
|
|
|
|
|
_items[idx]->subItems.clear();
|
2013-10-02 05:20:04 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
int MGM::getItemIndexById(int objId) {
|
2013-10-02 05:55:07 +00:00
|
|
|
for (uint i = 0; i < _items.size(); i++)
|
|
|
|
if (_items[i]->objId == objId)
|
|
|
|
return i;
|
2013-10-02 05:20:04 +00:00
|
|
|
|
|
|
|
return -1;
|
2013-10-01 19:56:50 +00:00
|
|
|
}
|
|
|
|
|
2013-09-18 15:04:36 +00:00
|
|
|
MovGraphLink::MovGraphLink() {
|
2013-07-06 19:56:11 +00:00
|
|
|
_distance = 0;
|
|
|
|
_angle = 0;
|
|
|
|
_flags = 0x10000000;
|
|
|
|
_movGraphNode2 = 0;
|
|
|
|
_movGraphNode1 = 0;
|
|
|
|
_field_3C = 0;
|
|
|
|
_field_38 = 0;
|
|
|
|
_movGraphReact = 0;
|
2013-08-14 18:11:12 +00:00
|
|
|
_name = 0;
|
2013-10-17 19:44:30 +00:00
|
|
|
|
|
|
|
_objtype = kObjTypeMovGraphLink;
|
2013-06-16 13:10:46 +00:00
|
|
|
}
|
|
|
|
|
2013-09-18 15:04:36 +00:00
|
|
|
bool MovGraphLink::load(MfcArchive &file) {
|
|
|
|
debug(5, "MovGraphLink::load()");
|
2013-07-12 06:03:02 +00:00
|
|
|
|
2013-07-06 19:56:11 +00:00
|
|
|
_dwordArray1.load(file);
|
|
|
|
_dwordArray2.load(file);
|
2013-06-16 13:10:46 +00:00
|
|
|
|
2013-07-06 19:56:11 +00:00
|
|
|
_flags = file.readUint32LE();
|
2013-06-16 13:10:46 +00:00
|
|
|
|
2013-07-06 19:56:11 +00:00
|
|
|
debug(8, "GraphNode1");
|
2013-09-18 15:04:36 +00:00
|
|
|
_movGraphNode1 = (MovGraphNode *)file.readClass();
|
2013-07-06 19:56:11 +00:00
|
|
|
debug(8, "GraphNode2");
|
2013-09-18 15:04:36 +00:00
|
|
|
_movGraphNode2 = (MovGraphNode *)file.readClass();
|
2013-06-16 13:10:46 +00:00
|
|
|
|
2013-07-06 19:56:11 +00:00
|
|
|
_distance = file.readDouble();
|
|
|
|
_angle = file.readDouble();
|
2013-06-16 13:10:46 +00:00
|
|
|
|
2013-07-06 19:56:11 +00:00
|
|
|
debug(8, "distance: %g, angle: %g", _distance, _angle);
|
2013-06-16 13:10:46 +00:00
|
|
|
|
2013-09-18 15:04:36 +00:00
|
|
|
_movGraphReact = (MovGraphReact *)file.readClass();
|
2013-07-06 19:56:11 +00:00
|
|
|
_name = file.readPascalString();
|
2013-06-16 13:10:46 +00:00
|
|
|
|
2013-07-06 19:56:11 +00:00
|
|
|
return true;
|
2013-06-16 13:10:46 +00:00
|
|
|
}
|
|
|
|
|
2013-10-22 21:49:55 +00:00
|
|
|
void MovGraphLink::calcNodeDistanceAndAngle() {
|
|
|
|
if (_movGraphNode1) {
|
|
|
|
double dx = _movGraphNode2->_x - _movGraphNode1->_x;
|
|
|
|
double dy = _movGraphNode2->_y - _movGraphNode1->_y;
|
|
|
|
|
|
|
|
_distance = sqrt(dy * dy + dx * dx);
|
|
|
|
_angle = atan2(dx, dy);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2013-09-18 15:04:36 +00:00
|
|
|
bool MovGraphNode::load(MfcArchive &file) {
|
|
|
|
debug(5, "MovGraphNode::load()");
|
2013-07-12 06:03:02 +00:00
|
|
|
|
2013-07-06 19:56:11 +00:00
|
|
|
_field_14 = file.readUint32LE();
|
|
|
|
_x = file.readUint32LE();
|
|
|
|
_y = file.readUint32LE();
|
|
|
|
_distance = file.readUint32LE();
|
2013-06-16 13:10:46 +00:00
|
|
|
|
2013-07-06 19:56:11 +00:00
|
|
|
return true;
|
2013-06-16 13:10:46 +00:00
|
|
|
}
|
|
|
|
|
2013-09-18 15:04:36 +00:00
|
|
|
ReactParallel::ReactParallel() {
|
2013-07-06 19:56:11 +00:00
|
|
|
_x1 = 0;
|
|
|
|
_x2 = 0;
|
|
|
|
_dy = 0;
|
|
|
|
_dx = 0;
|
|
|
|
_y1 = 0;
|
|
|
|
_y2 = 0;
|
2013-06-18 14:04:29 +00:00
|
|
|
}
|
|
|
|
|
2013-09-18 15:04:36 +00:00
|
|
|
bool ReactParallel::load(MfcArchive &file) {
|
|
|
|
debug(5, "ReactParallel::load()");
|
2013-07-12 06:03:02 +00:00
|
|
|
|
2013-07-06 19:56:11 +00:00
|
|
|
_x1 = file.readUint32LE();
|
|
|
|
_y1 = file.readUint32LE();
|
|
|
|
_x2 = file.readUint32LE();
|
|
|
|
_y2 = file.readUint32LE();
|
|
|
|
_dx = file.readUint32LE();
|
|
|
|
_dy = file.readUint32LE();
|
2013-06-18 14:04:29 +00:00
|
|
|
|
2013-07-06 19:56:11 +00:00
|
|
|
createRegion();
|
2013-06-18 14:04:29 +00:00
|
|
|
|
2013-07-06 19:56:11 +00:00
|
|
|
return true;
|
2013-06-18 14:04:29 +00:00
|
|
|
}
|
|
|
|
|
2013-09-18 15:04:36 +00:00
|
|
|
void ReactParallel::createRegion() {
|
2013-07-06 19:56:11 +00:00
|
|
|
_points = (Common::Point **)malloc(sizeof(Common::Point *) * 4);
|
|
|
|
|
|
|
|
for (int i = 0; i < 4; i++)
|
|
|
|
_points[i] = new Common::Point;
|
2013-06-18 14:04:29 +00:00
|
|
|
|
2013-09-11 14:57:18 +00:00
|
|
|
double at = atan2((double)(_x1 - _x2), (double)(_y1 - _y2)) + 1.570796;
|
2013-07-06 19:56:11 +00:00
|
|
|
double sn = sin(at);
|
|
|
|
double cs = cos(at);
|
2013-06-18 14:04:29 +00:00
|
|
|
|
2013-07-06 19:56:11 +00:00
|
|
|
_points[0]->x = (int16)(_x1 - _dx * cs);
|
|
|
|
_points[0]->y = (int16)(_y1 - _dx * sn);
|
2013-06-18 14:04:29 +00:00
|
|
|
|
2013-07-06 19:56:11 +00:00
|
|
|
_points[1]->x = (int16)(_x2 - _dx * cs);
|
|
|
|
_points[1]->y = (int16)(_y2 - _dx * sn);
|
2013-06-18 14:04:29 +00:00
|
|
|
|
2013-10-05 08:52:50 +00:00
|
|
|
_points[2]->x = (int16)(_x2 + _dy * cs);
|
2013-07-06 19:56:11 +00:00
|
|
|
_points[2]->y = (int16)(_y2 + _dy * sn);
|
2013-06-18 14:04:29 +00:00
|
|
|
|
2013-07-06 19:56:11 +00:00
|
|
|
_points[3]->x = (int16)(_x1 + _dy * cs);
|
|
|
|
_points[3]->y = (int16)(_y1 + _dy * sn);
|
2013-06-18 14:04:29 +00:00
|
|
|
|
2013-10-05 08:52:50 +00:00
|
|
|
_pointCount = 4;
|
2013-07-06 19:56:11 +00:00
|
|
|
// GdiObject::Attach(_rgn, CreatePolygonRgn(_points, 4, 2);
|
2013-06-18 14:04:29 +00:00
|
|
|
}
|
|
|
|
|
2013-10-04 20:14:09 +00:00
|
|
|
void ReactParallel::method14() {
|
|
|
|
warning("STUB: ReactParallel::method14()");
|
|
|
|
}
|
|
|
|
|
2013-09-18 15:04:36 +00:00
|
|
|
ReactPolygonal::ReactPolygonal() {
|
2013-07-06 19:56:11 +00:00
|
|
|
_field_C = 0;
|
|
|
|
_field_10 = 0;
|
2013-06-18 14:23:49 +00:00
|
|
|
}
|
|
|
|
|
2013-09-18 15:04:36 +00:00
|
|
|
bool ReactPolygonal::load(MfcArchive &file) {
|
|
|
|
debug(5, "ReactPolygonal::load()");
|
2013-07-12 06:03:02 +00:00
|
|
|
|
2013-07-06 19:56:11 +00:00
|
|
|
_field_C = file.readUint32LE();
|
|
|
|
_field_10 = file.readUint32LE();
|
|
|
|
_pointCount = file.readUint32LE();
|
2013-06-18 14:23:49 +00:00
|
|
|
|
2013-07-06 19:56:11 +00:00
|
|
|
if (_pointCount > 0) {
|
|
|
|
_points = (Common::Point **)malloc(sizeof(Common::Point *) * _pointCount);
|
2013-06-18 14:23:49 +00:00
|
|
|
|
2013-07-06 19:56:11 +00:00
|
|
|
for (int i = 0; i < _pointCount; i++) {
|
|
|
|
_points[i] = new Common::Point;
|
2013-06-18 14:23:49 +00:00
|
|
|
|
2013-07-06 19:56:11 +00:00
|
|
|
_points[i]->x = file.readUint32LE();
|
|
|
|
_points[i]->y = file.readUint32LE();
|
|
|
|
}
|
2013-06-18 14:23:49 +00:00
|
|
|
|
2013-07-06 19:56:11 +00:00
|
|
|
}
|
2013-06-18 14:23:49 +00:00
|
|
|
|
2013-07-06 19:56:11 +00:00
|
|
|
createRegion();
|
2013-06-18 14:23:49 +00:00
|
|
|
|
2013-07-06 19:56:11 +00:00
|
|
|
return true;
|
2013-06-18 14:23:49 +00:00
|
|
|
}
|
|
|
|
|
2013-09-18 15:04:36 +00:00
|
|
|
void ReactPolygonal::createRegion() {
|
2013-07-06 19:56:11 +00:00
|
|
|
if (_points) {
|
2013-06-18 14:23:49 +00:00
|
|
|
|
2013-07-06 19:56:11 +00:00
|
|
|
// GdiObject::Attach(_rgn, CreatePolygonRgn(_points, _pointCount, 2);
|
|
|
|
}
|
2013-06-18 14:23:49 +00:00
|
|
|
}
|
2013-06-18 14:04:29 +00:00
|
|
|
|
2013-10-04 20:14:09 +00:00
|
|
|
void ReactPolygonal::method14() {
|
|
|
|
warning("STUB: ReactPolygonal::method14()");
|
|
|
|
}
|
|
|
|
|
2013-10-05 08:52:50 +00:00
|
|
|
bool MovGraphReact::pointInRegion(int x, int y) {
|
|
|
|
if (_pointCount < 3) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
int counter = 0;
|
|
|
|
double xinters;
|
|
|
|
Common::Point p, p1, p2;
|
|
|
|
|
|
|
|
p.x = (double)x;
|
|
|
|
p.y = (double)y;
|
|
|
|
|
|
|
|
p1.x = (double)_points[0]->x;
|
|
|
|
p1.y = (double)_points[0]->y;
|
|
|
|
|
2013-10-06 05:11:09 +00:00
|
|
|
for (int i = 1; i <= _pointCount; i++) {
|
2013-10-05 08:52:50 +00:00
|
|
|
p2.x = (double)_points[i % _pointCount]->x;
|
|
|
|
p2.y = (double)_points[i % _pointCount]->y;
|
|
|
|
|
|
|
|
if (p.y > MIN(p1.y, p2.y)) {
|
|
|
|
if (p.y <= MAX(p1.y, p2.y)) {
|
|
|
|
if (p.x <= MAX(p1.x, p2.x)) {
|
|
|
|
if (p1.y != p2.y) {
|
|
|
|
xinters = (p.y - p1.y) * (p2.x - p1.x) / (p2.y - p1.y) + p1.x;
|
|
|
|
if (p1.x == p2.x || p.x <= xinters) {
|
|
|
|
counter++;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
p1 = p2;
|
|
|
|
}
|
2013-10-04 20:14:09 +00:00
|
|
|
|
2013-10-05 08:52:50 +00:00
|
|
|
if (counter % 2 == 0) {
|
|
|
|
return false;
|
|
|
|
} else {
|
|
|
|
return true;
|
|
|
|
}
|
2013-10-04 20:14:09 +00:00
|
|
|
}
|
|
|
|
|
2013-08-26 19:17:20 +00:00
|
|
|
int startWalkTo(int objId, int objKey, int x, int y, int a5) {
|
2013-09-18 18:30:06 +00:00
|
|
|
MctlCompound *mc = getSc2MctlCompoundBySceneId(g_fullpipe->_currentScene->_sceneId);
|
|
|
|
|
|
|
|
if (mc)
|
|
|
|
return (mc->method34(g_fullpipe->_currentScene->getStaticANIObject1ById(objId, objKey), x, y, a5, 0) != 0);
|
2013-08-26 19:17:20 +00:00
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
int doSomeAnimation(int objId, int objKey, int a3) {
|
|
|
|
warning("STUB: doSomeAnimation(%d, %d, %d)", objId, objKey, a3);
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
int doSomeAnimation2(int objId, int objKey) {
|
|
|
|
return doSomeAnimation(objId, objKey, 0);
|
|
|
|
}
|
|
|
|
|
2013-06-16 12:11:18 +00:00
|
|
|
} // End of namespace Fullpipe
|