00001 
00002 
00003 
00004 
00005 
00006 
00007 
00008 
00009 
00010 
00011 
00012 
00013 
00014 
00015 
00016 
00017 
00018 
00019 
00020 
00021 
00022 
00023 
00024 
00025 
00026 
00027 
00028 
00029 
00030 
00031 
00032 
00033 
00034 
00035 
00036 
00037 
00038 
00039 
00040 
00041 
00042 
00043 
00044 
00045 
00046 
00047 
00048 
00049 
00050 #ifndef  _SO_SUB_ELEMENT_
00051 #define  _SO_SUB_ELEMENT_
00052 
00053 #include <Inventor/elements/SoElement.h>
00054 
00055 #if defined(_DEBUG)
00056 #include <Inventor/SoDB.h>
00057 #endif
00058 
00060 
00061 
00062 
00063 
00064 
00066 
00067 
00068 
00069 
00070 
00071 
00072 
00073 
00074 
00075 #define SO_ELEMENT_ABSTRACT_HEADER(className)                                 \
00076   public:                                                                     \
00077                        \
00078     static SoType       getClassTypeId();                                     \
00079                                                                         \
00083     static int          getClassStackIndex();                                 \
00084   private:                                                          \
00085                            \
00086     static SoElement::SoElementKeyType getClassElementKey();                  \
00087     virtual SoElement::SoElementKeyType getElementKey() const;                \
00088     virtual int         getClassStackIndexInternal() const;                   \
00089   private:                                                                  \
00090     className();                                                              \
00091   private:                                                                    \
00092     static int         classStackIndex;                                       \
00093     static SoType      classTypeId;                                           \
00094     static SoElement::SoElementKeyType      s_classElementKey;
00095 
00096 
00097 
00098 
00099 #define SO_ELEMENT_HEADER(className)                                          \
00100     SO_ELEMENT_ABSTRACT_HEADER(className);                                    \
00101   private:                                                                    \
00102     static void *createInstance(SoType* dynamicType = NULL)
00103 
00105 
00106 
00107 
00108 
00110 
00111 
00112 
00113 
00114 
00115 #define SO__ELEMENT_ABSTRACT_VARS(className)                                  \
00116   SoType  className::classTypeId;                         \
00117   int className::classStackIndex = 0;                     \
00118   SoElement::SoElementKeyType className::s_classElementKey = SoElement::badElementKey();
00119 
00120 #define SO__ELEMENT_VARS(className)                                           \
00121 SO__ELEMENT_ABSTRACT_VARS(className)
00122 
00123 #if defined(_DEBUG)
00124 #define SO_ELEMENT_CHECK_INIT(className) { \
00125   if (classTypeId.isBad()) { \
00126     SoDebugError::post("Element Constructor", \
00127                        "Can't construct an element of type " \
00128                        SO__QUOTE(className) \
00129                        " until initClass() has been called"); \
00130     className::initClass(); \
00131   } \
00132   SoTypedObject::checkDatabase(SO__QUOTE(className), this, className::getClassTypeId(), classTypeId); \
00133 }
00134 #else
00135 #define SO_ELEMENT_CHECK_INIT(className) { \
00136   if (classTypeId.isBad()) { \
00137     className::initClass(); \
00138   } \
00139 }
00140 #endif
00141 
00142 
00143 
00144 
00145 #define SO__ELEMENT_ABSTRACT_METHODS(className)                               \
00146                                                                               \
00147 className::className()                                                        \
00148 {                                                                             \
00149   SO_ELEMENT_CHECK_INIT(className);                                    \
00150   commonInit() ;                                                              \
00151 }                                                                             \
00152                                                                               \
00153 SoType                                                                        \
00154 className::getClassTypeId()                                                   \
00155 {                                                                             \
00156   return classTypeId;                                                         \
00157 }                                                                             \
00158                                                                               \
00159 int                                                                           \
00160 className::getClassStackIndex()                                               \
00161 {                                                                             \
00162   return classStackIndex;                                                     \
00163 }                                                                             \
00164                                                                               \
00165 int                                                                           \
00166 className::getClassStackIndexInternal() const                                 \
00167 {                                                                             \
00168 return className::classStackIndex;                                            \
00169 }                                                                             \
00170                                                                               \
00171 SoElement::SoElementKeyType                                                   \
00172 className::getClassElementKey()                                               \
00173 {                                                                             \
00174 return className::s_classElementKey;                                          \
00175 }                                                                             \
00176                                                                               \
00177 SoElement::SoElementKeyType                                                   \
00178 className::getElementKey() const                                              \
00179 {                                                                             \
00180 return className::s_classElementKey;                                          \
00181 }
00182 
00183 
00184 
00185 
00186 
00187 
00188 #define SO__ELEMENT_METHODS(className)                                        \
00189                                                                               \
00190 className::className()                                                        \
00191 {                                                                             \
00192     SO_ELEMENT_CHECK_INIT(className);                                  \
00193     commonInit() ;                                                            \
00194     setTypeId(classTypeId);                                                   \
00195     setStackIndex(classStackIndex);                                           \
00196 }                                                                             \
00197                                                                               \
00198 void *                                                                        \
00199 className::createInstance(SoType *)                                           \
00200 {                                                                             \
00201   return new className;                                                       \
00202 }                                                                             \
00203                                                                               \
00204 SoType                                                                        \
00205 className::getClassTypeId()                                                   \
00206 {                                                                             \
00207   return classTypeId;                                                         \
00208 }                                                                             \
00209                                                                               \
00210 int                                                                           \
00211 className::getClassStackIndex()                                               \
00212 {                                                                             \
00213   return classStackIndex;                                                     \
00214 }                                                                             \
00215                                                                               \
00216 int                                                                           \
00217 className::getClassStackIndexInternal() const                                 \
00218 {                                                                             \
00219 return className::classStackIndex;                                            \
00220 }                                                                             \
00221                                                                               \
00222 SoElement::SoElementKeyType                                                   \
00223 className::getClassElementKey()                                               \
00224 {                                                                             \
00225   return className::s_classElementKey;                                        \
00226 }                                                                             \
00227                                                                               \
00228 SoElement::SoElementKeyType                                                   \
00229 className::getElementKey() const                                              \
00230 {                                                                             \
00231    return className::s_classElementKey;                                       \
00232 }
00233 
00234 
00236 
00237 
00238 
00239 
00240 
00241 #define SO_ELEMENT_ABSTRACT_SOURCE(className)                                 \
00242     SO__ELEMENT_ABSTRACT_VARS(className);                                     \
00243     SO__ELEMENT_ABSTRACT_METHODS(className)
00244 
00245 #define SO_ELEMENT_SOURCE(className)                                          \
00246     SO__ELEMENT_VARS(className);                                              \
00247     SO__ELEMENT_METHODS(className)
00248 
00250 
00251 
00252 
00253 
00254 
00255 
00256 #if defined(_DEBUG)
00257 #define SO_ELEMENT_INIT_CLASS_CHECK_PARENT(className, parentClass)                   \
00258   if (parentClass::getClassTypeId().isBad()) {                                        \
00259     SoDebugError::post( SO__QUOTE(className)"::initClass",                            \
00260       SO__QUOTE(className)" initialized before parent class " \
00261       SO__QUOTE(parentClass)"\n");                                                    \
00262     parentClass::initClass();                                                         \
00263   }
00264 #else
00265 #define SO_ELEMENT_INIT_CLASS_CHECK_PARENT(className, parentClass)                   \
00266   if (parentClass::getClassTypeId().isBad())                                          \
00267     parentClass::initClass()
00268 #endif
00269 
00270 #define SO_ELEMENT_INIT_ABSTRACT_CLASS_INTERNAL(className,parentClass) \
00271     SO_ELEMENT_INIT_CLASS_CHECK_PARENT(className, parentClass); \
00272     classTypeId = SoType::createType( parentClass::getClassTypeId(), \
00273                                       SO__QUOTE(className), \
00274                                       NULL); \
00275     classStackIndex = parentClass::getClassStackIndex(); \
00276     if(s_classElementKey == SoElement::badElementKey()) \
00277       s_classElementKey = createElementKey();
00278 
00279 #define SO_ELEMENT_INIT_ABSTRACT_CLASS(className, parentClass) \
00280   SO_ELEMENT_INIT_ABSTRACT_CLASS_INTERNAL(className, parentClass);
00281 
00282 #define SO_ELEMENT_INIT_CLASS_INTERNAL(className,parentClass) \
00283     SO_ELEMENT_INIT_CLASS_CHECK_PARENT(className, parentClass); \
00284     classTypeId = SoType::createType( parentClass::getClassTypeId(), \
00285                                       SO__QUOTE(className), \
00286                                       &className::createInstance); \
00287     if (classStackIndex == 0) \
00288     { \
00289       if (parentClass::getClassStackIndex() < 0) \
00290         classStackIndex = createStackIndex(classTypeId); \
00291       else \
00292         classStackIndex = parentClass::getClassStackIndex(); \
00293     } \
00294     if(s_classElementKey == SoElement::badElementKey()) \
00295       s_classElementKey = createElementKey();
00296 
00297 #define SO_ELEMENT_INIT_CLASS(className, parentClass) \
00298   SO_ELEMENT_INIT_CLASS_INTERNAL(className, parentClass);
00299 
00300 #if defined(_DEBUG)
00301 #define SO_ELEMENT_EXIT_CLASS(className) \
00302   if (! SoType::removeType(classTypeId.getName())) { \
00303     SoDebugError::post(SO__QUOTE(className)"::exitClass", \
00304                        "Unable to remove type (%s) for this class. Check exitClass() " \
00305                        "method is implemented and is called only once.\n", \
00306                        classTypeId.getName().getString() ); \
00307   } \
00308   else { \
00309     classTypeId = SoType::badType(); \
00310     classStackIndex = 0; \
00311   }
00312 #else
00313 #define SO_ELEMENT_EXIT_CLASS(className) \
00314   SoType::removeType(classTypeId.getName()); \
00315   classTypeId = SoType::badType(); \
00316   classStackIndex = 0
00317 #endif
00318 
00319 #endif 
00320 
00321 
00322