Added ResourceConstListener

I don't know yet if it's a good idea but at least it's an idea


Former-commit-id: b034a787d8d8ccce0804f1276958b32c9e149147
This commit is contained in:
Lynix 2015-01-07 23:47:09 +01:00
parent b6c5668232
commit 88af94b987
4 changed files with 134 additions and 0 deletions

View File

@ -17,6 +17,7 @@
#include <Nazara/Core/ThreadSafetyOff.hpp> #include <Nazara/Core/ThreadSafetyOff.hpp>
#endif #endif
class NzResourceConstListener;
class NzResourceListener; class NzResourceListener;
class NAZARA_API NzResource class NAZARA_API NzResource
@ -25,6 +26,7 @@ class NAZARA_API NzResource
NzResource(bool persistent = true); NzResource(bool persistent = true);
virtual ~NzResource(); virtual ~NzResource();
void AddResourceConstListener(const NzResourceConstListener* listener, int index = 0) const;
void AddResourceListener(NzResourceListener* listener, int index = 0) const; void AddResourceListener(NzResourceListener* listener, int index = 0) const;
void AddResourceReference() const; void AddResourceReference() const;
@ -32,6 +34,7 @@ class NAZARA_API NzResource
bool IsPersistent() const; bool IsPersistent() const;
void RemoveResourceConstListener(const NzResourceConstListener* listener) const;
void RemoveResourceListener(NzResourceListener* listener) const; void RemoveResourceListener(NzResourceListener* listener) const;
bool RemoveResourceReference() const; bool RemoveResourceReference() const;
@ -43,13 +46,16 @@ class NAZARA_API NzResource
void NotifyModified(unsigned int code); void NotifyModified(unsigned int code);
private: private:
using ResourceConstListenerMap = std::unordered_map<const NzResourceConstListener*, std::pair<int, unsigned int>>;
using ResourceListenerMap = std::unordered_map<NzResourceListener*, std::pair<int, unsigned int>>; using ResourceListenerMap = std::unordered_map<NzResourceListener*, std::pair<int, unsigned int>>;
void RemoveResourceConstListenerIterator(ResourceConstListenerMap::iterator iterator) const;
void RemoveResourceListenerIterator(ResourceListenerMap::iterator iterator) const; void RemoveResourceListenerIterator(ResourceListenerMap::iterator iterator) const;
NazaraMutexAttrib(m_mutex, mutable) NazaraMutexAttrib(m_mutex, mutable)
// Je fais précéder le nom par 'resource' pour éviter les éventuels conflits de noms // Je fais précéder le nom par 'resource' pour éviter les éventuels conflits de noms
mutable ResourceConstListenerMap m_resourceConstListeners;
mutable ResourceListenerMap m_resourceListeners; mutable ResourceListenerMap m_resourceListeners;
std::atomic_bool m_resourcePersistent; std::atomic_bool m_resourcePersistent;
mutable std::atomic_uint m_resourceReferenceCount; mutable std::atomic_uint m_resourceReferenceCount;

View File

@ -0,0 +1,26 @@
// Copyright (C) 2015 Jérôme Leclercq
// This file is part of the "Nazara Engine - Core module"
// For conditions of distribution and use, see copyright notice in Config.hpp
#pragma once
#ifndef NAZARA_RESOURCECONSTLISTENER_HPP
#define NAZARA_RESOURCECONSTLISTENER_HPP
#include <Nazara/Prerequesites.hpp>
class NzResource;
class NAZARA_API NzResourceConstListener
{
public:
NzResourceConstListener() = default;
virtual ~NzResourceConstListener();
virtual bool OnResourceCreated(const NzResource* resource, int index) const;
virtual bool OnResourceDestroy(const NzResource* resource, int index) const;
virtual bool OnResourceModified(const NzResource* resource, int index, unsigned int code) const;
virtual void OnResourceReleased(const NzResource* resource, int index) const;
};
#endif // NAZARA_RESOURCECONSTLISTENER_HPP

View File

