wxWidgets/include/wx/tracker.h
2008-01-06 18:01:28 +00:00

140 lines
3.4 KiB
C++

/////////////////////////////////////////////////////////////////////////////
// Name: wx/tracker.h
// Purpose: Support class for object lifetime tracking (wxWeakRef<T>)
// Author: Arne Steinarson
// Created: 2007-12-28
// RCS-ID: $Id:$
// Copyright: (c) 2007 Arne Steinarson
// Licence: wxWindows licence
/////////////////////////////////////////////////////////////////////////////
#ifndef _WX_TRACKER_H_
#define _WX_TRACKER_H_
// This structure represents an object tracker and is stored in a linked list
// in the tracked object. It is only used in one of its derived forms.
struct wxTrackerNode
{
wxTrackerNode() : m_next(NULL) { }
virtual ~wxTrackerNode() { }
virtual void OnObjectDestroy() = 0;
// This is to tell the difference between different tracker node types.
// It's a replacement of dynamic_cast<> for this class since dynamic_cast
// may be disabled (and we don't have wxDynamicCast since wxTrackerNode
// is not derived wxObject).
enum wxTrackerNodeType { WeakRef, EventConnectionRef };
virtual wxTrackerNodeType GetType() = 0;
protected:
wxTrackerNode *m_next;
friend class wxTrackableBase; // For list access
friend class wxEvtHandler; // For list access
};
// Add-on base class for a trackable object.
struct wxTrackableBase
{
wxTrackableBase() : m_first(NULL) { }
~wxTrackableBase()
{
// Notify all registered refs
// OnObjectDestroy has to remove the item from the link chain
while ( m_first )
{
wxTrackerNode *first = m_first;
first->OnObjectDestroy();
RemoveNode(first);
}
}
void AddNode(wxTrackerNode *node)
{
node->m_next = m_first;
m_first = node;
}
void RemoveNode(wxTrackerNode *node)
{
for ( wxTrackerNode **pn = &m_first; *pn; pn = &(*pn)->m_next )
{
if ( *pn == node )
{
*pn = node->m_next;
return;
}
}
wxFAIL_MSG( "node should be present" );
}
wxTrackerNode *GetFirst() { return m_first; }
protected:
wxTrackerNode *m_first;
};
// The difference to wxTrackableBase is that this class adds
// a VTable to enable dynamic_cast query for wxTrackable.
struct wxTrackable : wxTrackableBase
{
virtual ~wxTrackable() { }
};
// Helper to decide if an object has a base class or not
// (strictly speaking, this test succeeds if a type is convertible
// to another type in some way.)
template <class T, class B>
struct wxHasBase
{
static char Match(B *pb);
static int Match(...);
enum { value = sizeof(Match((T*)NULL)) == sizeof(char) };
};
// A structure to cast to wxTrackableBase, using either static_cast<> or
// dynamic_cast<>.
template <class T, bool is_static>
struct wxTrackableCaster;
template <class T>
struct wxTrackableCaster<T, true>
{
static wxTrackableBase* Cast(T* pt)
{
return static_cast<wxTrackableBase *>(pt);
}
};
#ifdef HAVE_DYNAMIC_CAST
template <class T>
struct wxTrackableCaster<T, false>
{
static wxTrackableBase *Cast(T* pt)
{
return dynamic_cast<wxTrackableBase *>(pt);
}
};
#else // !HAVE_DYNAMIC_CAST
template <class T>
struct wxTrackableCaster<T, false>
{
static wxTrackableBase *Cast(T* pt) { return NULL; }
};
#endif // HAVE_DYNAMIC_CAST/!HAVE_DYNAMIC_CAST
#endif // _WX_TRACKER_H_