added wxWeakRef<T> (slightly modified patch 1860953)
git-svn-id: https://svn.wxwidgets.org/svn/wx/wxWidgets/trunk@51042 c3d73ce0-8a6f-49c7-b76d-6d57e0e08775
This commit is contained in:
@@ -529,6 +529,7 @@ IMPORTANT: please read docs/tech/tn0016.txt before modifying this file!
|
|||||||
wx/thrimpl.cpp
|
wx/thrimpl.cpp
|
||||||
wx/timer.h
|
wx/timer.h
|
||||||
wx/tokenzr.h
|
wx/tokenzr.h
|
||||||
|
wx/tracker.h
|
||||||
wx/txtstrm.h
|
wx/txtstrm.h
|
||||||
wx/types.h
|
wx/types.h
|
||||||
wx/unichar.h
|
wx/unichar.h
|
||||||
@@ -538,6 +539,7 @@ IMPORTANT: please read docs/tech/tn0016.txt before modifying this file!
|
|||||||
wx/vector.h
|
wx/vector.h
|
||||||
wx/version.h
|
wx/version.h
|
||||||
wx/volume.h
|
wx/volume.h
|
||||||
|
wx/weakref.h
|
||||||
wx/wfstream.h
|
wx/wfstream.h
|
||||||
wx/wx.h
|
wx/wx.h
|
||||||
wx/wxchar.h
|
wx/wxchar.h
|
||||||
|
139
include/wx/tracker.h
Normal file
139
include/wx/tracker.h
Normal file
@@ -0,0 +1,139 @@
|
|||||||
|
/////////////////////////////////////////////////////////////////////////////
|
||||||
|
// 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_
|
||||||
|
|
89
include/wx/weakref.h
Normal file
89
include/wx/weakref.h
Normal file
@@ -0,0 +1,89 @@
|
|||||||
|
/////////////////////////////////////////////////////////////////////////////
|
||||||
|
// Name: wx/weakref.h
|
||||||
|
// Purpose: wxWeakRef - Generic weak references for wxWidgets
|
||||||
|
// Author: Arne Steinarson
|
||||||
|
// Created: 2007-12-27
|
||||||
|
// RCS-ID: $Id:$
|
||||||
|
// Copyright: (c) 2007 Arne Steinarson
|
||||||
|
// Licence: wxWindows licence
|
||||||
|
/////////////////////////////////////////////////////////////////////////////
|
||||||
|
|
||||||
|
#ifndef _WX_WEAKREF_H_
|
||||||
|
#define _WX_WEAKREF_H_
|
||||||
|
|
||||||
|
#include <wx/tracker.h>
|
||||||
|
|
||||||
|
// A weak reference to an object of type T, where T has type wxTrackable
|
||||||
|
// as one of its base classes (in a static or dynamic sense).
|
||||||
|
template<class T>
|
||||||
|
class wxWeakRef : public wxTrackerNode
|
||||||
|
{
|
||||||
|
public:
|
||||||
|
wxWeakRef(T *pobj = NULL) : m_pobj(NULL) { Assign(pobj); }
|
||||||
|
|
||||||
|
virtual ~wxWeakRef() { Assign(NULL); }
|
||||||
|
|
||||||
|
// Smart pointer functions
|
||||||
|
operator T*(){ return m_pobj; }
|
||||||
|
T* operator->(){ return m_pobj; }
|
||||||
|
T* operator=(T *pobj)
|
||||||
|
{
|
||||||
|
Assign(pobj);
|
||||||
|
return m_pobj;
|
||||||
|
}
|
||||||
|
|
||||||
|
virtual void OnObjectDestroy()
|
||||||
|
{
|
||||||
|
// Tracked object itself removes us from list of trackers
|
||||||
|
wxASSERT( m_pobj );
|
||||||
|
m_pobj = NULL;
|
||||||
|
}
|
||||||
|
|
||||||
|
virtual wxTrackerNodeType GetType() { return WeakRef; }
|
||||||
|
|
||||||
|
protected:
|
||||||
|
wxTrackableBase *GetTrackable(T *pobj)
|
||||||
|
{
|
||||||
|
// this uses static_cast if possible or dynamic_cast otherwise
|
||||||
|
return wxTrackableCaster<T, wxHasBase<T, wxTrackableBase>::value>
|
||||||
|
::Cast(pobj);
|
||||||
|
}
|
||||||
|
|
||||||
|
void Assign(T* pobj)
|
||||||
|
{
|
||||||
|
if ( m_pobj == pobj )
|
||||||
|
return;
|
||||||
|
|
||||||
|
// First release old object if any
|
||||||
|
if ( m_pobj )
|
||||||
|
{
|
||||||
|
// Remove ourselves from object tracker list
|
||||||
|
GetTrackable(m_pobj)->RemoveNode(this);
|
||||||
|
m_pobj = NULL;
|
||||||
|
}
|
||||||
|
|
||||||
|
// Now set new trackable object
|
||||||
|
if ( pobj )
|
||||||
|
{
|
||||||
|
wxTrackableBase * const pt = GetTrackable(pobj);
|
||||||
|
wxCHECK_RET( pt, "type must derive from wxTrackableBase" );
|
||||||
|
|
||||||
|
pt->AddNode(this);
|
||||||
|
m_pobj = pobj;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
T *m_pobj;
|
||||||
|
};
|
||||||
|
|
||||||
|
// Provide some basic types of weak references
|
||||||
|
class WXDLLIMPEXP_FWD_BASE wxObject;
|
||||||
|
class WXDLLIMPEXP_FWD_BASE wxEvtHandler;
|
||||||
|
class WXDLLIMPEXP_FWD_CORE wxWindow;
|
||||||
|
|
||||||
|
typedef wxWeakRef<wxObject> wxObjectRef;
|
||||||
|
typedef wxWeakRef<wxEvtHandler> wxEvtHandlerRef;
|
||||||
|
typedef wxWeakRef<wxWindow> wxWindowRef;
|
||||||
|
|
||||||
|
#endif // _WX_WEAKREF_H_
|
||||||
|
|
Reference in New Issue
Block a user