@ -5,6 +5,7 @@
#include <Nazara/Core/Resource.hpp> #include <Nazara/Core/Resource.hpp>
#include <Nazara/Core/Config.hpp> #include <Nazara/Core/Config.hpp>
#include <Nazara/Core/Error.hpp> #include <Nazara/Core/Error.hpp>
#include <Nazara/Core/ResourceConstListener.hpp>
#include <Nazara/Core/ResourceListener.hpp> #include <Nazara/Core/ResourceListener.hpp>
#if NAZARA_CORE_THREADSAFE && NAZARA_THREADSAFETY_RESOURCE #if NAZARA_CORE_THREADSAFE && NAZARA_THREADSAFETY_RESOURCE
@ -34,6 +35,19 @@ NzResource::~NzResource()
#endif #endif
} }
void NzResource::AddResourceConstListener(const NzResourceConstListener* listener, int index) const
{
///DOC: Est ignoré si appelé depuis un évènement
NazaraLock(m_mutex)
if (!m_resourceListenersLocked)
{
auto pair = m_resourceConstListeners.insert(std::make_pair(listener, std::make_pair(index, 1U)));
if (!pair.second)
pair.first->second.second++;
}
}
void NzResource::AddResourceListener(NzResourceListener* listener, int index) const void NzResource::AddResourceListener(NzResourceListener* listener, int index) const
{ {
///DOC: Est ignoré si appelé depuis un évènement ///DOC: Est ignoré si appelé depuis un évènement
@ -62,6 +76,19 @@ bool NzResource::IsPersistent() const
return m_resourcePersistent; return m_resourcePersistent;
} }
void NzResource::RemoveResourceConstListener(const NzResourceConstListener* listener) const
{
///DOC: Est ignoré si appelé depuis un évènement
NazaraLock(m_mutex);
if (!m_resourceListenersLocked)
{
ResourceConstListenerMap::iterator it = m_resourceConstListeners.find(listener);
if (it != m_resourceConstListeners.end())
RemoveResourceConstListenerIterator(it);
}
}
void NzResource::RemoveResourceListener(NzResourceListener* listener) const void NzResource::RemoveResourceListener(NzResourceListener* listener) const
{ {
///DOC: Est ignoré si appelé depuis un évènement ///DOC: Est ignoré si appelé depuis un évènement
@ -115,6 +142,15 @@ void NzResource::NotifyCreated()
m_resourceListenersLocked = true; m_resourceListenersLocked = true;
auto constIt = m_resourceConstListeners.begin();
while (constIt != m_resourceConstListeners.end())
{
if (!constIt->first->OnResourceCreated(this, constIt->second.first))
RemoveResourceConstListenerIterator(constIt++);
else
++constIt;
}
auto it = m_resourceListeners.begin(); auto it = m_resourceListeners.begin();
while (it != m_resourceListeners.end()) while (it != m_resourceListeners.end())
{ {
@ -133,6 +169,15 @@ void NzResource::NotifyDestroy()
m_resourceListenersLocked = true; m_resourceListenersLocked = true;
auto constIt = m_resourceConstListeners.begin();
while (constIt != m_resourceConstListeners.end())
{
if (!constIt->first->OnResourceDestroy(this, constIt->second.first))
RemoveResourceConstListenerIterator(constIt++);
else
++constIt;
}
auto it = m_resourceListeners.begin(); auto it = m_resourceListeners.begin();
while (it != m_resourceListeners.end()) while (it != m_resourceListeners.end())
{ {
@ -151,6 +196,15 @@ void NzResource::NotifyModified(unsigned int code)
m_resourceListenersLocked = true; m_resourceListenersLocked = true;
auto constIt = m_resourceConstListeners.begin();
while (constIt != m_resourceConstListeners.end())
{
if (!constIt->first->OnResourceModified(this, constIt->second.first, code))
RemoveResourceConstListenerIterator(constIt++);
else
++constIt;
}
auto it = m_resourceListeners.begin(); auto it = m_resourceListeners.begin();
while (it != m_resourceListeners.end()) while (it != m_resourceListeners.end())
{ {
@ -163,6 +217,15 @@ void NzResource::NotifyModified(unsigned int code)
m_resourceListenersLocked = false; m_resourceListenersLocked = false;
} }
void NzResource::RemoveResourceConstListenerIterator(ResourceConstListenerMap::iterator iterator) const
{
unsigned int& referenceCount = iterator->second.second;
if (referenceCount == 1)
m_resourceConstListeners.erase(iterator);
else
referenceCount--;
}
void NzResource::RemoveResourceListenerIterator(ResourceListenerMap::iterator iterator) const void NzResource::RemoveResourceListenerIterator(ResourceListenerMap::iterator iterator) const
{ {
unsigned int& referenceCount = iterator->second.second; unsigned int& referenceCount = iterator->second.second;

View File

@ -0,0 +1,39 @@
// Copyright (C) 2014 Jérôme Leclercq
// This file is part of the "Nazara Engine - Core module"
// For conditions of distribution and use, see copyright notice in Config.hpp
#include <Nazara/Core/ResourceConstListener.hpp>
#include <Nazara/Core/Debug.hpp>
NzResourceConstListener::~NzResourceConstListener() = default;
bool NzResourceConstListener::OnResourceCreated(const NzResource* resource, int index) const
{
NazaraUnused(resource);
NazaraUnused(index);
return true;
}
bool NzResourceConstListener::OnResourceDestroy(const NzResource* resource, int index) const
{
NazaraUnused(resource);
NazaraUnused(index);
return true;
}
bool NzResourceConstListener::OnResourceModified(const NzResource* resource, int index, unsigned int code) const
{
NazaraUnused(resource);
NazaraUnused(index);
NazaraUnused(code);
return true;
}
void NzResourceConstListener::OnResourceReleased(const NzResource* resource, int index) const
{
NazaraUnused(resource);
NazaraUnused(index);
}