zjf
2023-03-06 392b76515f40376b6d36f40a114850ef63650384
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
/////////////////////////////////////////////////////////////////////////////// 
// Copyright (C) 2002-2016, Open Design Alliance (the "Alliance"). 
// All rights reserved. 
// 
// This software and its documentation and related materials are owned by 
// the Alliance. The software may only be incorporated into application 
// programs owned by members of the Alliance, subject to a signed 
// Membership Agreement and Supplemental Software License Agreement with the
// Alliance. The structure and organization of this software are the valuable  
// trade secrets of the Alliance and its suppliers. The software is also 
// protected by copyright law and international treaty provisions. Application  
// programs incorporating this software must include the following statement 
// with their copyright notices:
//   
//   This application incorporates Teigha(R) software pursuant to a license 
//   agreement with Open Design Alliance.
//   Teigha(R) Copyright (C) 2002-2016 by Open Design Alliance. 
//   All rights reserved.
//
// By use of this software, its documentation or related materials, you 
// acknowledge and accept the above terms.
///////////////////////////////////////////////////////////////////////////////
 
 
 
 
#ifndef _ODSMARTCLSIDPTR_INCLUDED_
#define _ODSMARTCLSIDPTR_INCLUDED_
 
#include "TD_PackPush.h"
#include "SmartPtr.h"
 
class OdRxObject;
 
