colobot/src/object/object_manager.cpp

392 lines
12 KiB
C++
Raw Normal View History

/*
* This file is part of the Colobot: Gold Edition source code
2015-08-22 14:40:02 +00:00
* Copyright (C) 2001-2015, Daniel Roux, EPSITEC SA & TerranovaTeam
* http://epsitec.ch; http://colobot.info; http://github.com/colobot
*
* 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 3 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, see http://gnu.org/licenses
*/
#include "object/object_manager.h"
2014-12-20 19:36:09 +00:00
2015-08-15 12:02:07 +00:00
#include "common/global.h"
#include "common/make_unique.h"
2014-12-20 19:36:09 +00:00
#include "math/all.h"
#include "object/object.h"
#include "object/object_create_exception.h"
#include "object/object_create_params.h"
#include "object/object_factory.h"
#include "object/old_object.h"
#include "object/auto/auto.h"
2014-12-20 19:36:09 +00:00
#include "physics/physics.h"
#include <algorithm>
template<> CObjectManager* CSingleton<CObjectManager>::m_instance = nullptr;
CObjectManager::CObjectManager(Gfx::CEngine* engine,
Gfx::CTerrain* terrain,
Gfx::COldModelManager* oldModelManager,
Gfx::CModelManager* modelManager,
2015-07-10 17:53:32 +00:00
Gfx::CParticle* particle)
2015-09-06 11:23:20 +00:00
: m_objectFactory(MakeUnique<CObjectFactory>(engine,
terrain,
oldModelManager,
modelManager,
particle)),
m_nextId(0),
m_activeObjectIterators(0),
m_shouldCleanRemovedObjects(false)
{
}
CObjectManager::~CObjectManager()
{
}
bool CObjectManager::DeleteObject(CObject* instance)
{
2014-12-20 18:09:53 +00:00
assert(instance != nullptr);
// TODO: temporarily...
auto oldObj = dynamic_cast<COldObject*>(instance);
if (oldObj != nullptr)
oldObj->DeleteObject();
auto it = m_objects.find(instance->GetID());
if (it != m_objects.end())
{
it->second.reset();
m_shouldCleanRemovedObjects = true;
return true;
}
return false;
}
2015-09-06 11:23:20 +00:00
void CObjectManager::CleanRemovedObjectsIfNeeded()
{
2015-09-06 11:23:20 +00:00
if (m_activeObjectIterators != 0)
return;
if (! m_shouldCleanRemovedObjects)
return;
auto it = m_objects.begin();
if (it != m_objects.end())
{
if (it->second == nullptr)
it = m_objects.erase(it);
}
m_shouldCleanRemovedObjects = false;
}
void CObjectManager::DeleteAllObjects()
{
for (auto& it : m_objects)
{
// TODO: temporarily...
auto oldObj = dynamic_cast<COldObject*>(it.second.get());
if (oldObj != nullptr)
{
bool all = true;
oldObj->DeleteObject(all);
}
}
m_objects.clear();
m_nextId = 0;
}
CObject* CObjectManager::GetObjectById(unsigned int id)
{
if (m_objects.count(id) == 0) return nullptr;
return m_objects[id].get();
}
CObject* CObjectManager::GetObjectByRank(unsigned int id)
{
if (id >= m_objects.size()) return nullptr;
auto it = m_objects.begin();
for (unsigned int i = 0; i < id; i++, ++it);
return it->second.get();
}
2015-07-22 10:45:50 +00:00
CObject* CObjectManager::CreateObject(ObjectCreateParams params)
{
if (params.id < 0)
{
params.id = m_nextId;
m_nextId++;
}
else
{
if (params.id >= m_nextId)
{
m_nextId = params.id + 1;
}
}
2015-07-22 10:45:50 +00:00
assert(m_objects.find(params.id) == m_objects.end());
auto objectUPtr = m_objectFactory->CreateObject(params);
if (objectUPtr == nullptr)
throw CObjectCreateException("Something went wrong in CObjectFactory", params.type);
CObject* objectPtr = objectUPtr.get();
m_objects[params.id] = std::move(objectUPtr);
return objectPtr;
}
2015-08-14 16:19:58 +00:00
CObject* CObjectManager::CreateObject(Math::Vector pos, float angle, ObjectType type, float power)
2014-12-20 19:36:09 +00:00
{
ObjectCreateParams params;
params.pos = pos;
params.angle = angle;
params.type = type;
params.power = power;
2015-07-22 10:45:50 +00:00
return CreateObject(params);
2014-12-20 19:36:09 +00:00
}
std::vector<CObject*> CObjectManager::GetObjectsOfTeam(int team)
{
std::vector<CObject*> result;
for (CObject* object : GetAllObjects())
{
if (object->GetTeam() == team)
{
result.push_back(object);
}
}
return result;
}
bool CObjectManager::TeamExists(int team)
{
if(team == 0) return true;
for (CObject* object : GetAllObjects())
{
if (!object->GetActive())
continue;
if (object->GetTeam() == team)
return true;
}
return false;
}
void CObjectManager::DestroyTeam(int team)
{
assert(team != 0);
for (CObject* object : GetAllObjects())
{
if (object->GetTeam() == team)
2015-08-13 16:54:44 +00:00
{
if (object->Implements(ObjectInterfaceType::Destroyable))
{
dynamic_cast<CDestroyableObject*>(object)->DestroyObject(DestructionType::Explosion);
}
else
{
DeleteObject(object);
}
}
}
}
2015-08-07 18:48:55 +00:00
int CObjectManager::CountObjectsImplementing(ObjectInterfaceType interface)
{
int count = 0;
for (CObject* object : GetAllObjects())
{
if (object->Implements(interface))
count++;
}
return count;
}
2014-12-20 19:36:09 +00:00
CObject* CObjectManager::Radar(CObject* pThis, ObjectType type, float angle, float focus, float minDist, float maxDist, bool furthest, RadarFilter filter, bool cbotTypes)
{
std::vector<ObjectType> types;
if (type != OBJECT_NULL)
types.push_back(type);
return Radar(pThis, types, angle, focus, minDist, maxDist, furthest, filter, cbotTypes);
2014-12-20 19:36:09 +00:00
}
CObject* CObjectManager::Radar(CObject* pThis, std::vector<ObjectType> type, float angle, float focus, float minDist, float maxDist, bool furthest, RadarFilter filter, bool cbotTypes)
{
Math::Vector iPos;
float iAngle;
if (pThis != nullptr)
{
iPos = pThis->GetPosition();
iAngle = pThis->GetRotationY();
iAngle = Math::NormAngle(iAngle); // 0..2*Math::PI
}
else
{
iPos = Math::Vector();
iAngle = 0.0f;
}
2014-12-20 19:36:09 +00:00
return Radar(pThis, iPos, iAngle, type, angle, focus, minDist, maxDist, furthest, filter, cbotTypes);
}
CObject* CObjectManager::Radar(CObject* pThis, Math::Vector thisPosition, float thisAngle, ObjectType type, float angle, float focus, float minDist, float maxDist, bool furthest, RadarFilter filter, bool cbotTypes)
{
std::vector<ObjectType> types;
if (type != OBJECT_NULL)
types.push_back(type);
return Radar(pThis, thisPosition, thisAngle, types, angle, focus, minDist, maxDist, furthest, filter, cbotTypes);
2014-12-20 19:36:09 +00:00
}
CObject* CObjectManager::Radar(CObject* pThis, Math::Vector thisPosition, float thisAngle, std::vector<ObjectType> type, float angle, float focus, float minDist, float maxDist, bool furthest, RadarFilter filter, bool cbotTypes)
{
CObject *pObj, *pBest;
Math::Vector iPos, oPos;
float best, iAngle, d, a;
ObjectType oType;
2014-12-20 19:36:09 +00:00
minDist *= g_unit;
maxDist *= g_unit;
2014-12-20 19:36:09 +00:00
iPos = thisPosition;
iAngle = thisAngle+angle;
iAngle = Math::NormAngle(iAngle); // 0..2*Math::PI
2015-07-07 16:54:36 +00:00
int filter_team = filter & 0xFF;
RadarFilter filter_flying = static_cast<RadarFilter>(filter & (FILTER_ONLYLANDING | FILTER_ONLYFLYING));
RadarFilter filter_enemy = static_cast<RadarFilter>(filter & (FILTER_FRIENDLY | FILTER_ENEMY | FILTER_NEUTRAL));
2014-12-20 19:36:09 +00:00
if ( !furthest ) best = 100000.0f;
else best = 0.0f;
pBest = nullptr;
for ( auto it = m_objects.begin() ; it != m_objects.end() ; ++it )
2014-12-20 19:36:09 +00:00
{
pObj = it->second.get();
if ( pObj == pThis ) continue; // pThis may be nullptr but it doesn't matter
2015-07-12 18:49:00 +00:00
if (pObj == nullptr) continue;
2015-07-10 17:03:27 +00:00
if (IsObjectBeingTransported(pObj)) continue;
if ( !pObj->GetDetectable() ) continue;
2014-12-20 19:36:09 +00:00
if ( pObj->GetProxyActivate() ) continue;
2014-12-20 19:36:09 +00:00
oType = pObj->GetType();
if (cbotTypes)
{
2014-12-20 19:36:09 +00:00
// TODO: handle this differently (new class describing types? CObjectType::GetBaseType()?)
if ( oType == OBJECT_RUINmobilew2 ||
oType == OBJECT_RUINmobilet1 ||
oType == OBJECT_RUINmobilet2 ||
oType == OBJECT_RUINmobiler1 ||
oType == OBJECT_RUINmobiler2 )
{
oType = OBJECT_RUINmobilew1; // any ruin
}
2014-12-20 19:36:09 +00:00
if ( oType == OBJECT_BARRIER2 ||
oType == OBJECT_BARRIER3 ) // barriers?
{
oType = OBJECT_BARRIER1; // any barrier
}
// END OF TODO
}
2015-07-14 19:39:51 +00:00
if ( std::find(type.begin(), type.end(), oType) == type.end() && type.size() > 0 ) continue;
if ( (oType == OBJECT_TOTO || oType == OBJECT_CONTROLLER) && type.size() == 0 ) continue; // allow OBJECT_TOTO and OBJECT_CONTROLLER only if explicitly asked in type parameter
2015-07-07 16:54:36 +00:00
if ( filter_flying == FILTER_ONLYLANDING )
2014-12-20 19:36:09 +00:00
{
if ( pObj->Implements(ObjectInterfaceType::Movable) )
{
CPhysics* physics = dynamic_cast<CMovableObject*>(pObj)->GetPhysics();
if ( physics != nullptr )
{
if ( !physics->GetLand() ) continue;
}
}
2014-12-20 19:36:09 +00:00
}
2015-07-07 16:54:36 +00:00
if ( filter_flying == FILTER_ONLYFLYING )
2014-12-20 19:36:09 +00:00
{
if ( !pObj->Implements(ObjectInterfaceType::Movable) ) continue;
CPhysics* physics = dynamic_cast<CMovableObject*>(pObj)->GetPhysics();
if ( physics == nullptr ) continue;
if ( physics->GetLand() ) continue;
2014-12-20 19:36:09 +00:00
}
2015-07-07 16:54:36 +00:00
if ( filter_team != 0 && pObj->GetTeam() != filter_team )
continue;
if( pThis != nullptr )
{
RadarFilter enemy = FILTER_NONE;
if ( pObj->GetTeam() == 0 ) enemy = static_cast<RadarFilter>(enemy | FILTER_NEUTRAL);
if ( pObj->GetTeam() != 0 && pObj->GetTeam() == pThis->GetTeam() ) enemy = static_cast<RadarFilter>(enemy | FILTER_FRIENDLY);
if ( pObj->GetTeam() != 0 && pObj->GetTeam() != pThis->GetTeam() ) enemy = static_cast<RadarFilter>(enemy | FILTER_ENEMY);
2015-07-07 16:54:36 +00:00
if ( filter_enemy != 0 && (filter_enemy & enemy) == 0 ) continue;
}
oPos = pObj->GetPosition();
2014-12-20 19:36:09 +00:00
d = Math::DistanceProjected(iPos, oPos);
if ( d < minDist || d > maxDist ) continue; // too close or too far?
2014-12-20 19:36:09 +00:00
a = Math::RotateAngle(oPos.x-iPos.x, iPos.z-oPos.z); // CW !
if ( Math::TestAngle(a, iAngle-focus/2.0f, iAngle+focus/2.0f) || focus >= Math::PI*2.0f )
{
if ( (!furthest && d < best) ||
(furthest && d > best) )
{
best = d;
pBest = pObj;
}
}
}
2014-12-20 19:36:09 +00:00
return pBest;
}
CObject* CObjectManager::FindNearest(CObject* pThis, ObjectType type, float maxDist, bool cbotTypes)
{
return Radar(pThis, type, 0.0f, Math::PI*2.0f, 0.0f, maxDist, false, FILTER_NONE, cbotTypes);
}
CObject* CObjectManager::FindNearest(CObject* pThis, std::vector<ObjectType> type, float maxDist, bool cbotTypes)
{
return Radar(pThis, type, 0.0f, Math::PI*2.0f, 0.0f, maxDist, false, FILTER_NONE, cbotTypes);
}
CObject* CObjectManager::FindNearest(CObject* pThis, Math::Vector thisPosition, ObjectType type, float maxDist, bool cbotTypes)
{
return Radar(pThis, thisPosition, 0.0f, type, 0.0f, Math::PI*2.0f, 0.0f, maxDist, false, FILTER_NONE, cbotTypes);
}
CObject* CObjectManager::FindNearest(CObject* pThis, Math::Vector thisPosition, std::vector<ObjectType> type, float maxDist, bool cbotTypes)
{
return Radar(pThis, thisPosition, 0.0f, type, 0.0f, Math::PI*2.0f, 0.0f, maxDist, false, FILTER_NONE, cbotTypes);
}