mirror of
https://github.com/vcmi/vcmi.git
synced 2025-01-12 02:28:11 +02:00
Merge pull request #2555 from vcmi/fix_rmg_object_pos
Fix hota offset + 2 possible crashes
This commit is contained in:
commit
4d4eaed808
@ -344,16 +344,6 @@ void Object::Instance::finalize(RmgMap & map)
|
|||||||
setTemplate(terrainType->getId());
|
setTemplate(terrainType->getId());
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
if (dObject.ID == Obj::MONSTER)
|
|
||||||
{
|
|
||||||
//Make up for extra offset in HotA creature templates
|
|
||||||
auto visitableOffset = dObject.getVisitableOffset();
|
|
||||||
auto fixedPos = getPosition(true) + visitableOffset;
|
|
||||||
vstd::abetween(fixedPos.x, visitableOffset.x, map.width() - 1);
|
|
||||||
vstd::abetween(fixedPos.y, visitableOffset.y, map.height() - 1);
|
|
||||||
int3 parentPos = getPosition(true) - getPosition(false);
|
|
||||||
setPosition(fixedPos - parentPos);
|
|
||||||
}
|
|
||||||
|
|
||||||
if (dObject.isVisitable() && !map.isOnMap(dObject.visitablePos()))
|
if (dObject.isVisitable() && !map.isOnMap(dObject.visitablePos()))
|
||||||
throw rmgException(boost::to_string(boost::format("Visitable tile %s of object %d at %s is outside the map") % dObject.visitablePos().toString() % dObject.id % dObject.pos.toString()));
|
throw rmgException(boost::to_string(boost::format("Visitable tile %s of object %d at %s is outside the map") % dObject.visitablePos().toString() % dObject.id % dObject.pos.toString()));
|
||||||
|
@ -129,6 +129,19 @@ int3 ObjectManager::findPlaceForObject(const rmg::Area & searchArea, rmg::Object
|
|||||||
{
|
{
|
||||||
float bestWeight = 0.f;
|
float bestWeight = 0.f;
|
||||||
int3 result(-1, -1, -1);
|
int3 result(-1, -1, -1);
|
||||||
|
|
||||||
|
//Blocked area might not cover object position if it has an offset from (0,0)
|
||||||
|
auto outsideTheMap = [this, &obj]() -> bool
|
||||||
|
{
|
||||||
|
for (const auto& oi : obj.instances())
|
||||||
|
{
|
||||||
|
if (!map.isOnMap(oi->getPosition(true)))
|
||||||
|
{
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return false;
|
||||||
|
};
|
||||||
|
|
||||||
if(optimizer & OptimizeType::DISTANCE)
|
if(optimizer & OptimizeType::DISTANCE)
|
||||||
{
|
{
|
||||||
@ -149,6 +162,9 @@ int3 ObjectManager::findPlaceForObject(const rmg::Area & searchArea, rmg::Object
|
|||||||
|
|
||||||
if(!searchArea.contains(obj.getArea()) || !searchArea.overlap(obj.getAccessibleArea()))
|
if(!searchArea.contains(obj.getArea()) || !searchArea.overlap(obj.getAccessibleArea()))
|
||||||
continue;
|
continue;
|
||||||
|
|
||||||
|
if (outsideTheMap())
|
||||||
|
continue;
|
||||||
|
|
||||||
float weight = weightFunction(tile);
|
float weight = weightFunction(tile);
|
||||||
if(weight > bestWeight)
|
if(weight > bestWeight)
|
||||||
@ -168,9 +184,12 @@ int3 ObjectManager::findPlaceForObject(const rmg::Area & searchArea, rmg::Object
|
|||||||
|
|
||||||
if (obj.getVisibleTop().y < 0)
|
if (obj.getVisibleTop().y < 0)
|
||||||
continue;
|
continue;
|
||||||
|
|
||||||
if(!searchArea.contains(obj.getArea()) || !searchArea.overlap(obj.getAccessibleArea()))
|
if(!searchArea.contains(obj.getArea()) || !searchArea.overlap(obj.getAccessibleArea()))
|
||||||
continue;
|
continue;
|
||||||
|
|
||||||
|
if (outsideTheMap())
|
||||||
|
continue;
|
||||||
|
|
||||||
float weight = weightFunction(tile);
|
float weight = weightFunction(tile);
|
||||||
if(weight > bestWeight)
|
if(weight > bestWeight)
|
||||||
@ -416,7 +435,7 @@ bool ObjectManager::createRequiredObjects()
|
|||||||
|
|
||||||
//create object on specific positions
|
//create object on specific positions
|
||||||
//TODO: implement guards
|
//TODO: implement guards
|
||||||
for (const auto &objInfo : instantObjects)
|
for (const auto &objInfo : instantObjects) //Unused ATM
|
||||||
{
|
{
|
||||||
rmg::Object rmgObject(*objInfo.obj);
|
rmg::Object rmgObject(*objInfo.obj);
|
||||||
rmgObject.setPosition(objInfo.pos);
|
rmgObject.setPosition(objInfo.pos);
|
||||||
@ -433,6 +452,28 @@ bool ObjectManager::createRequiredObjects()
|
|||||||
|
|
||||||
void ObjectManager::placeObject(rmg::Object & object, bool guarded, bool updateDistance, bool createRoad/* = false*/)
|
void ObjectManager::placeObject(rmg::Object & object, bool guarded, bool updateDistance, bool createRoad/* = false*/)
|
||||||
{
|
{
|
||||||
|
//object.finalize(map);
|
||||||
|
|
||||||
|
if (object.instances().size() == 1 && object.instances().front()->object().ID == Obj::MONSTER)
|
||||||
|
{
|
||||||
|
//Fix for HoTA offset - lonely guards
|
||||||
|
|
||||||
|
auto monster = object.instances().front();
|
||||||
|
if (!monster->object().appearance)
|
||||||
|
{
|
||||||
|
//Needed to determine visitable offset
|
||||||
|
monster->setAnyTemplate();
|
||||||
|
}
|
||||||
|
object.getPosition();
|
||||||
|
auto visitableOffset = monster->object().getVisitableOffset();
|
||||||
|
auto fixedPos = monster->getPosition(true) + visitableOffset;
|
||||||
|
|
||||||
|
//Do not place guard outside the map
|
||||||
|
vstd::abetween(fixedPos.x, visitableOffset.x, map.width() - 1);
|
||||||
|
vstd::abetween(fixedPos.y, visitableOffset.y, map.height() - 1);
|
||||||
|
int3 parentOffset = monster->getPosition(true) - monster->getPosition(false);
|
||||||
|
monster->setPosition(fixedPos - parentOffset);
|
||||||
|
}
|
||||||
object.finalize(map);
|
object.finalize(map);
|
||||||
|
|
||||||
Zone::Lock lock(zone.areaMutex);
|
Zone::Lock lock(zone.areaMutex);
|
||||||
@ -443,8 +484,8 @@ void ObjectManager::placeObject(rmg::Object & object, bool guarded, bool updateD
|
|||||||
zone.freePaths().add(object.getVisitablePosition());
|
zone.freePaths().add(object.getVisitablePosition());
|
||||||
zone.areaUsed().unite(object.getArea());
|
zone.areaUsed().unite(object.getArea());
|
||||||
zone.areaUsed().erase(object.getVisitablePosition());
|
zone.areaUsed().erase(object.getVisitablePosition());
|
||||||
|
|
||||||
if(guarded)
|
if(guarded) //We assume the monster won't be guarded
|
||||||
{
|
{
|
||||||
auto guardedArea = object.instances().back()->getAccessibleArea();
|
auto guardedArea = object.instances().back()->getAccessibleArea();
|
||||||
guardedArea.add(object.instances().back()->getVisitablePosition());
|
guardedArea.add(object.instances().back()->getVisitablePosition());
|
||||||
@ -501,6 +542,7 @@ void ObjectManager::placeObject(rmg::Object & object, bool guarded, bool updateD
|
|||||||
{
|
{
|
||||||
case Obj::RANDOM_TREASURE_ART:
|
case Obj::RANDOM_TREASURE_ART:
|
||||||
case Obj::RANDOM_MINOR_ART: //In OH3 quest artifacts have higher value than normal arts
|
case Obj::RANDOM_MINOR_ART: //In OH3 quest artifacts have higher value than normal arts
|
||||||
|
case Obj::RANDOM_RESOURCE:
|
||||||
{
|
{
|
||||||
if (auto * qap = zone.getModificator<QuestArtifactPlacer>())
|
if (auto * qap = zone.getModificator<QuestArtifactPlacer>())
|
||||||
{
|
{
|
||||||
@ -629,8 +671,12 @@ bool ObjectManager::addGuard(rmg::Object & object, si32 strength, bool zoneGuard
|
|||||||
});
|
});
|
||||||
|
|
||||||
auto & instance = object.addInstance(*guard);
|
auto & instance = object.addInstance(*guard);
|
||||||
instance.setPosition(guardPos - object.getPosition());
|
|
||||||
instance.setAnyTemplate(); //terrain is irrelevant for monsters, but monsters need some template now
|
instance.setAnyTemplate(); //terrain is irrelevant for monsters, but monsters need some template now
|
||||||
|
|
||||||
|
//Fix HoTA monsters with offset template
|
||||||
|
auto visitableOffset = instance.object().getVisitableOffset();
|
||||||
|
auto fixedPos = guardPos - object.getPosition() + visitableOffset;
|
||||||
|
instance.setPosition(fixedPos);
|
||||||
|
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
Loading…
Reference in New Issue
Block a user