2014-10-08 10:58:52 +02:00
|
|
|
#ifndef AISTATE_H
|
|
|
|
#define AISTATE_H
|
|
|
|
|
|
|
|
#include <typeinfo>
|
|
|
|
|
2014-10-10 23:28:49 +02:00
|
|
|
namespace MWMechanics
|
|
|
|
{
|
2014-10-08 10:58:52 +02:00
|
|
|
|
2014-10-10 23:28:49 +02:00
|
|
|
/** \brief stores one object of any class derived from Base.
|
2015-06-04 20:11:40 +02:00
|
|
|
* Requesting a certain derived class via get() either returns
|
2014-10-10 23:28:49 +02:00
|
|
|
* the stored object if it has the correct type or otherwise replaces
|
|
|
|
* it with an object of the requested type.
|
|
|
|
*/
|
|
|
|
template< class Base >
|
|
|
|
class DerivedClassStorage
|
|
|
|
{
|
|
|
|
private:
|
|
|
|
Base* mStorage;
|
2014-10-08 10:58:52 +02:00
|
|
|
|
2014-10-10 23:28:49 +02:00
|
|
|
//if needed you have to provide a clone member function
|
|
|
|
DerivedClassStorage( const DerivedClassStorage& other );
|
|
|
|
DerivedClassStorage& operator=( const DerivedClassStorage& );
|
|
|
|
|
|
|
|
public:
|
|
|
|
/// \brief returns reference to stored object or deletes it and creates a fitting
|
|
|
|
template< class Derived >
|
|
|
|
Derived& get()
|
|
|
|
{
|
|
|
|
Derived* result = dynamic_cast<Derived*>(mStorage);
|
|
|
|
|
|
|
|
if(!result)
|
|
|
|
{
|
|
|
|
if(mStorage)
|
|
|
|
delete mStorage;
|
|
|
|
mStorage = result = new Derived();
|
|
|
|
}
|
|
|
|
|
|
|
|
//return a reference to the (new allocated) object
|
|
|
|
return *result;
|
|
|
|
}
|
2020-03-10 10:28:23 +04:00
|
|
|
|
|
|
|
template< class Derived >
|
|
|
|
void copy(DerivedClassStorage& destination) const
|
|
|
|
{
|
|
|
|
Derived* result = dynamic_cast<Derived*>(mStorage);
|
|
|
|
if (result != nullptr)
|
|
|
|
destination.store<Derived>(*result);
|
|
|
|
}
|
2014-10-08 10:58:52 +02:00
|
|
|
|
2014-10-10 23:28:49 +02:00
|
|
|
template< class Derived >
|
|
|
|
void store( const Derived& payload )
|
2014-10-08 10:58:52 +02:00
|
|
|
{
|
|
|
|
if(mStorage)
|
|
|
|
delete mStorage;
|
2014-10-10 23:28:49 +02:00
|
|
|
mStorage = new Derived(payload);
|
2014-10-08 10:58:52 +02:00
|
|
|
}
|
|
|
|
|
2014-10-10 23:28:49 +02:00
|
|
|
/// \brief takes ownership of the passed object
|
|
|
|
template< class Derived >
|
|
|
|
void moveIn( Derived* p )
|
|
|
|
{
|
|
|
|
if(mStorage)
|
|
|
|
delete mStorage;
|
|
|
|
mStorage = p;
|
|
|
|
}
|
2014-10-08 10:58:52 +02:00
|
|
|
|
2014-10-10 23:28:49 +02:00
|
|
|
bool empty() const
|
|
|
|
{
|
2018-10-09 10:21:12 +04:00
|
|
|
return mStorage == nullptr;
|
2014-10-10 23:28:49 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
const std::type_info& getType() const
|
|
|
|
{
|
|
|
|
return typeid(mStorage);
|
|
|
|
}
|
|
|
|
|
2018-10-09 10:21:12 +04:00
|
|
|
DerivedClassStorage():mStorage(nullptr){}
|
2014-10-10 23:28:49 +02:00
|
|
|
~DerivedClassStorage()
|
|
|
|
{
|
|
|
|
if(mStorage)
|
|
|
|
delete mStorage;
|
2015-06-04 20:11:40 +02:00
|
|
|
}
|
2014-10-08 10:58:52 +02:00
|
|
|
};
|
|
|
|
|
2014-10-10 23:28:49 +02:00
|
|
|
|
2014-10-10 00:15:01 +02:00
|
|
|
/// \brief base class for the temporary storage of AiPackages.
|
|
|
|
/**
|
|
|
|
* Each AI package with temporary values needs a AiPackageStorage class
|
2014-12-31 18:41:57 +01:00
|
|
|
* which is derived from AiTemporaryBase. The Actor holds a container
|
2014-10-10 00:15:01 +02:00
|
|
|
* AiState where one of these storages can be stored at a time.
|
|
|
|
* The execute(...) member function takes this container as an argument.
|
|
|
|
* */
|
2014-10-08 10:58:52 +02:00
|
|
|
struct AiTemporaryBase
|
|
|
|
{
|
2015-06-04 20:11:40 +02:00
|
|
|
virtual ~AiTemporaryBase(){}
|
2014-10-08 10:58:52 +02:00
|
|
|
};
|
|
|
|
|
2014-10-10 00:15:01 +02:00
|
|
|
/// \brief Container for AI package status.
|
2014-10-08 10:58:52 +02:00
|
|
|
typedef DerivedClassStorage<AiTemporaryBase> AiState;
|
|
|
|
}
|
|
|
|
|
|
|
|
#endif // AISTATE_H
|