1 ///////////////////////////////////////////////////////////////////////////// 
   3 // Purpose:     wxObject implementation 
   4 // Author:      Julian Smart 
   8 // Copyright:   (c) Julian Smart and Markus Holzem 
   9 // Licence:     wxWindows license 
  10 ///////////////////////////////////////////////////////////////////////////// 
  13 #pragma implementation "object.h" 
  16 // For compilers that support precompilation, includes "wx.h". 
  17 #include "wx/wxprec.h" 
  26         #include "wx/objstrm.h" 
  27         #include "wx/serbase.h" 
  32     #endif // wxUSE_SERIAL 
  38 #if (defined(__WXDEBUG__) && wxUSE_MEMORY_TRACING) || wxUSE_DEBUG_CONTEXT 
  39 #include "wx/memory.h" 
  42 #if defined(__WXDEBUG__) || wxUSE_DEBUG_CONTEXT 
  44     #include "wx/ioswrap.h" 
  45     #if defined(__VISAGECPP__) 
  46     // help with VA debugging 
  47         #define DEBUG_PRINTF(NAME)   { static int raz=0; \ 
  48           printf( #NAME " %i\n",raz); fflush(stdout);       \ 
  52         #define DEBUG_PRINTF(NAME) 
  56 wxClassInfo 
wxObject::sm_classwxObject((wxChar 
*) wxT("wxObject"), (wxChar 
*) NULL
, (wxChar 
*) NULL
, (int ) sizeof(wxObject
), (wxObjectConstructorFn
) NULL
); 
  57 wxClassInfo
* wxClassInfo::sm_first 
= (wxClassInfo 
*) NULL
; 
  58 wxHashTable
* wxClassInfo::sm_classTable 
= (wxHashTable
*) NULL
; 
  60 #if defined(__WXDEBUG__) && defined(__VISAGECPP__) 
  61   int wxObject::N 
= 0;  // total number of objects 
  62   int wxObject::Nid 
= 0;// object serial counter 
  65 // These are here so we can avoid 'always true/false' warnings 
  66 // by referring to these instead of TRUE/FALSE 
  67 const bool wxTrue 
= TRUE
; 
  68 const bool wxFalse 
= FALSE
; 
  71  * wxWindows root object. 
  76     m_refData 
= (wxObjectRefData 
*) NULL
; 
  78     m_serialObj 
= (wxObject_Serialize 
*)NULL
; 
  80 #if defined(__WXDEBUG__) && defined(__VISAGECPP__) 
  83 //  {  printf("wxObject %i/%i \t",id,N); 
  95 #if defined(__WXDEBUG__) && defined(__VISAGECPP__) 
  97 //  {  printf("~wxObject %i/%i \t",id,N); 
 103  * Is this object a kind of (a subclass of) 'info'? 
 104  * E.g. is wxWindow a kind of wxObject? 
 105  * Go from this class to superclass, taking into account 
 106  * two possible base classes. 
 109 bool wxObject::IsKindOf(wxClassInfo 
*info
) const 
 111     wxClassInfo 
*thisInfo 
= GetClassInfo(); 
 113         return thisInfo
->IsKindOf(info
); 
 118 wxObject 
*wxObject::Clone() const 
 120     wxObject 
*object 
= GetClassInfo()->CreateObject(); 
 126 void wxObject::CopyObject(wxObject
& object_dest
) const 
 128 void wxObject::CopyObject(wxObject
& WXUNUSED(object_dest
)) const 
 129 #endif // Debug/!Debug 
 131     wxASSERT(object_dest
.GetClassInfo()->IsKindOf(GetClassInfo())); 
 134 #if wxUSE_STD_IOSTREAM && (defined(__WXDEBUG__) || wxUSE_DEBUG_CONTEXT) 
 135 void wxObject::Dump(ostream
& str
) 
 137     if (GetClassInfo() && GetClassInfo()->GetClassName()) 
 138         str 
<< GetClassInfo()->GetClassName(); 
 140         str 
<< "unknown object class"; 
 144 #if defined(__WXDEBUG__) && wxUSE_MEMORY_TRACING 
 150 void *wxObject::operator new (size_t size
, wxChar 
* fileName
, int lineNum
) 
 152     return wxDebugAlloc(size
, fileName
, lineNum
, TRUE
); 
 155 #if defined(__VISAGECPP__) 
 157 void wxObject::operator delete (void * buf
,const char * _fname
, size_t _line
) 
 161 #  endif  //__DEBUG_ALLOC__ 
 163 void wxObject::operator delete (void * buf
) 
 167 #endif // __VISAGECPP__ 
 170 #if defined(__VISUALC__) && (__VISUALC__ >= 1200) 
 171 void wxObject::operator delete(void* pData
, wxChar
* /* fileName */, int /* lineNum */) 
 173     ::operator delete(pData
); 
 177 // Cause problems for VC++ - crashes 
 178 #if (!defined(__VISUALC__) && wxUSE_ARRAY_MEMORY_OPERATORS ) || defined(__MWERKS__) 
 179 void * wxObject::operator new[] (size_t size
, wxChar 
* fileName
, int lineNum
) 
 181     return wxDebugAlloc(size
, fileName
, lineNum
, TRUE
, TRUE
); 
 184 void wxObject::operator delete[] (void * buf
) 
 186     wxDebugFree(buf
, TRUE
); 
 193  * Class info: provides run-time class type information. 
 196 wxClassInfo::wxClassInfo(wxChar 
*cName
, wxChar 
*baseName1
, wxChar 
*baseName2
, int sz
, wxObjectConstructorFn constr
) 
 199     m_baseClassName1 
= baseName1
; 
 200     m_baseClassName2 
= baseName2
; 
 203     m_objectConstructor 
= constr
; 
 208     m_baseInfo1 
= (wxClassInfo 
*) NULL
; 
 209     m_baseInfo2 
= (wxClassInfo 
*) NULL
; 
 212 wxObject 
*wxClassInfo::CreateObject() 
 214     if (m_objectConstructor
) 
 215         return (wxObject 
*)(*m_objectConstructor
)(); 
 217         return (wxObject 
*) NULL
; 
 220 wxClassInfo 
*wxClassInfo::FindClass(wxChar 
*c
) 
 222     wxClassInfo 
*p 
= sm_first
; 
 225         if (p 
&& p
->GetClassName() && wxStrcmp(p
->GetClassName(), c
) == 0) 
 229     return (wxClassInfo 
*) NULL
; 
 232 // Climb upwards through inheritance hierarchy. 
 233 // Dual inheritance is catered for. 
 234 bool wxClassInfo::IsKindOf(wxClassInfo 
*info
) const 
 243         if (m_baseInfo1
->IsKindOf(info
)) 
 247         return m_baseInfo2
->IsKindOf(info
); 
 252 // Set pointers to base class(es) to speed up IsKindOf 
 253 void wxClassInfo::InitializeClasses() 
 255     // using IMPLEMENT_DYNAMIC_CLASS() macro twice (which may happen if you 
 256     // link any object module twice mistakenly) will break this function 
 257     // because it will enter an infinite loop and eventually die with "out of 
 258     // memory" - as this is quite hard to detect if you're unaware of this, 
 259     // try to do some checks here 
 261     // more classes than we'll ever have 
 262     static const size_t nMaxClasses 
= 10000; 
 266     wxClassInfo::sm_classTable 
= new wxHashTable(wxKEY_STRING
); 
 268     // Index all class infos by their class name 
 269     wxClassInfo 
*info 
= sm_first
; 
 272         if (info
->m_className
) 
 274             wxASSERT_MSG( ++nClass 
< nMaxClasses
, 
 275                           _T("an infinite loop detected - have you used IMPLEMENT_DYNAMIC_CLASS() twice (may be by linking some object module(s) twice)?") ); 
 277             sm_classTable
->Put(info
->m_className
, (wxObject 
*)info
); 
 283     // Set base pointers for each wxClassInfo 
 287         if (info
->GetBaseClassName1()) 
 288             info
->m_baseInfo1 
= (wxClassInfo 
*)sm_classTable
->Get(info
->GetBaseClassName1()); 
 289         if (info
->GetBaseClassName2()) 
 290             info
->m_baseInfo2 
= (wxClassInfo 
*)sm_classTable
->Get(info
->GetBaseClassName2()); 
 295 void wxClassInfo::CleanUpClasses() 
 297     delete wxClassInfo::sm_classTable
; 
 298     wxClassInfo::sm_classTable 
= NULL
; 
 301 wxObject 
*wxCreateDynamicObject(const wxChar 
*name
) 
 303 #if defined(__WXDEBUG__) || wxUSE_DEBUG_CONTEXT 
 304  DEBUG_PRINTF(wxObject 
*wxCreateDynamicObject
) 
 307     if (wxClassInfo::sm_classTable
) 
 309         wxClassInfo 
*info 
= (wxClassInfo 
*)wxClassInfo::sm_classTable
->Get(name
); 
 311             return (wxObject 
*)NULL
; 
 313         return info
->CreateObject(); 
 317         wxClassInfo 
*info 
= wxClassInfo::sm_first
; 
 320             if (info
->m_className 
&& wxStrcmp(info
->m_className
, name
) == 0) 
 321                 return info
->CreateObject(); 
 324         return (wxObject
*) NULL
; 
 330 #include "wx/serbase.h" 
 331 #include "wx/dynlib.h" 
 333 wxObject
* wxCreateStoredObject( wxInputStream 
&stream 
) 
 335     wxObjectInputStream 
obj_s(stream
); 
 336     return obj_s
.LoadObject(); 
 339 void wxObject::StoreObject( wxObjectOutputStream
& stream 
) 
 341 #if defined(__WXDEBUG__) || wxUSE_DEBUG_CONTEXT 
 342  DEBUG_PRINTF(wxObject::StoreObject
) 
 345     wxString obj_name 
= wxString(GetClassInfo()->GetClassName()) + "_Serialize"; 
 346     wxLibrary 
*lib 
= wxTheLibraries
.LoadLibrary("wxserial"); 
 349         wxLogError(_("Can't load wxSerial dynamic library.")); 
 353         m_serialObj 
= (WXSERIAL(wxObject
) *)lib
->CreateObject( obj_name 
); 
 356             wxLogError(_("Can't find the serialization object '%s' " 
 357                         "for the object '%s'."), 
 359                     GetClassInfo()->GetClassName()); 
 362         m_serialObj
->SetObject(this); 
 365     m_serialObj
->StoreObject(stream
); 
 368 void wxObject::LoadObject( wxObjectInputStream
& stream 
) 
 370 #if defined(__WXDEBUG__) || wxUSE_DEBUG_CONTEXT 
 371  DEBUG_PRINTF(wxObject::LoadObject
) 
 374     wxString obj_name 
= wxString(GetClassInfo()->GetClassName()) + "_Serialize"; 
 375     wxLibrary 
*lib 
= wxTheLibraries
.LoadLibrary("wxserial"); 
 378         m_serialObj 
= (WXSERIAL(wxObject
) *)lib
->CreateObject( obj_name 
); 
 381             wxLogError(_("Can't find the serialization object '%s' " 
 382                         "for the object '%s'."), 
 384                     GetClassInfo()->GetClassName()); 
 387         m_serialObj
->SetObject(this); 
 390     m_serialObj
->LoadObject(stream
); 
 393 #endif // wxUSE_SERIAL 
 396  * wxObject: cloning of objects 
 399 void wxObject::Ref(const wxObject
& clone
) 
 401 #if defined(__WXDEBUG__) || wxUSE_DEBUG_CONTEXT 
 402  DEBUG_PRINTF(wxObject::Ref
) 
 404      // delete reference to old data 
 406     // reference new data 
 407     if (clone
.m_refData
) { 
 408         m_refData 
= clone
.m_refData
; 
 409         ++(m_refData
->m_count
); 
 413 void wxObject::UnRef() 
 417         wxASSERT_MSG( m_refData
->m_count 
> 0, _T("invalid ref data count") ); 
 419         if ( !--m_refData
->m_count 
) 
 421         m_refData 
= (wxObjectRefData 
*) NULL
; 
 429 wxObjectRefData::wxObjectRefData(void) : m_count(1) 
 433 wxObjectRefData::~wxObjectRefData()