Coin3D is Free Software,
published under the
BSD 3-clause license
.
https://bitbucket.org/Coin3D/
http://www.kongsberg.com/kogt/
SoSubNode.h
1
#ifndef COIN_SOSUBNODE_H
2
#define COIN_SOSUBNODE_H
3
4
/**************************************************************************\
5
* Copyright (c) Kongsberg Oil & Gas Technologies AS
6
* All rights reserved.
7
*
8
* Redistribution and use in source and binary forms, with or without
9
* modification, are permitted provided that the following conditions are
10
* met:
11
*
12
* Redistributions of source code must retain the above copyright notice,
13
* this list of conditions and the following disclaimer.
14
*
15
* Redistributions in binary form must reproduce the above copyright
16
* notice, this list of conditions and the following disclaimer in the
17
* documentation and/or other materials provided with the distribution.
18
*
19
* Neither the name of the copyright holder nor the names of its
20
* contributors may be used to endorse or promote products derived from
21
* this software without specific prior written permission.
22
*
23
* THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
24
* "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
25
* LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR
26
* A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT
27
* HOLDER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
28
* SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
29
* LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
30
* DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
31
* THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
32
* (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
33
* OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
34
\**************************************************************************/
35
36
/*
37
The most used macros and their arguments:
38
39
header files:
40
SO_NODE_HEADER(classname)
41
SO_NODE_ABSTRACT_HEADER(classname)
42
43
source files:
44
SO_NODE_SOURCE(classname)
45
SO_NODE_ABSTRACT_SOURCE(classname)
46
in constructor:
47
SO_NODE_CONSTRUCTOR(classname)
48
SO_NODE_DEFINE_ENUM_VALUE(enumname, enumvalue)
49
SO_NODE_ADD_FIELD(fieldname, (defaultvalue))
50
in initClass():
51
SO_NODE_INIT_CLASS(classname, parentclassname, parentclassname)
52
SO_NODE_INIT_ABSTRACT_CLASS(classname, parentclassname, parentclassname)
53
*/
54
55
// *************************************************************************
56
57
#include <cstring>
/* strcmp used in assert() */
58
#include <Inventor/SbBasic.h>
59
#include <Inventor/SbName.h>
60
#include <
Inventor/SoType.h
>
61
#include <Inventor/fields/SoFieldData.h>
62
#include <Inventor/nodes/SoNode.h>
63
#include <
Inventor/C/tidbits.h
>
64
65
// *************************************************************************
66
67
#define PRIVATE_NODE_TYPESYSTEM_HEADER( ) \
68
public: \
69
static SoType getClassTypeId(void); \
70
virtual SoType getTypeId(void) const; \
71
private: \
72
static SoType classTypeId
73
74
// FIXME: document. 20000103 mortene.
75
#define SO_NODE_ABSTRACT_HEADER(_class_) \
76
PRIVATE_NODE_TYPESYSTEM_HEADER(); \
77
protected: \
78
static const SoFieldData ** getFieldDataPtr(void); \
79
virtual const SoFieldData * getFieldData(void) const; \
80
private: \
81
static void atexit_cleanup(void); \
82
static const SoFieldData ** parentFieldData; \
83
static SoFieldData * fieldData; \
84
/* Counts number of instances of subclasses aswell as "direct" */
\
85
/* instances from non-abstract classes. */
\
86
static unsigned int classinstances
87
88
89
// FIXME: document. 20000103 mortene.
90
#define SO_NODE_HEADER(_class_) \
91
SO_NODE_ABSTRACT_HEADER(_class_); \
92
private: \
93
static void * createInstance(void)
94
95
// *************************************************************************
96
97
#define PRIVATE_NODE_TYPESYSTEM_SOURCE(_class_) \
98
SoType _class_::getClassTypeId(void) { return _class_::classTypeId; } \
99
SoType _class_::getTypeId(void) const { return _class_::classTypeId; } \
100
SoType _class_::classTypeId STATIC_SOTYPE_INIT
101
102
103
// FIXME: document. 20000103 mortene.
104
#define SO_NODE_ABSTRACT_SOURCE(_class_) \
105
PRIVATE_NODE_TYPESYSTEM_SOURCE(_class_); \
106
\
107
unsigned int _class_::classinstances = 0; \
108
const SoFieldData ** _class_::parentFieldData = NULL; \
109
SoFieldData * _class_::fieldData = NULL; \
110
\
111
const SoFieldData ** \
112
_class_::getFieldDataPtr(void) \
113
{ \
114
return const_cast<const SoFieldData **>(&_class_::fieldData); \
115
} \
116
\
117
const SoFieldData * \
118
_class_::getFieldData(void) const \
119
{ \
120
return _class_::fieldData; \
121
} \
122
\
123
void \
124
_class_::atexit_cleanup(void) \
125
{ \
126
delete _class_::fieldData; \
127
_class_::fieldData = NULL; \
128
_class_::parentFieldData = NULL; \
129
SoType::removeType(_class_::classTypeId.getName()); \
130
_class_::classTypeId STATIC_SOTYPE_INIT; \
131
_class_::classinstances = 0; \
132
}
133
134
// FIXME: document. 20000103 mortene.
135
#define SO_NODE_SOURCE(_class_) \
136
SO_NODE_ABSTRACT_SOURCE(_class_) \
137
\
138
\
141
void * \
142
_class_::createInstance(void) \
143
{ \
144
return new _class_; \
145
}
146
147
// *************************************************************************
148
149
// FIXME: document. 20000103 mortene.
150
#define SO_NODE_IS_FIRST_INSTANCE() \
151
(classinstances == 1)
152
153
// FIXME: document. 20000103 mortene.
154
#define SO_NODE_CONSTRUCTOR_NOLOCK(_class_) \
155
do { \
156
_class_::classinstances++; \
157
/* Catch attempts to use a node class which has not been initialized. */
\
158
assert(_class_::classTypeId != SoType::badType() && "you forgot init()!"); \
159
/* Initialize a fielddata container for the class only once. */
\
160
if (!_class_::fieldData) { \
161
/* FIXME: this is a "static" memory leak. 20030131 mortene. */
\
162
/* Internal Coin nodes are handled properly though. pederb, 20041122 */
\
163
_class_::fieldData = \
164
new SoFieldData(_class_::parentFieldData ? \
165
*_class_::parentFieldData : NULL); \
166
} \
167
/* Extension classes from the application programmers should not be \
168
considered native. This is important to get the export code to do \
169
the Right Thing. */
\
170
this->isBuiltIn = FALSE; \
171
} WHILE_0
172
173
// FIXME: document. 20000103 mortene.
174
#define SO_NODE_CONSTRUCTOR(_class_) \
175
do { \
176
SoBase::staticDataLock(); \
177
SO_NODE_CONSTRUCTOR_NOLOCK(_class_); \
178
SoBase::staticDataUnlock(); \
179
} WHILE_0
180
181
// *************************************************************************
182
183
// FIXME: create-type with get-next and inc-next must be an atomic step in an MT
184
// environment 20020216 larsa
185
// FIXME: you can not use the _parentclass_ argument directly as a class specifier
186
// in initClass because such usage causes dynamic loading to fail when trying to
187
// invoke the initClass method in the case where _parentclass_ also has to be
188
// dynamically loaded. that is the reason for this roundabout way of getting the
189
// parent class type. 20020216 larsa
190
191
#define PRIVATE_COMMON_INIT_CODE(_class_, _classname_, _createfunc_, _parentclass_) \
192
do { \
193
/* Make sure we only initialize once. */
\
194
assert(_class_::classTypeId == SoType::badType() && "don't init() twice!"); \
195
/* Make sure superclass gets initialized before subclass. */
\
196
assert(strcmp(SO__QUOTE(_parentclass_), "inherited")); \
197
SoType parentType(SoType::fromName(SO__QUOTE(_parentclass_))); \
198
assert(parentType != SoType::badType() && "you forgot init() on parentclass!"); \
199
\
200
/* Set up entry in the type system. */
\
201
_class_::classTypeId = \
202
SoType::createType(parentType, \
203
_classname_, \
204
_createfunc_, \
205
(uint16_t)SoNode::getNextActionMethodIndex()); \
206
SoNode::incNextActionMethodIndex(); \
207
\
208
/* Store parent's fielddata pointer for later use in the constructor. */
\
209
_class_::parentFieldData = _parentclass_::getFieldDataPtr(); \
210
\
211
/* Make sure also external nodes are cleaned up */
\
212
cc_coin_atexit_static_internal(_class_::atexit_cleanup); \
213
} WHILE_0
214
215
216
// FIXME: document. 20000103 mortene.
217
#define SO_NODE_INIT_CLASS(_class_, _parentclass_, _parentname_) \
218
do { \
219
const char * classname = SO__QUOTE(_class_); \
220
PRIVATE_COMMON_INIT_CODE(_class_, classname, &_class_::createInstance, _parentclass_); \
221
} WHILE_0
222
223
// FIXME: document. 20000103 mortene.
224
#define SO_NODE_INIT_ABSTRACT_CLASS(_class_, _parentclass_, _parentname_) \
225
do { \
226
const char * classname = SO__QUOTE(_class_); \
227
PRIVATE_COMMON_INIT_CODE(_class_, classname, NULL, _parentclass_); \
228
} WHILE_0
229
230
231
// *************************************************************************
232
233
// FIXME: document. 20000103 mortene.
234
#define SO_NODE_ADD_FIELD(_field_, _defaultval_) \
235
do { \
236
this->_field_.setValue _defaultval_;\
237
this->_field_.setContainer(this); \
238
fieldData->addField(this, SO__QUOTE(_field_), &this->_field_); \
239
} WHILE_0
240
241
// New for Coin-3
242
#define SO_NODE_ADD_EMPTY_MFIELD(_field_) \
243
do { \
244
this->_field_.setContainer(this); \
245
fieldData->addField(this, SO__QUOTE(_field_), &this->_field_);\
246
} WHILE_0
247
248
249
// FIXME: document. 20000103 mortene.
250
#define SO_NODE_DEFINE_ENUM_VALUE(_enumname_, _enumval_) \
251
do { \
252
fieldData->addEnumValue(SO__QUOTE(_enumname_), \
253
SO__QUOTE(_enumval_), _enumval_); \
254
} WHILE_0
255
256
// *************************************************************************
257
258
#endif // !COIN_SOSUBNODE_H
tidbits.h
SoType.h
include
Inventor
nodes
SoSubNode.h
Generated on Thu Oct 18 2018 20:22:52 for Coin by
doxygen
1.8.14