surface_render.cpp 14.2 KB
Newer Older
Pierre Kraemer's avatar
Pierre Kraemer committed
1 2 3
#include "surface_render.h"

#include "mapHandler.h"
4
#include "slot_debug.h"
Pierre Kraemer's avatar
Pierre Kraemer committed
5 6 7 8 9 10 11

namespace CGoGN
{

namespace SCHNApps
{

12
bool Surface_Render_Plugin::enable()
Pierre Kraemer's avatar
Pierre Kraemer committed
13
{
14
	//	magic line that init static variables of GenericMap in the plugins
15
	GenericMap::copyAllStatics(m_schnapps->getStaticPointers());
16

17 18
	m_dockTab = new Surface_Render_DockTab(m_schnapps, this);
	m_schnapps->addPluginDockTab(this, m_dockTab, "Surface_Render");
Pierre Kraemer's avatar
Pierre Kraemer committed
19 20 21 22 23

	m_flatShader = new CGoGN::Utils::ShaderFlat();
	m_flatShader->setAmbiant(CGoGN::Geom::Vec4f(0.2f, 0.2f, 0.2f, 0.1f));
	m_flatShader->setExplode(1.0f);

24 25 26 27 28 29
	m_phongShader = new CGoGN::Utils::ShaderPhong();
	m_phongShader->setAmbiant(CGoGN::Geom::Vec4f(0.2f, 0.2f, 0.2f, 0.1f));
	m_phongShader->setSpecular(CGoGN::Geom::Vec4f(0.9f, 0.9f, 0.9f, 1.0f));
	m_phongShader->setShininess(80.0f);

	m_colorPerVertexShader = new CGoGN::Utils::ShaderColorPerVertex();
Pierre Kraemer's avatar
Pierre Kraemer committed
30 31 32 33 34 35 36

	m_simpleColorShader = new CGoGN::Utils::ShaderSimpleColor();

	m_pointSprite = new CGoGN::Utils::PointSprite();

	registerShader(m_flatShader);
	registerShader(m_phongShader);
37
	registerShader(m_colorPerVertexShader);
Pierre Kraemer's avatar
Pierre Kraemer committed
38 39 40 41 42 43 44
	registerShader(m_simpleColorShader);
	registerShader(m_pointSprite);

	connect(m_schnapps, SIGNAL(selectedViewChanged(View*, View*)), this, SLOT(selectedViewChanged(View*, View*)));
	connect(m_schnapps, SIGNAL(selectedMapChanged(MapHandlerGen*, MapHandlerGen*)), this, SLOT(selectedMapChanged(MapHandlerGen*, MapHandlerGen*)));
	connect(m_schnapps, SIGNAL(mapAdded(MapHandlerGen*)), this, SLOT(mapAdded(MapHandlerGen*)));
	connect(m_schnapps, SIGNAL(mapRemoved(MapHandlerGen*)), this, SLOT(mapRemoved(MapHandlerGen*)));
45
	connect(m_schnapps, SIGNAL(schnappsClosing()), this, SLOT(schnappsClosing()));
Pierre Kraemer's avatar
Pierre Kraemer committed
46 47 48 49 50 51 52 53 54

	foreach(MapHandlerGen* map, m_schnapps->getMapSet().values())
		mapAdded(map);

	m_dockTab->updateMapParameters();

	return true;
}

55
void Surface_Render_Plugin::disable()
Pierre Kraemer's avatar
Pierre Kraemer committed
56 57 58 59 60 61
{
	delete m_flatShader;
	delete m_phongShader;
	delete m_simpleColorShader;
	delete m_pointSprite;

62 63 64 65
	disconnect(m_schnapps, SIGNAL(selectedViewChanged(View*, View*)), this, SLOT(selectedViewChanged(View*, View*)));
	disconnect(m_schnapps, SIGNAL(selectedMapChanged(MapHandlerGen*, MapHandlerGen*)), this, SLOT(selectedMapChanged(MapHandlerGen*, MapHandlerGen*)));
	disconnect(m_schnapps, SIGNAL(mapAdded(MapHandlerGen*)), this, SLOT(mapAdded(MapHandlerGen*)));
	disconnect(m_schnapps, SIGNAL(mapRemoved(MapHandlerGen*)), this, SLOT(mapRemoved(MapHandlerGen*)));
66
	disconnect(m_schnapps, SIGNAL(schnappsClosing()), this, SLOT(schnappsClosing()));
67

Pierre Kraemer's avatar
Pierre Kraemer committed
68 69 70 71
	foreach(MapHandlerGen* map, m_schnapps->getMapSet().values())
		mapRemoved(map);
}

72
void Surface_Render_Plugin::drawMap(View* view, MapHandlerGen* map)
Pierre Kraemer's avatar
Pierre Kraemer committed
73
{
74 75
	const MapParameters& p = h_viewParameterSet[view][map];
	if(p.positionVBO)
Pierre Kraemer's avatar
Pierre Kraemer committed
76
	{
sylvain thery's avatar
sylvain thery committed
77 78 79 80 81 82 83 84 85
        if(p.renderFaces)
        {
            glPolygonMode(GL_FRONT_AND_BACK, GL_FILL);
            glEnable(GL_LIGHTING);
            glEnable(GL_POLYGON_OFFSET_FILL);
            glPolygonOffset(1.0f, 1.0f);
            switch(p.faceStyle)
            {
                case MapParameters::FLAT :
86 87 88 89 90 91 92 93
					if(p.colorVBO)
					{
						m_colorPerVertexShader->setAttributePosition(p.positionVBO);
						m_colorPerVertexShader->setAttributeColor(p.colorVBO);
						map->draw(m_colorPerVertexShader, CGoGN::Algo::Render::GL2::TRIANGLES);
					}
					else
					{
94
						m_flatShader->setBackColor(p.backColor);
95 96
						m_flatShader->setAttributePosition(p.positionVBO);
						m_flatShader->setDiffuse(p.diffuseColor);
97
						m_flatShader->setDoubleSided(p.renderBackfaces);
98 99
						map->draw(m_flatShader, CGoGN::Algo::Render::GL2::TRIANGLES);
					}
sylvain thery's avatar
sylvain thery committed
100 101 102 103
                    break;
                case MapParameters::PHONG :
                    if(p.normalVBO != NULL)
                    {
104
						m_phongShader->setBackColor(p.backColor);
sylvain thery's avatar
sylvain thery committed
105 106 107
                        m_phongShader->setAttributePosition(p.positionVBO);
                        m_phongShader->setAttributeNormal(p.normalVBO);
                        m_phongShader->setDiffuse(p.diffuseColor);
108
						m_phongShader->setDoubleSided(p.renderBackfaces);
sylvain thery's avatar
sylvain thery committed
109 110 111 112 113 114
                        map->draw(m_phongShader, CGoGN::Algo::Render::GL2::TRIANGLES);
                    }
                    break;
            }
            glDisable(GL_POLYGON_OFFSET_FILL);
        }
115 116

		if(p.renderVertices)
Pierre Kraemer's avatar
Pierre Kraemer committed
117
		{
118
			m_pointSprite->setSize(p.basePSradius * p.verticesScaleFactor); 
119
			m_pointSprite->setAttributePosition(p.positionVBO);
120
			m_pointSprite->setColor(p.vertexColor);
121 122
			map->draw(m_pointSprite, CGoGN::Algo::Render::GL2::POINTS);
		}
123

124 125 126 127
		if(p.renderEdges)
		{
			glLineWidth(1.0f);
			m_simpleColorShader->setAttributePosition(p.positionVBO);
128
			m_simpleColorShader->setColor(p.simpleColor);
129 130
			map->draw(m_simpleColorShader, CGoGN::Algo::Render::GL2::LINES);
		}
sylvain thery's avatar
sylvain thery committed
131

132 133 134 135 136 137 138 139
		if(p.renderBoundary)
		{
			glLineWidth(5.0f);
			CGoGN::Geom::Vec4f c(0.8f, 0.8f, 0.1f, 1.0f);
			m_simpleColorShader->setColor(c);
			m_simpleColorShader->setAttributePosition(p.positionVBO);
			map->draw(m_simpleColorShader, CGoGN::Algo::Render::GL2::BOUNDARY);
		}
Pierre Kraemer's avatar
Pierre Kraemer committed
140 141 142 143 144 145 146
	}
}





147
void Surface_Render_Plugin::selectedViewChanged(View *prev, View *cur)
Pierre Kraemer's avatar
Pierre Kraemer committed
148
{
149
	DEBUG_SLOT();
Pierre Kraemer's avatar
Pierre Kraemer committed
150 151 152
	m_dockTab->updateMapParameters();
}

153
void Surface_Render_Plugin::selectedMapChanged(MapHandlerGen *prev, MapHandlerGen *cur)
Pierre Kraemer's avatar
Pierre Kraemer committed
154
{
155
	DEBUG_SLOT();
Pierre Kraemer's avatar
Pierre Kraemer committed
156 157 158
	m_dockTab->updateMapParameters();
}

159
void Surface_Render_Plugin::mapAdded(MapHandlerGen *map)
Pierre Kraemer's avatar
Pierre Kraemer committed
160
{
161
	DEBUG_SLOT();
Pierre Kraemer's avatar
Pierre Kraemer committed
162 163
	connect(map, SIGNAL(vboAdded(Utils::VBO*)), this, SLOT(vboAdded(Utils::VBO*)));
	connect(map, SIGNAL(vboRemoved(Utils::VBO*)), this, SLOT(vboRemoved(Utils::VBO*)));
164
	connect(map, SIGNAL(boundingBoxModified()), this, SLOT(boundingBoxModified()));
Pierre Kraemer's avatar
Pierre Kraemer committed
165 166
}

167
void Surface_Render_Plugin::mapRemoved(MapHandlerGen *map)
Pierre Kraemer's avatar
Pierre Kraemer committed
168
{
169
	DEBUG_SLOT();
Pierre Kraemer's avatar
Pierre Kraemer committed
170 171
	disconnect(map, SIGNAL(vboAdded(Utils::VBO*)), this, SLOT(vboAdded(Utils::VBO*)));
	disconnect(map, SIGNAL(vboRemoved(Utils::VBO*)), this, SLOT(vboRemoved(Utils::VBO*)));
172
	disconnect(map, SIGNAL(boundingBoxModified()), this, SLOT(boundingBoxModified()));
173 174 175

	if(map == m_schnapps->getSelectedMap())
		m_dockTab->updateMapParameters();
Pierre Kraemer's avatar
Pierre Kraemer committed
176 177 178 179 180 181
}





182
void Surface_Render_Plugin::vboAdded(Utils::VBO *vbo)
Pierre Kraemer's avatar
Pierre Kraemer committed
183
{
184
	DEBUG_SLOT();
Pierre Kraemer's avatar
Pierre Kraemer committed
185 186 187 188 189 190 191 192
	MapHandlerGen* map = static_cast<MapHandlerGen*>(QObject::sender());

	if(map == m_schnapps->getSelectedMap())
	{
		if(vbo->dataSize() == 3)
		{
			m_dockTab->addPositionVBO(QString::fromStdString(vbo->name()));
			m_dockTab->addNormalVBO(QString::fromStdString(vbo->name()));
193
			m_dockTab->addColorVBO(QString::fromStdString(vbo->name()));
Pierre Kraemer's avatar
Pierre Kraemer committed
194 195 196 197
		}
	}
}

198
void Surface_Render_Plugin::vboRemoved(Utils::VBO *vbo)
Pierre Kraemer's avatar
Pierre Kraemer committed
199
{
200
	DEBUG_SLOT();
Pierre Kraemer's avatar
Pierre Kraemer committed
201 202 203 204 205 206 207 208
	MapHandlerGen* map = static_cast<MapHandlerGen*>(QObject::sender());

	if(map == m_schnapps->getSelectedMap())
	{
		if(vbo->dataSize() == 3)
		{
			m_dockTab->removePositionVBO(QString::fromStdString(vbo->name()));
			m_dockTab->removeNormalVBO(QString::fromStdString(vbo->name()));
Pierre Kraemer's avatar
Pierre Kraemer committed
209
			m_dockTab->removeColorVBO(QString::fromStdString(vbo->name()));
Pierre Kraemer's avatar
Pierre Kraemer committed
210 211 212 213 214 215 216 217 218 219 220 221 222 223 224 225 226 227 228 229 230
		}
	}

	QSet<View*> viewsToUpdate;

	QHash<View*, QHash<MapHandlerGen*, MapParameters> >::iterator i;
	for (i = h_viewParameterSet.begin(); i != h_viewParameterSet.end(); ++i)
	{
		View* view = i.key();
		QHash<MapHandlerGen*, MapParameters>& viewParamSet = i.value();
		MapParameters& mapParam = viewParamSet[map];
		if(mapParam.positionVBO == vbo)
		{
			mapParam.positionVBO = NULL;
			if(view->isLinkedToMap(map)) viewsToUpdate.insert(view);
		}
		if(mapParam.normalVBO == vbo)
		{
			mapParam.normalVBO = NULL;
			if(view->isLinkedToMap(map)) viewsToUpdate.insert(view);
		}
Pierre Kraemer's avatar
Pierre Kraemer committed
231 232 233 234 235
		if(mapParam.colorVBO == vbo)
		{
			mapParam.colorVBO = NULL;
			if(view->isLinkedToMap(map)) viewsToUpdate.insert(view);
		}
Pierre Kraemer's avatar
Pierre Kraemer committed
236 237 238 239 240 241
	}

	foreach(View* v, viewsToUpdate)
		v->updateGL();
}

242
void Surface_Render_Plugin::boundingBoxModified()
243
{
244 245
	DEBUG_SLOT();
	MapHandlerGen* map = static_cast<MapHandlerGen*>(QObject::sender());
246

247 248 249 250 251 252
	QList<View*> views = map->getLinkedViews();
	foreach(View* v, views)
	{
		if (h_viewParameterSet.contains(v))
			h_viewParameterSet[v][map].basePSradius = map->getBBdiagSize() / (2 * std::sqrt(map->getNbOrbits(EDGE)));
	}
253
}
Pierre Kraemer's avatar
Pierre Kraemer committed
254

255
void Surface_Render_Plugin::changePositionVBO(const QString& view, const QString& map, const QString& vbo)
Pierre Kraemer's avatar
Pierre Kraemer committed
256
{
257
	DEBUG_SLOT();
Pierre Kraemer's avatar
Pierre Kraemer committed
258 259 260 261 262 263 264 265 266 267 268 269 270 271
	View* v = m_schnapps->getView(view);
	MapHandlerGen* m = m_schnapps->getMap(map);
	if(v && m)
	{
		Utils::VBO* vbuf = m->getVBO(vbo);
		h_viewParameterSet[v][m].positionVBO = vbuf;
		if(v->isSelectedView())
		{
			if(v->isLinkedToMap(m))	v->updateGL();
			if(m->isSelectedMap()) m_dockTab->updateMapParameters();
		}
	}
}

272
void Surface_Render_Plugin::changeNormalVBO(const QString& view, const QString& map, const QString& vbo)
Pierre Kraemer's avatar
Pierre Kraemer committed
273
{
274
	DEBUG_SLOT();
Pierre Kraemer's avatar
Pierre Kraemer committed
275 276 277 278 279 280 281 282 283 284
	View* v = m_schnapps->getView(view);
	MapHandlerGen* m = m_schnapps->getMap(map);
	if(v && m)
	{
		Utils::VBO* vbuf = m->getVBO(vbo);
		h_viewParameterSet[v][m].normalVBO = vbuf;
		if(v->isSelectedView())
		{
			if(v->isLinkedToMap(m))	v->updateGL();
			if(m->isSelectedMap()) m_dockTab->updateMapParameters();
285 286 287 288 289 290 291 292 293 294 295 296 297 298 299 300 301
		}
	}
}

void Surface_Render_Plugin::changeColorVBO(const QString& view, const QString& map, const QString& vbo)
{
	DEBUG_SLOT();
	View* v = m_schnapps->getView(view);
	MapHandlerGen* m = m_schnapps->getMap(map);
	if(v && m)
	{
		Utils::VBO* vbuf = m->getVBO(vbo);
		h_viewParameterSet[v][m].colorVBO = vbuf;
		if(v->isSelectedView())
		{
			if(v->isLinkedToMap(m))	v->updateGL();
			if(m->isSelectedMap()) m_dockTab->updateMapParameters();
Pierre Kraemer's avatar
Pierre Kraemer committed
302 303 304 305
		}
	}
}

306
void Surface_Render_Plugin::changeRenderVertices(const QString& view, const QString& map, bool b)
Pierre Kraemer's avatar
Pierre Kraemer committed
307
{
308
	DEBUG_SLOT();
Pierre Kraemer's avatar
Pierre Kraemer committed
309 310 311 312 313
	View* v = m_schnapps->getView(view);
	MapHandlerGen* m = m_schnapps->getMap(map);
	if(v && m)
	{
		h_viewParameterSet[v][m].renderVertices = b;
314 315
		if (b)
			h_viewParameterSet[v][m].basePSradius = m->getBBdiagSize() / (2 * std::sqrt(m->getNbOrbits(EDGE)));
Pierre Kraemer's avatar
Pierre Kraemer committed
316 317 318 319 320 321 322 323
		if(v->isSelectedView())
		{
			if(v->isLinkedToMap(m))	v->updateGL();
			if(m->isSelectedMap()) m_dockTab->updateMapParameters();
		}
	}
}

324
void Surface_Render_Plugin::changeVerticesScaleFactor(const QString& view, const QString& map, float f)
Pierre Kraemer's avatar
Pierre Kraemer committed
325
{
326
	DEBUG_SLOT();
Pierre Kraemer's avatar
Pierre Kraemer committed
327 328 329 330 331 332 333 334 335 336 337 338 339
	View* v = m_schnapps->getView(view);
	MapHandlerGen* m = m_schnapps->getMap(map);
	if(v && m)
	{
		h_viewParameterSet[v][m].verticesScaleFactor = f;
		if(v->isSelectedView())
		{
			if(v->isLinkedToMap(m))	v->updateGL();
			if(m->isSelectedMap()) m_dockTab->updateMapParameters();
		}
	}
}

340
void Surface_Render_Plugin::changeRenderEdges(const QString& view, const QString& map, bool b)
Pierre Kraemer's avatar
Pierre Kraemer committed
341
{
342
	DEBUG_SLOT();
Pierre Kraemer's avatar
Pierre Kraemer committed
343 344 345 346 347
	View* v = m_schnapps->getView(view);
	MapHandlerGen* m = m_schnapps->getMap(map);
	if(v && m)
	{
		h_viewParameterSet[v][m].renderEdges = b;
348 349
		if (b)
			h_viewParameterSet[v][m].basePSradius = m->getBBdiagSize() / (16 * std::sqrt(m->getNbOrbits(EDGE)));
Pierre Kraemer's avatar
Pierre Kraemer committed
350 351 352 353 354 355 356 357
		if(v->isSelectedView())
		{
			if(v->isLinkedToMap(m))	v->updateGL();
			if(m->isSelectedMap()) m_dockTab->updateMapParameters();
		}
	}
}

358
void Surface_Render_Plugin::changeRenderFaces(const QString& view, const QString& map, bool b)
Pierre Kraemer's avatar
Pierre Kraemer committed
359
{
360
	DEBUG_SLOT();
Pierre Kraemer's avatar
Pierre Kraemer committed
361 362 363 364 365 366 367 368 369 370 371 372 373
	View* v = m_schnapps->getView(view);
	MapHandlerGen* m = m_schnapps->getMap(map);
	if(v && m)
	{
		h_viewParameterSet[v][m].renderFaces = b;
		if(v->isSelectedView())
		{
			if(v->isLinkedToMap(m))	v->updateGL();
			if(m->isSelectedMap()) m_dockTab->updateMapParameters();
		}
	}
}

Sylvain Thery's avatar
Sylvain Thery committed
374
void Surface_Render_Plugin::changeFacesStyle(const QString& view, const QString& map, int iStyle)
Pierre Kraemer's avatar
Pierre Kraemer committed
375
{
Sylvain Thery's avatar
Sylvain Thery committed
376
	MapParameters::FaceShadingStyle style = MapParameters::FaceShadingStyle(iStyle);
377
	DEBUG_SLOT();
Pierre Kraemer's avatar
Pierre Kraemer committed
378 379 380 381 382 383 384 385 386 387 388 389 390
	View* v = m_schnapps->getView(view);
	MapHandlerGen* m = m_schnapps->getMap(map);
	if(v && m)
	{
		h_viewParameterSet[v][m].faceStyle = style;
		if(v->isSelectedView())
		{
			if(v->isLinkedToMap(m))	v->updateGL();
			if(m->isSelectedMap()) m_dockTab->updateMapParameters();
		}
	}
}

391 392
void Surface_Render_Plugin::changeRenderBoundary(const QString& view, const QString& map, bool b)
{
393
	DEBUG_SLOT();
394 395 396 397 398 399 400 401 402 403 404 405 406
	View* v = m_schnapps->getView(view);
	MapHandlerGen* m = m_schnapps->getMap(map);
	if(v && m)
	{
		h_viewParameterSet[v][m].renderBoundary = b;
		if(v->isSelectedView())
		{
			if(v->isLinkedToMap(m))	v->updateGL();
			if(m->isSelectedMap()) m_dockTab->updateMapParameters();
		}
	}
}

407 408
void Surface_Render_Plugin::changeFaceColor(const QString& view, const QString& map, float r, float g, float b)
{
409
	DEBUG_SLOT();
410 411 412 413 414 415 416 417 418 419 420 421 422 423 424
	View* v = m_schnapps->getView(view);
	MapHandlerGen* m = m_schnapps->getMap(map);
	if(v && m)
	{
		h_viewParameterSet[v][m].diffuseColor = Geom::Vec4f(r,g,b,0);
		if(v->isSelectedView())
		{
			if(v->isLinkedToMap(m))	v->updateGL();
			if(m->isSelectedMap()) m_dockTab->updateMapParameters();
		}
	}
}

void Surface_Render_Plugin::changeEdgeColor(const QString& view, const QString& map, float r, float g, float b)
{
425
	DEBUG_SLOT();
426 427 428 429 430 431 432 433 434 435 436 437 438 439 440
	View* v = m_schnapps->getView(view);
	MapHandlerGen* m = m_schnapps->getMap(map);
	if(v && m)
	{
		h_viewParameterSet[v][m].simpleColor = Geom::Vec4f(r,g,b,0);
		if(v->isSelectedView())
		{
			if(v->isLinkedToMap(m))	v->updateGL();
			if(m->isSelectedMap()) m_dockTab->updateMapParameters();
		}
	}
}

void Surface_Render_Plugin::changeVertexColor(const QString& view, const QString& map, float r, float g, float b)
{
441
	DEBUG_SLOT();
442 443 444 445 446 447 448 449 450 451 452 453 454 455
	View* v = m_schnapps->getView(view);
	MapHandlerGen* m = m_schnapps->getMap(map);
	if(v && m)
	{
		h_viewParameterSet[v][m].vertexColor = Geom::Vec4f(r,g,b,0);
		if(v->isSelectedView())
		{
			if(v->isLinkedToMap(m))	v->updateGL();
			if(m->isSelectedMap()) m_dockTab->updateMapParameters();
		}
	}
}


456 457 458 459 460 461 462 463 464 465 466 467 468 469 470 471
void Surface_Render_Plugin::changeBackColor(const QString& view, const QString& map, float r, float g, float b)
{
	DEBUG_SLOT();
	View* v = m_schnapps->getView(view);
	MapHandlerGen* m = m_schnapps->getMap(map);
	if (v && m)
	{
		h_viewParameterSet[v][m].backColor = Geom::Vec4f(r, g, b, 0);
		if (v->isSelectedView())
		{
			if (v->isLinkedToMap(m))	v->updateGL();
			if (m->isSelectedMap()) m_dockTab->updateMapParameters();
		}
	}
}

472 473 474 475 476 477 478 479 480 481 482 483 484 485 486
void Surface_Render_Plugin::changeRenderBackfaces(const QString& view, const QString& map, bool b)
{
	DEBUG_SLOT();
	View* v = m_schnapps->getView(view);
	MapHandlerGen* m = m_schnapps->getMap(map);
	if (v && m)
	{
		h_viewParameterSet[v][m].renderBackfaces = b;
		if (v->isSelectedView())
		{
			if (v->isLinkedToMap(m))	v->updateGL();
			if (m->isSelectedMap()) m_dockTab->updateMapParameters();
		}
	}
}
487 488


489

490
void Surface_Render_Plugin::schnappsClosing()
491 492 493
{
	m_dockTab->m_colorDial->close();
}
Sylvain Thery's avatar
Sylvain Thery committed
494 495 496 497 498
#if CGOGN_QT_DESIRED_VERSION == 5
	Q_PLUGIN_METADATA(IID "CGoGN.SCHNapps.Plugin")
#else
	Q_EXPORT_PLUGIN2(Surface_Render_Plugin, Surface_Render_Plugin)
#endif
499

Pierre Kraemer's avatar
Pierre Kraemer committed
500 501 502 503

} // namespace SCHNApps

} // namespace CGoGN