// Ryzom - MMORPG Framework
// Copyright (C) 2010 Winch Gate Property Limited
//
// This source file has been modified by the following contributors:
// Copyright (C) 2020 Jan BOON (Kaetemi)
//
// This program is free software: you can redistribute it and/or modify
// it under the terms of the GNU Affero 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 Affero General Public License for more details.
//
// You should have received a copy of the GNU Affero General Public License
// along with this program. If not, see .
#include "stdpch.h"
#include "nel/misc/path.h"
#include "nel/misc/matrix.h"
#include "nel/misc/vector.h"
#include "nel/misc/vectord.h"
#include "nel/misc/sheet_id.h"
#include "nel/pacs/u_primitive_block.h"
#include "nel/pacs/u_move_container.h"
#include "nel/pacs/u_move_primitive.h"
#include "nel/pacs/u_global_position.h"
#include "nel/3d/u_scene.h"
#include "nel/3d/u_instance.h"
#include "ig_callback.h"
#include "fix_season_data.h"
#include "sheet_manager.h"
#include "ig_enum.h"
#include "weather.h"
#include "pacs_client.h"
// Client Sheets
#include "client_sheets/plant_sheet.h"
#include
#ifdef DEBUG_NEW
#define new DEBUG_NEW
#endif
#define RZ_PRIM_ZEXT_BLOCK_AVOIDANCE 2.0f
H_AUTO_DECL(RZ_IGCallback)
///===================================================================================
CIGCallback::CIGCallback() : _MoveContainer(NULL)
{
H_AUTO_USE(RZ_IGCallback)
//nlinfo("**** YOYO: CREATING IG CALLBACK: %x", this);
}
///===================================================================================
CIGCallback::~CIGCallback()
{
//nlinfo("**** YOYO: DELETING IG CALLBACK: %x", this);
H_AUTO_USE(RZ_IGCallback)
deleteIGs();
}
///===================================================================================
void CIGCallback::resetContainer()
{
H_AUTO_USE(RZ_IGCallback)
if (!_MoveContainer) return;
deleteIGs();
_MoveContainer = NULL;
}
///===================================================================================
void CIGCallback::setMoveContainer(NLPACS::UMoveContainer *mc)
{
H_AUTO_USE(RZ_IGCallback)
nlassert(!_MoveContainer);
_MoveContainer = mc;
}
///===================================================================================
void CIGCallback::addIG(NL3D::UInstanceGroup *ig)
{
H_AUTO_USE(RZ_IGCallback)
nlassert(_MoveContainer);
CIGInstance *igi;
try
{
igi = new CIGInstance(ig, this);
_IGInstances.push_back(igi);
ig->setAddRemoveInstanceCallback(igi);
ig->setTransformNameCallback(igi);
ig->setIGAddBeginCallback(igi);
}
catch(...)
{
delete igi;
throw;
}
}
///===================================================================================
void CIGCallback::addIGWithNumZC(NL3D::UInstanceGroup *ig, sint numZC)
{
H_AUTO_USE(RZ_IGCallback)
// Check the ig is valid.
if(ig == 0)
return;
nlassert(_MoveContainer);
CIGInstance *igi;
try
{
igi = new CIGInstance(ig, this);
igi->numZC(numZC);
_IGInstances.push_back(igi);
ig->setAddRemoveInstanceCallback(igi);
ig->setTransformNameCallback(igi);
ig->setIGAddBeginCallback(igi);
}
catch(...)
{
delete igi;
throw;
}
}
///===================================================================================
void CIGCallback::addIGs(const std::vector &igs)
{
H_AUTO_USE(RZ_IGCallback)
_IGInstances.reserve(_IGInstances.size() + igs.size());
for(uint k = 0; k < igs.size(); ++k)
{
addIG(igs[k]);
}
}
//-----------------------------------------------
// addIGsWithNumZC :
// Add a vector of instance groups with the num ZC associated.
//-----------------------------------------------
void CIGCallback::addIGsWithNumZC(std::vector > &igs)
{
H_AUTO_USE(RZ_IGCallback)
_IGInstances.reserve(_IGInstances.size() + igs.size());
for(uint k = 0; k < igs.size(); ++k)
{
addIGWithNumZC(igs[k].first, igs[k].second);
}
}
///===================================================================================
CIGCallback::CIGInstance::CIGInstance(NL3D::UInstanceGroup *ig, CIGCallback *owner)
: _Owner(owner), _IG(ig), _HasManagedFXs(false)
{
H_AUTO_USE(RZ_IGCallback)
nlassert(owner);
nlassert(ig);
}
///===================================================================================
CIGCallback::CIGInstance::~CIGInstance()
{
H_AUTO_USE(RZ_IGCallback)
releaseMovePrimitives();
if (_HasManagedFXs)
{
CTimedFXManager::getInstance().remove(_ManagedFXHandle);
}
if (_IG)
{
_IG->setAddRemoveInstanceCallback(NULL);
_IG->setTransformNameCallback(NULL);
_IG->setIGAddBeginCallback(NULL);
}
}
///===================================================================================
void CIGCallback::CIGInstance::instanceGroupAdded()
{
H_AUTO_USE(RZ_IGCallback)
// Check the pointer on the IG.
nlassert(_IG);
nlassert(_Owner);
// See what objects need collisions primitives
if (!_MovePrimitives.empty()) return; // already added
std::vector addedPrims;
uint numInstances = _IG->getNumInstance();
for(uint k = 0; k < numInstances; ++k)
{
TPacsPrimMap::iterator pbIt = PacsPrims.find(NLMISC::toLowerAscii(NLMISC::CFile::getFilenameWithoutExtension(_IG->getShapeName(k))));
if (pbIt != PacsPrims.end())
{
// compute orientation and position
NLMISC::CMatrix instanceMatrix;
_IG->getInstanceMatrix(k, instanceMatrix);
NLMISC::CVector pos;
float angle;
NLMISC::CVector scale = _IG->getInstanceScale(k);
NLPACS::UMoveContainer::getPACSCoordsFromMatrix(pos, angle, instanceMatrix);
// insert the matching primitive block
addedPrims.clear();
_Owner->_MoveContainer->addCollisionnablePrimitiveBlock(pbIt->second, 0, 1, &addedPrims, angle, pos, true, scale);
// Yoyo: For each primitive, increment its height, to avoid blocking bugs
for(uint i=0;igetObstacle() && addedPrims[i]->getTriggerType()==NLPACS::UMovePrimitive::NotATrigger)
{
addedPrims[i]->setHeight(addedPrims[i]->getHeight() + RZ_PRIM_ZEXT_BLOCK_AVOIDANCE);
}
}
// for future remove
_MovePrimitives.insert(_MovePrimitives.end(), addedPrims.begin(), addedPrims.end());
}
}
// update additionnal datas from sheets
updateFromSheets();
updateManagedFXs();
// free mem for sheet pointers
NLMISC::contReset(_EntitySheets);
_Owner->notifyIGAdded(_IG);
}
///===================================================================================
void CIGCallback::CIGInstance::instanceGroupRemoved()
{
H_AUTO_USE(RZ_IGCallback)
// If this zone is a ZC.
releaseMovePrimitives();
if (_HasManagedFXs)
{
CTimedFXManager::getInstance().remove(_ManagedFXHandle);
_HasManagedFXs = false;
}
}
///===================================================================================
void CIGCallback::CIGInstance::releaseMovePrimitives()
{
H_AUTO_USE(RZ_IGCallback)
nlassert(_Owner);
// remove all primitives from the move container
nlassert(_Owner->_MoveContainer);
for(TMovePrimitiveVect::iterator it = _MovePrimitives.begin(); it != _MovePrimitives.end(); ++it)
{
_Owner->_MoveContainer->removePrimitive(*it);
}
NLMISC::contReset(_MovePrimitives);
}
///===================================================================================
void CIGCallback::deleteIGs()
{
H_AUTO_USE(RZ_IGCallback)
for(TIGInstanceList::iterator it = _IGInstances.begin(); it != _IGInstances.end(); ++it)
{
delete *it;
}
_IGInstances.clear();
}
///===================================================================================
void CIGCallback::forceAddAll()
{
H_AUTO_USE(RZ_IGCallback)
for(TIGInstanceList::iterator it = _IGInstances.begin(); it != _IGInstances.end(); ++it)
{
(*it)->forceAdd();
}
}
///===================================================================================
void CIGCallback::CIGInstance::startAddingIG(uint numInstances)
{
H_AUTO_USE(RZ_IGCallback)
// make room for sheets ptr
_EntitySheets.resize(numInstances);
std::fill(_EntitySheets.begin(), _EntitySheets.end(), (CEntitySheet *) NULL);
}
///===================================================================================
void CIGCallback::CIGInstance::buildSheetVector()
{
H_AUTO_USE(RZ_IGCallback)
uint numInstances = _IG->getNumInstance();
_EntitySheets.resize(numInstances);
for(uint k = 0; k < numInstances; ++k)
{
_EntitySheets[k] = NULL;
const std::string &name = _IG->getInstanceName(k);
if (NLMISC::nlstricmp(NLMISC::CFile::getExtension(name), "plant") == 0)
{
NLMISC::CSheetId sheetId;
if (sheetId.buildSheetId(name))
{
_EntitySheets[k] = SheetMngr.get(sheetId);
}
}
}
}
///===================================================================================
void CIGCallback::CIGInstance::eraseSheetVector()
{
H_AUTO_USE(RZ_IGCallback)
NLMISC::contReset(_EntitySheets);
}
///===================================================================================
std::string CIGCallback::CIGInstance::transformName(uint instanceIndex, const std::string &instanceName, const std::string &shapeName)
{
H_AUTO_USE(RZ_IGCallback)
/** we look if there'is a matching form for this instance name
* If this is the case we can choose the shape we want to instanciate
*/
string ext = NLMISC::CFile::getExtension(instanceName);
if (NLMISC::nlstricmp(ext, "pacs_prim") == 0)
{
return ""; // Don't instanciate pacs_prim
}
if (NLMISC::nlstricmp(ext, "plant") != 0)
{
return shapeName; // if there's no attached sheet, use the shape name
}
// We cache the last id
static std::string lastName;
static CEntitySheet *lastSheet = NULL;
CEntitySheet *sh;
if (instanceName == lastName)
{
sh = lastSheet;
}
else
{
NLMISC::CSheetId sheetId;
if (sheetId.buildSheetId(instanceName))
{
sh = SheetMngr.get(sheetId);
lastSheet = sh;
lastName = instanceName;
}
else
{
return shapeName;
}
}
if (!sh) return shapeName;
if (sh->type() == CEntitySheet::PLANT)
{
// store sheet in the sheet list
_EntitySheets[instanceIndex] = sh;
return ((CPlantSheet *) sh)->getShapeName();
}
else
{
return shapeName;
}
}
///===================================================================================
void CIGCallback::CIGInstance::updateFromSheets()
{
H_AUTO_USE(RZ_IGCallback)
nlassert(_EntitySheets.size() == _IG->getNumInstance());
// See for which objects distance should be overriden (object which use a .PLANT sheet)
uint numInstances = (uint)_EntitySheets.size();
for(uint k = 0; k < numInstances; ++k)
{
if (_EntitySheets[k] && _EntitySheets[k]->Type == CEntitySheet::PLANT)
{
CPlantSheet *ps = NLMISC::safe_cast(_EntitySheets[k]);
if (ps->getMaxDist() != -1) _IG->setDistMax(k, ps->getMaxDist());
if (ps->getCoarseMeshDist() != -1) _IG->setCoarseMeshDist(k, ps->getCoarseMeshDist());
}
}
}
///===================================================================================
void CIGCallback::CIGInstance::shutDownFXs()
{
H_AUTO_USE(RZ_IGCallback)
if (!_HasManagedFXs) return;
CTimedFXManager::getInstance().shutDown(_ManagedFXHandle);
_HasManagedFXs = false;
}
///===================================================================================
void CIGCallback::CIGInstance::updateManagedFXs()
{
H_AUTO_USE(RZ_IGCallback)
nlassert(_EntitySheets.size() == _IG->getNumInstance());
// See for which objects distance should be overriden (object which use a .PLANT sheet)
uint numInstances = (uint)_EntitySheets.size();
// vector of fx that should be managed by the dedicated manager. static for malloc perf
static std::vector timedFXs;
timedFXs.clear();
for(uint k = 0; k < numInstances; ++k)
{
if (_EntitySheets[k] && _EntitySheets[k]->Type == CEntitySheet::PLANT)
{
CPlantSheet *ps = NLMISC::safe_cast(_EntitySheets[k]);
// check for managed fxs for the current season
if (CurrSeason < EGSPD::CSeason::Invalid)
{
if (!ps->getFXSheet(CurrSeason).FXName.empty())
{
timedFXs.push_back(CTimedFX());
timedFXs.back().SpawnPosition = _IG->getInstancePos(k);
timedFXs.back().Rot = _IG->getInstanceRot(k);
timedFXs.back().Scale = _IG->getInstanceScale(k);
timedFXs.back().FXSheet = &ps->getFXSheet(CurrSeason);
}
}
}
}
if (!timedFXs.empty())
{
_ManagedFXHandle = CTimedFXManager::getInstance().add(timedFXs, CurrSeason);
_HasManagedFXs = true;
}
}
///===================================================================================
bool CIGCallback::enumIGs(IIGEnum *callback)
{
H_AUTO_USE(RZ_IGCallback)
nlassert(callback);
for(TIGInstanceList::iterator it = _IGInstances.begin(); it != _IGInstances.end(); ++it)
{
if ((*it)->getIG() != NULL && (*it)->getIG() != (NL3D::UInstanceGroup *)-1)
{
bool res = callback->enumIG((*it)->getIG());
if (!res) return false;
}
}
return true;
}
///===================================================================================
void CIGCallback::changeSeason()
{
H_AUTO_USE(RZ_IGCallback)
// for now, this only update managed fxs so that they are displayed the same way on both clients
for(TIGInstanceList::iterator it = _IGInstances.begin(); it != _IGInstances.end(); ++it)
{
if ((*it)->getIG() != NULL && (*it)->getIG() != (NL3D::UInstanceGroup *)-1)
{
(*it)->buildSheetVector(); // the sheet vector is deleted after use, so need to rebuild it
(*it)->shutDownFXs();
(*it)->updateManagedFXs();
(*it)->eraseSheetVector();
}
}
}
///===================================================================================
///===================================================================================
void createInstancesFromMoveContainer(NL3D::UScene *scene, NLPACS::UMoveContainer *mc, std::vector *instances /*=NULL*/)
{
H_AUTO_USE(RZ_IGCallback)
nlassert(scene);
nlassert(mc);
mc->evalCollision(0.1f, 0);
// get the primitives
std::vector prims;
mc->getPrimitives(prims);
if (instances)
{
instances->reserve(prims.size());
}
for(uint k = 0; k < prims.size(); ++k)
{
NL3D::UInstance newInstance;
float angle = 0.f;
NLMISC::CVector scale;
switch(prims[k]->getPrimitiveType())
{
case NLPACS::UMovePrimitive::_2DOrientedBox:
newInstance = scene->createInstance("unit_box.shape");
angle = (float) prims[k]->getOrientation(prims[k]->getFirstWorldImageV());
prims[k]->getSize(scale.x, scale.y);
break;
case NLPACS::UMovePrimitive::_2DOrientedCylinder:
newInstance = scene->createInstance("unit_cylinder.shape");
scale.x = scale.y = prims[k]->getRadius();
break;
default:
nlwarning("createInstancesFromMoveContainer : unsupported type encountered");
continue;
break;
}
if (!newInstance.empty())
{
scale.z = prims[k]->getHeight();
NLMISC::CVectorD worldPos = prims[k]->getFinalPosition(prims[k]->getFirstWorldImageV());
newInstance.setPos(NLMISC::CVector((float) worldPos.x, (float) worldPos.y, (float) worldPos.z));
newInstance.setScale(scale);
newInstance.setRotQuat(NLMISC::CQuat(NLMISC::CVector::K, angle));
if (instances)
{
instances->push_back(newInstance);
}
}
else
{
nlwarning("createInstancesFromMoveContainer : couldn't create shape");
}
}
}