#define ODRX_DEFINE_CLSID_RETRIEVER(Class) \
class OdRxClsIdHolderFor_##Class\
{\
public:\
  static inline const OdRxClass* classId()\
  {\
    OdRxClass* pClass = static_cast<OdRxClass*>(::odrxClassDictionary()->getAt(OD_T(#Class)).get());\
    if(!pClass)\
      throw OdError(eNotInitializedYet);\
    return pClass;\
  }\
}
 
#define ODRX_DEFINE_CLSID_SMARTPTR(Class) \
ODRX_DEFINE_CLSID_RETRIEVER(Class);\
typedef OdClsIdSmartPtr<OdRxClsIdHolderFor_##Class, Class> Class##ClsIdPtr
 
 
/** \details
    This template class implements SmartPointers for Class ID objects.
    This class is the Teigha ClassIdSmartPointer template class.
 
    <group Other_Classes>
*/
template <class ClsIdRetriever, class T> class OdClsIdSmartPtr : public OdBaseObjectPtr
{
  /** \details
      Increments the reference count for the object referenced by this SmartPointer object.
  */
  inline void internalAddRef() { if(m_pObject) { static_cast<T*>(m_pObject)->addRef(); } }
  
  /** \details
    Assigns the specified object to this SmartPointer object.  
      
    \remarks
    The reference count of the specified object is incremented.
 
    If this SmartPointer is currently referencing another object, that object 
    is released prior to the assignment.  
 
    \param pObject [in]  Pointer to the object to be assigned.
  */
  inline void assign(const T* pObject)
  {
    release();
    m_pObject = (OdRxObject*)pObject;
    internalAddRef();
  }
  
  /** \details
    Performs a "safe" assignment of the specified object to this SmartPointer object. 
      
    \param pObject [in]  Pointer to the object to be assigned.
 
    Throws:
    eNotThatKindOfClass if not successful. 
  */
  inline void internalQueryX(const OdRxObject* pObject)
  {
    if(pObject)
    {
      OdRxObject* pX = pObject->queryX(ClsIdRetriever::classId());
      if(pX)
        m_pObject = pX;
      else
        throw OdError(eNotThatKindOfClass);
    }
  }
  
  inline void assign(const OdRxObject* pObject)
  {
    release();
    internalQueryX(pObject);
  }
  
  // Note: Using of SmartPtr<T> as bool expression produce ambiguous call with some compilers.
  // Use isNull() method instead.
 
  /** \details
      Declared private to prevent use. 
  */
  bool operator !() const { ODA_FAIL(); return false; }
  
  /** \details
      Declared private to prevent use. 
  */
  operator bool() const { ODA_FAIL(); return false; }
 
  /** \details
      Declared private to prevent use. 
  */
  operator bool() { ODA_FAIL(); return false; }
  
public:
  /** \details
    \param pObject [in]  Pointer to the object to be assigned to the new SmartPointer object.
 
    \remarks
    If pObject is specified, the specified object is assigned to this SmartPointer object.
    
    \remarks
    If OdRxObjMod or const OdBaseObjectPtr& are specified, the reference count of the referenced object 
    is not incremented. 
  */
  inline OdClsIdSmartPtr() { }
  
  inline OdClsIdSmartPtr(const T* pObject, OdRxObjMod) {m_pObject = (OdRxObject*)pObject; }
  
  inline OdClsIdSmartPtr(const T* pObject) {m_pObject = (OdRxObject*)pObject; internalAddRef(); }
  
  inline OdClsIdSmartPtr(const OdRxObject* pObject) { internalQueryX(pObject); }
  
  /** \details
    Returns the OdRxClass of the object referenced by this SmartPointer object.
  */
  static inline const OdRxClass* classId()
  {
    return ClsIdRetriever::classId();
  }
 
  inline OdClsIdSmartPtr(OdRxObject* pObject, OdRxObjMod)
  {
    internalQueryX(pObject);
    if(pObject)
      pObject->release();
  }
  
  inline OdClsIdSmartPtr(const OdClsIdSmartPtr& pObject)
  {
    m_pObject = const_cast<T*>(static_cast<const T*>(pObject.get()));
    internalAddRef();
  }
 
  inline OdClsIdSmartPtr(const OdRxObjectPtr& pObject) { internalQueryX(pObject.get()); }
  
  inline OdClsIdSmartPtr(const OdBaseObjectPtr& pObject) { internalQueryX(pObject.get()); }
 
  static inline OdClsIdSmartPtr cast(const OdRxObject* pObject)
  {
    OdClsIdSmartPtr pRes;
    if (pObject)
      pRes.attach(static_cast<T*>(pObject->queryX(ClsIdRetriever::classId())));
    return pRes;
  }
  
  /** \details
    Assigns the specified object to this SmartPointer object.  
      
    \param pObject [in]  Pointer to the object to be assigned.
 
    \remarks
    The reference count of the specified object is not incremented.
 
    If this SmartPointer is currently referencing another object, that object 
    is released prior to the assignment.
  */
  inline void attach(const T* pObject) { release(); m_pObject = const_cast<T*>(pObject); }
  
  inline void attach(OdRxObject* pObject)
  {
    release();
    internalQueryX(pObject);
    if(pObject)
      pObject->release();
  }
  
  /** \remarks
    Decrements the reference count of the object referenced by this
    SmartPointer object.
 
    When the reference count reaches zero, the referenced object is deleted.
  */
  inline ~OdClsIdSmartPtr() { release(); }
  
  /** \details
    Releases this SmartPointer's reference to the referenced object.
    
    \remarks
    Decrements the reference count of the referenced object. 
    
    When the reference count reaches zero, the referenced object is deleted.
  */
  inline void release()
  {
    if (m_pObject)
    {
      static_cast<T*>(m_pObject)->release();
      m_pObject = 0;
    }
  }
  
  /** \details
    Releases this SmartPointer's reference to the referenced object.
      
    \remarks
    Returns the object referenced by this SmartPointer object.
    
    The referenced object's reference count is not modified.
  */
  inline T* detach()
  {
    T* res = static_cast<T*>(m_pObject);
    m_pObject = 0;
    return res;
  }
  
  inline OdClsIdSmartPtr& operator = (const OdClsIdSmartPtr& pObject)
  { assign(pObject); return *this; }
    
  inline OdClsIdSmartPtr& operator = (const OdBaseObjectPtr& pObject)
  { assign(pObject.get()); return *this; }
  
  inline OdClsIdSmartPtr& operator = (const T* pObject)
  { assign(pObject); return *this; }
  
  /** \details
    Returns the object referenced by this SmartPointer object.  
 
    \remarks
    This SmartPointer maintains its reference to the referenced object.
    
    The reference count of the referenced object is unchanged.
  */
  inline const T* get() const { return const_cast<T*>(static_cast<const T*>(m_pObject)); }
  
  inline T* get() { return static_cast<T*>(m_pObject); }
  
  /** \details
    Returns the object referenced by this SmartPointer object.
      
    \remarks
    The reference count of the referenced object is unchanged.
  */
  inline T* operator ->() { return static_cast<T*>(m_pObject); }
  
  inline const T* operator ->() const { return const_cast<T*>(static_cast<const T*>(m_pObject)); }
  
#ifdef ODA_GCC_2_95
  /** \details
    Returns the object referenced by this SmartPointer object.
    
    \remarks
    This SmartPointer maintains its reference to the referenced object.
    
    The reference count of the referenced object is unchanged.
  */
  inline operator T*() const { return static_cast<T*>(m_pObject); }
  
#else
  /** \details
    Returns the object referenced by this SmartPointer object.
      
    \remarks
    This SmartPointer maintains its reference to the referenced object.
 
    The reference count of the referenced object is unchanged.
  */
  inline operator T*() { return static_cast<T*>(m_pObject); }
  
  /** \details
    Returns the object referenced by this SmartPointer object.
      
    \remarks
    The reference count of the referenced object is unchanged.
  */
  inline operator const T*() const { return static_cast<const T*>(m_pObject); }
#endif
    
  inline bool operator==(const void* p) const { return (m_pObject==p); }
 
  inline bool operator==(const OdClsIdSmartPtr& ptr) const { return operator==((void*)ptr.get()); }
  
  inline bool operator!=(const void* p) const { return (m_pObject!=p); }
 
  inline bool operator!=(const OdClsIdSmartPtr& ptr) const { return operator!=((void*)ptr.get()); }
};
 
#include "TD_PackPop.h"
 
#endif //_ODSMARTCLSIDPTR_INCLUDED_