source: src/UIElements/Views/Qt4/Qt3D/GLWorldScene.cpp@ 136842

Action_Thermostats Add_AtomRandomPerturbation Add_FitFragmentPartialChargesAction Add_RotateAroundBondAction Add_SelectAtomByNameAction Added_ParseSaveFragmentResults AddingActions_SaveParseParticleParameters Adding_Graph_to_ChangeBondActions Adding_MD_integration_tests Adding_ParticleName_to_Atom Adding_StructOpt_integration_tests AtomFragments Automaking_mpqc_open AutomationFragmentation_failures Candidate_v1.5.4 Candidate_v1.6.0 Candidate_v1.6.1 ChangeBugEmailaddress ChangingTestPorts ChemicalSpaceEvaluator CombiningParticlePotentialParsing Combining_Subpackages Debian_Package_split Debian_package_split_molecuildergui_only Disabling_MemDebug Docu_Python_wait EmpiricalPotential_contain_HomologyGraph EmpiricalPotential_contain_HomologyGraph_documentation Enable_parallel_make_install Enhance_userguide Enhanced_StructuralOptimization Enhanced_StructuralOptimization_continued Example_ManyWaysToTranslateAtom Exclude_Hydrogens_annealWithBondGraph FitPartialCharges_GlobalError Fix_BoundInBox_CenterInBox_MoleculeActions Fix_ChargeSampling_PBC Fix_ChronosMutex Fix_FitPartialCharges Fix_FitPotential_needs_atomicnumbers Fix_ForceAnnealing Fix_IndependentFragmentGrids Fix_ParseParticles Fix_ParseParticles_split_forward_backward_Actions Fix_PopActions Fix_QtFragmentList_sorted_selection Fix_Restrictedkeyset_FragmentMolecule Fix_StatusMsg Fix_StepWorldTime_single_argument Fix_Verbose_Codepatterns Fix_fitting_potentials Fixes ForceAnnealing_goodresults ForceAnnealing_oldresults ForceAnnealing_tocheck ForceAnnealing_with_BondGraph ForceAnnealing_with_BondGraph_continued ForceAnnealing_with_BondGraph_continued_betteresults ForceAnnealing_with_BondGraph_contraction-expansion FragmentAction_writes_AtomFragments FragmentMolecule_checks_bonddegrees GeometryObjects Gui_Fixes Gui_displays_atomic_force_velocity ImplicitCharges IndependentFragmentGrids IndependentFragmentGrids_IndividualZeroInstances IndependentFragmentGrids_IntegrationTest IndependentFragmentGrids_Sole_NN_Calculation JobMarket_RobustOnKillsSegFaults JobMarket_StableWorkerPool JobMarket_unresolvable_hostname_fix MoreRobust_FragmentAutomation ODR_violation_mpqc_open PartialCharges_OrthogonalSummation PdbParser_setsAtomName PythonUI_with_named_parameters QtGui_reactivate_TimeChanged_changes Recreated_GuiChecks Rewrite_FitPartialCharges RotateToPrincipalAxisSystem_UndoRedo SaturateAtoms_findBestMatching SaturateAtoms_singleDegree StoppableMakroAction Subpackage_CodePatterns Subpackage_JobMarket Subpackage_LinearAlgebra Subpackage_levmar Subpackage_mpqc_open Subpackage_vmg Switchable_LogView ThirdParty_MPQC_rebuilt_buildsystem TrajectoryDependenant_MaxOrder TremoloParser_IncreasedPrecision TremoloParser_MultipleTimesteps TremoloParser_setsAtomName Ubuntu_1604_changes stable
Last change on this file since 136842 was 136842, checked in by Frederik Heber <heber@…>, 9 years ago

GLWorldScene needs to store observed values to ensure duration of their lifetime.

  • Property mode set to 100644
File size: 26.2 KB
Line 
1/*
2 * Project: MoleCuilder
3 * Description: creates and alters molecular systems
4 * Copyright (C) 2010-2012 University of Bonn. All rights reserved.
5 * Copyright (C) 2013 Frederik Heber. All rights reserved.
6 *
7 *
8 * This file is part of MoleCuilder.
9 *
10 * MoleCuilder is free software: you can redistribute it and/or modify
11 * it under the terms of the GNU General Public License as published by
12 * the Free Software Foundation, either version 2 of the License, or
13 * (at your option) any later version.
14 *
15 * MoleCuilder is distributed in the hope that it will be useful,
16 * but WITHOUT ANY WARRANTY; without even the implied warranty of
17 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
18 * GNU General Public License for more details.
19 *
20 * You should have received a copy of the GNU General Public License
21 * along with MoleCuilder. If not, see <http://www.gnu.org/licenses/>.
22 */
23
24/*
25 * GLWorldScene.cpp
26 *
27 * This is based on the Qt3D example "teaservice", specifically parts of teaservice.cpp.
28 *
29 * Created on: Aug 17, 2011
30 * Author: heber
31 */
32
33// include config.h
34#ifdef HAVE_CONFIG_H
35#include <config.h>
36#endif
37
38#include "GLWorldScene.hpp"
39#include <Qt3D/qglview.h>
40#include <Qt3D/qglbuilder.h>
41#include <Qt3D/qglscenenode.h>
42#include <Qt3D/qglsphere.h>
43#include <Qt3D/qglcylinder.h>
44
45#include "UIElements/Views/Qt4/Qt3D/GLMoleculeObject.hpp"
46#include "UIElements/Views/Qt4/Qt3D/GLMoleculeObject_atom.hpp"
47#include "UIElements/Views/Qt4/Qt3D/GLMoleculeObject_bond.hpp"
48#include "UIElements/Views/Qt4/Qt3D/GLMoleculeObject_molecule.hpp"
49#include "UIElements/Views/Qt4/Qt3D/GLMoleculeObject_shape.hpp"
50
51#include "UIElements/Qt4/InstanceBoard/QtObservedInstanceBoard.hpp"
52
53#include "CodePatterns/MemDebug.hpp"
54
55#include "CodePatterns/Log.hpp"
56
57#include <boost/assign.hpp>
58
59#include "Actions/SelectionAction/Atoms/AtomByIdAction.hpp"
60#include "Actions/SelectionAction/Atoms/NotAtomByIdAction.hpp"
61#include "Actions/SelectionAction/Molecules/MoleculeByIdAction.hpp"
62#include "Actions/SelectionAction/Molecules/NotMoleculeByIdAction.hpp"
63#include "Atom/atom.hpp"
64#include "Bond/bond.hpp"
65#include "Descriptors/AtomIdDescriptor.hpp"
66#include "Descriptors/MoleculeIdDescriptor.hpp"
67#include "Helpers/helpers.hpp"
68#include "Shapes/ShapeRegistry.hpp"
69#include "molecule.hpp"
70#include "World.hpp"
71
72#include <iostream>
73
74using namespace MoleCuilder;
75
76GLWorldScene::GLWorldScene(
77 QtObservedInstanceBoard * _board,
78 QObject *parent) :
79 QObject(parent),
80 BondNodeParentMaps(std::vector<BondNodeParentMap_t>(2)),
81 selectionMode(SelectAtom),
82 board(_board)
83{
84 qRegisterMetaType<atomId_t>("atomId_t");
85 qRegisterMetaType<GLMoleculeObject_bond::SideOfBond>("GLMoleculeObject_bond::SideOfBond");
86
87 int sphereDetails[] = {5, 3, 2, 0};
88 int cylinderDetails[] = {16, 8, 6, 3};
89 for (int i=0;i<GLMoleculeObject::DETAILTYPES_MAX;i++){
90 QGLBuilder emptyBuilder;
91 GLMoleculeObject::meshEmpty[i] = emptyBuilder.finalizedSceneNode();
92 QGLBuilder sphereBuilder;
93 sphereBuilder << QGLSphere(2.0, sphereDetails[i]);
94 GLMoleculeObject::meshSphere[i] = sphereBuilder.finalizedSceneNode();
95 GLMoleculeObject::meshSphere[i]->setOption(QGLSceneNode::CullBoundingBox, true);
96 QGLBuilder cylinderBuilder;
97 cylinderBuilder << QGLCylinder(.25,.25,1.0,cylinderDetails[i]);
98 GLMoleculeObject::meshCylinder[i] = cylinderBuilder.finalizedSceneNode();
99 GLMoleculeObject::meshCylinder[i]->setOption(QGLSceneNode::CullBoundingBox, true);
100 }
101 connect(board, SIGNAL(moleculeInserted(QtObservedMolecule::ptr)),
102 this, SLOT(insertMolecule(QtObservedMolecule::ptr)));
103 connect(board, SIGNAL(moleculeRemoved(ObservedValue_Index_t)),
104 this, SLOT(removeMolecule(ObservedValue_Index_t)));
105
106 connect(board, SIGNAL(atomInserted(QtObservedAtom::ptr)),
107 this, SLOT(connectAtom(QtObservedAtom::ptr)), Qt::DirectConnection);
108 connect(this, SIGNAL(atomConnected(QtObservedAtom::ptr)),
109 this, SLOT(insertAtom(QtObservedAtom::ptr)));
110 connect(board, SIGNAL(atomRemoved(ObservedValue_Index_t)),
111 this, SLOT(removeAtom(ObservedValue_Index_t)));
112
113 connect(board, SIGNAL(bondInserted(QtObservedBond::ptr)),
114 this, SLOT(connectBond(QtObservedBond::ptr)), Qt::DirectConnection);
115 connect(this, SIGNAL(bondConnected(QtObservedBond::ptr)),
116 this, SLOT(insertBond(QtObservedBond::ptr)));
117 connect(board, SIGNAL(bondRemoved(ObservedValue_Index_t)),
118 this, SLOT(removeBond(ObservedValue_Index_t)));
119
120// connect(this, SIGNAL(updated()), this, SLOT(update()));
121}
122
123GLWorldScene::~GLWorldScene()
124{
125 // remove all elements
126 GLMoleculeObject::cleanMaterialMap();
127}
128
129void GLWorldScene::clickAtom(atomId_t no)
130{
131 LOG(3, "INFO: GLWorldScene - atom " << no << " has been clicked.");
132 const atom * const Walker = const_cast<const World &>(World::getInstance()).
133 getAtom(AtomById(no));
134 ASSERT( Walker != NULL,
135 "GLWorldScene::clickAtom() - clicked atom has disappeared.");
136 if (selectionMode == SelectAtom){
137 if (!World::getInstance().isSelected(Walker))
138 SelectionAtomById(std::vector<atomId_t>(1,no));
139 else
140 SelectionNotAtomById(std::vector<atomId_t>(1,no));
141 }else if (selectionMode == SelectMolecule){
142 const molecule *mol = Walker->getMolecule();
143 ASSERT(mol, "Atom without molecule has been clicked.");
144 molids_t ids(1, mol->getId());
145 if (!World::getInstance().isSelected(mol))
146 SelectionMoleculeById(ids);
147 else
148 SelectionNotMoleculeById(ids);
149 }
150 emit clicked(no);
151}
152
153/** Prepares adding an atom to the scene
154 *
155 * We need to listen to moleculeChanged() in order to properly re-parent()
156 * this QObject
157 *
158 * @param _atom atom to connect
159 */
160void GLWorldScene::connectAtom(QtObservedAtom::ptr _atom)
161{
162 connect(_atom.get(), SIGNAL(moleculeChanged()), this, SLOT(reparentAtom()));
163
164 // store the object, as we need it on reparenting
165#ifndef NDEBUG
166 std::pair< ObservedAtoms_t::iterator, bool > inserter =
167#endif
168 ObservedAtoms.insert( std::make_pair(_atom->getIndex(), _atom) );
169 ASSERT( inserter.second,
170 "GLWorldScene::connectAtom() - observed atom "+toString(_atom)+" already stored?");
171
172 {
173 const ObservedValue_Index_t atomid = _atom->getIndex();
174 boost::recursive_mutex::scoped_lock lock(AtomNodeParentMap_mutex);
175 AtomNodeParentMap.left.insert( std::make_pair(atomid, (ObservedValue_Index_t)NULL) );
176 }
177
178 emit atomConnected(_atom);
179}
180
181/** Adds an atom to the scene.
182 *
183 * @param _atom atom to add
184 */
185void GLWorldScene::insertAtom(QtObservedAtom::ptr _atom)
186{
187 LOG(3, "INFO: GLWorldScene: Received signal atomInserted for atom "
188 << _atom->getAtomIndex());
189
190 const ObservedValue_Index_t atomid = _atom->getIndex();
191 QObject *parent = static_cast<QObject *>(this);
192 GLMoleculeObject_atom *atomObject = NULL;
193 {
194 boost::recursive_mutex::scoped_lock lock(AtomNodeParentMap_mutex);
195 AtomNodeParentMap_t::left_iterator parentiter = AtomNodeParentMap.left.find(atomid);
196 ASSERT (parentiter != AtomNodeParentMap.left.end(),
197 "GLWorldScene::insertAtom() - parent to atom id "+toString(atomid)+" unknown?");
198 const ObservedValue_Index_t parentindex = parentiter->second;
199 if (parentindex != (ObservedValue_Index_t)NULL) {
200 const MoleculeNodeMap::iterator moliter = MoleculesinSceneMap.find(parentindex);
201 if (moliter != MoleculesinSceneMap.end())
202 parent = moliter->second;
203 }
204
205 atomObject = new GLMoleculeObject_atom(
206 GLMoleculeObject::meshSphere,
207 parent,
208 _atom);
209 ASSERT( atomObject != NULL,
210 "GLWorldScene::atomInserted - could not create atom object for "
211 +toString(_atom->getAtomIndex()));
212 }
213 AtomNodeMap::iterator iter = AtomsinSceneMap.find(atomid);
214 ASSERT(iter == AtomsinSceneMap.end(),
215 "GLWorldScene::insertAtom - same atom with id "
216 +toString(_atom->getAtomIndex())+" added again.");
217 AtomsinSceneMap.insert( make_pair(atomid, atomObject) );
218
219 connect (atomObject, SIGNAL(clicked(atomId_t)), this, SLOT(clickAtom(atomId_t)));
220 connect (atomObject, SIGNAL(changed()), this, SIGNAL(changed()));
221 connect (atomObject, SIGNAL(hoverChanged(GLMoleculeObject *)), this, SIGNAL(changed()));
222 connect (atomObject, SIGNAL(hoverChanged(GLMoleculeObject *)), this, SLOT(hoverChangedSignalled(GLMoleculeObject *)));
223
224 emit changed();
225 emit changeOccured();
226}
227
228/** Removes an atom.
229 *
230 * @param _atomid index of the atom that is removed
231 */
232void GLWorldScene::removeAtom(ObservedValue_Index_t _atomid)
233{
234 LOG(3, "INFO: GLWorldScene: Received signal atomRemoved for atom "+toString(_atomid)+".");
235 // bonds are removed by signal coming from ~bond
236
237 // remove atoms
238 {
239 AtomNodeMap::iterator iter = AtomsinSceneMap.find(_atomid);
240 ASSERT(iter != AtomsinSceneMap.end(),
241 "GLWorldScene::removeAtom() - atom "+toString(_atomid)+" not on display.");
242 GLMoleculeObject_atom *atomObject = iter->second;
243 AtomsinSceneMap.erase(iter);
244 delete atomObject;
245 }
246 {
247 boost::recursive_mutex::scoped_lock lock(AtomNodeParentMap_mutex);
248 AtomNodeParentMap.left.erase(_atomid);
249 }
250 {
251 const size_t erased = ObservedAtoms.erase(_atomid);
252 ASSERT( erased == 1,
253 "GLWorldScene::removeAtom() - could not erase stored observed value "+toString(_atomid));
254 }
255 emit changed();
256 emit changeOccured();
257}
258
259/** Prepares adding a bond to the scene
260 *
261 * We need to listen to moleculeChanged() in order to properly re-parent()
262 * this QObject
263 *
264 * @param _bond bond to connect
265 */
266void GLWorldScene::connectBond(QtObservedBond::ptr _bond)
267{
268 connect(_bond.get(), SIGNAL(leftmoleculeChanged()), this, SLOT(reparentBondLeft()));
269 connect(_bond.get(), SIGNAL(rightmoleculeChanged()), this, SLOT(reparentBondRight()));
270
271 // store the object, as we need it on reparenting
272#ifndef NDEBUG
273 std::pair< ObservedBonds_t::iterator, bool > inserter =
274#endif
275 ObservedBonds.insert( std::make_pair(_bond->getIndex(), _bond) );
276 ASSERT( inserter.second,
277 "GLWorldScene::connectBond() - observed bond "+toString(_bond)+" already stored?");
278
279 {
280 const ObservedValue_Index_t bondid = _bond->getIndex();
281 boost::recursive_mutex::scoped_lock lock(BondNodeParentMap_mutex);
282 if ((_bond->getLeftAtom() != NULL) && (_bond->getLeftAtom()->getMoleculeRef() != NULL))
283 BondNodeParentMaps[0].left.insert(
284 std::make_pair(bondid, _bond->getLeftAtom()->getMoleculeRef()->getIndex()) );
285 else
286 BondNodeParentMaps[0].left.insert(
287 std::make_pair(bondid, (ObservedValue_Index_t)NULL) );
288 if ((_bond->getRightAtom() != NULL) && (_bond->getRightAtom()->getMoleculeRef() != NULL))
289 BondNodeParentMaps[1].left.insert(
290 std::make_pair(bondid, _bond->getRightAtom()->getMoleculeRef()->getIndex()) );
291 else
292 BondNodeParentMaps[1].left.insert(
293 std::make_pair(bondid, (ObservedValue_Index_t)NULL) );
294 }
295
296 emit bondConnected(_bond);
297}
298
299/** Adds a bond to the scene.
300 *
301 * @param _bond bond to add
302 */
303void GLWorldScene::insertBond(QtObservedBond::ptr _bond)
304{
305 static const std::vector< GLMoleculeObject_bond::SideOfBond > bondsides =
306 boost::assign::list_of<GLMoleculeObject_bond::SideOfBond>
307 (GLMoleculeObject_bond::left)
308 (GLMoleculeObject_bond::right);
309 LOG(3, "INFO: GLWorldScene::insertBond() - Adding bonds " << _bond->getBondIndex());
310
311 const ObservedValue_Index_t bondid = _bond->getIndex();
312#ifdef NDEBUG
313 BondNodeMap::iterator iter = BondsinSceneMap.find(bondid);
314 ASSERT( iter == BondsinSceneMap.end(),
315 "GLWorldScene::insertBond() - bond "+toString(bondid)+" is already known.");
316#endif
317 {
318 boost::recursive_mutex::scoped_lock lock(BondNodeParentMap_mutex);
319 for (size_t i=0;i<2;++i) {
320 BondNodeParentMap_t::left_iterator parentiter = BondNodeParentMaps[i].left.find(bondid);
321 ASSERT (parentiter != BondNodeParentMaps[i].left.end(),
322 "GLWorldScene::insertBond() - parent to bond id "+toString(bondid)+" unknown?");
323 QObject *parent = this;
324 if (parentiter->second != (ObservedValue_Index_t)NULL) {
325 const MoleculeNodeMap::iterator moliter = MoleculesinSceneMap.find(parentiter->second);
326 if (moliter != MoleculesinSceneMap.end())
327 parent = moliter->second;
328 }
329
330 GLMoleculeObject_bond *bondObject = new GLMoleculeObject_bond(
331 GLMoleculeObject::meshCylinder,
332 parent,
333 _bond,
334 bondsides[i]);
335 connect (bondObject, SIGNAL(changed()), this, SIGNAL(changed()));
336 BondsinSceneMap.insert( std::make_pair(bondid, bondObject) );
337 }
338 }
339
340 emit changed();
341 emit changeOccured();
342}
343
344/** Removes a bond.
345 *
346 * @param _bondid id of bond to remove
347 */
348void GLWorldScene::removeBond(ObservedValue_Index_t _bondid)
349{
350 LOG(3, "INFO: GLWorldScene::removedBond() - Removing bond to id " << _bondid);
351
352 // left bond
353 {
354 std::pair<BondNodeMap::iterator, BondNodeMap::iterator> iters =
355 BondsinSceneMap.equal_range(_bondid);
356 ASSERT( iters.first != iters.second,
357 "GLWorldScene::removedBond() - could not find bond to id "+toString(_bondid));
358 for (BondNodeMap::iterator iter = iters.first; iter != iters.second; ++iter) {
359 GLMoleculeObject_bond *bondObject = iter->second;
360 delete bondObject; // is done by signal from bond itself
361 //LOG(4, "INFO: Still present bonds " << BondsinSceneMap << ".");
362 }
363 BondsinSceneMap.erase(_bondid);
364 }
365 {
366 boost::recursive_mutex::scoped_lock lock(BondNodeParentMap_mutex);
367 BondNodeParentMaps[0].left.erase(_bondid);
368 BondNodeParentMaps[1].left.erase(_bondid);
369 }
370 {
371 const size_t erased = ObservedAtoms.erase(_bondid);
372 ASSERT( erased == 1,
373 "GLWorldScene::removeBond() - could not erase stored observed value "+toString(_bondid));
374 }
375
376 emit changed();
377 emit changeOccured();
378}
379
380void GLWorldScene::hoverChangedSignalled(GLMoleculeObject *ob)
381{
382 // Find the atom, ob corresponds to.
383 hoverAtomId = -1;
384 GLMoleculeObject_atom *atomObject = dynamic_cast<GLMoleculeObject_atom *>(ob);
385 if (atomObject){
386 for (AtomNodeMap::iterator iter = AtomsinSceneMap.begin();iter != AtomsinSceneMap.end(); ++ iter){
387 if (iter->second == atomObject)
388 hoverAtomId = iter->second->objectId();
389 }
390
391 // Propagate signal.
392 emit hoverChanged(hoverAtomId);
393 } else {
394 // Find the atom, ob corresponds to.
395 GLMoleculeObject_molecule *moleculeObject = dynamic_cast<GLMoleculeObject_molecule *>(ob);
396 if (moleculeObject){
397 // Propagate signal.
398 emit hoverChanged(moleculeObject->objectId(), 0);
399 }
400 }
401}
402
403void GLWorldScene::clickMolecule(moleculeId_t no)
404{
405 LOG(3, "INFO: GLMoleculeObject_molecule - mol " << no << " has been clicked.");
406 const molecule * const mol= const_cast<const World &>(World::getInstance()).
407 getMolecule(MoleculeById(no));
408 ASSERT(mol, "Atom without molecule has been clicked.");
409 molids_t ids(1, mol->getId());
410 if (!World::getInstance().isSelected(mol))
411 SelectionMoleculeById(ids);
412 else
413 SelectionNotMoleculeById(ids);
414 emit clicked(no);
415}
416
417/** Inserts a molecule into the scene.
418 *
419 * @param _mol molecule to insert
420 */
421void GLWorldScene::insertMolecule(QtObservedMolecule::ptr _mol)
422{
423 const ObservedValue_Index_t molid = _mol->getIndex();
424 LOG(3, "INFO: GLWorldScene: Received signal moleculeInserted for molecule "
425 << _mol->getMolIndex());
426
427 MoleculeNodeMap::const_iterator iter = MoleculesinSceneMap.find(molid);
428 ASSERT( iter == MoleculesinSceneMap.end(),
429 "GLWorldScene::insertMolecule() - molecule's id "+toString(molid)
430 +" already present.");
431
432 // add new object
433 LOG(1, "DEBUG: Adding GLMoleculeObject_molecule to id " << molid);
434 GLMoleculeObject_molecule *molObject =
435 new GLMoleculeObject_molecule(
436 GLMoleculeObject::meshEmpty,
437 this,
438 _mol);
439 ASSERT( molObject != NULL,
440 "GLWorldScene::insertMolecule - could not create molecule object for "
441 +toString(molid));
442#ifndef NDEBUG
443 foreach (QObject *obj, molObject->children()) {
444 GLMoleculeObject *meshobj = qobject_cast<GLMoleculeObject *>(obj);
445 ASSERT( meshobj == NULL,
446 "GLWorldScene::insertMolecule() - there are already atoms or bonds attached to a to molecule.");
447 }
448#endif
449
450 // check all atoms for not yet assigned parents
451 {
452 boost::recursive_mutex::scoped_lock lock(AtomNodeParentMap_mutex);
453 std::pair<AtomNodeParentMap_t::right_const_iterator, AtomNodeParentMap_t::right_const_iterator> iters =
454 AtomNodeParentMap.right.equal_range(molid);
455 for (AtomNodeParentMap_t::right_const_iterator iter = iters.first;
456 iter != iters.second; ++iter) {
457 AtomNodeMap::const_iterator atomiter = AtomsinSceneMap.find(iter->second);
458 if (atomiter != AtomsinSceneMap.end())
459 resetParent(atomiter->second, molObject);
460 }
461 }
462 // check all bonds for not yet assigned parents
463 {
464 boost::recursive_mutex::scoped_lock lock(BondNodeParentMap_mutex);
465 for (size_t i=0;i<2;++i) {
466 std::pair<BondNodeParentMap_t::right_const_iterator, BondNodeParentMap_t::right_const_iterator> iters =
467 BondNodeParentMaps[i].right.equal_range(molid);
468 for (BondNodeParentMap_t::right_const_iterator iter = iters.first;
469 iter != iters.second; ++iter) {
470 BondNodeMap::const_iterator bonditer = BondsinSceneMap.find(iter->second);
471 if (bonditer != BondsinSceneMap.end())
472 resetParent(bonditer->second, molObject);
473 }
474 }
475 }
476
477#ifndef NDEBUG
478 std::pair<MoleculeNodeMap::iterator, bool> inserter =
479#endif
480 MoleculesinSceneMap.insert( make_pair(molid, molObject) );
481 ASSERT(inserter.second,
482 "GLWorldScene::insertMolecule() - molecule "+toString(_mol->getMolIndex())
483 +" already present in scene.");
484
485 connect (molObject, SIGNAL(changed()), this, SIGNAL(changed()));
486 connect (molObject, SIGNAL(changeOccured()), this, SIGNAL(changeOccured()));
487
488 emit changed();
489 emit changeOccured();
490}
491
492/** Removes a molecule from the scene.
493 *
494 * @param _molid index of the molecule to remove
495 */
496void GLWorldScene::removeMolecule(ObservedValue_Index_t _molid)
497{
498 LOG(3, "INFO: GLWorldScene: Received signal moleculeRemoved for molecule "+toString(_molid)+".");
499
500 MoleculeNodeMap::iterator iter = MoleculesinSceneMap.find(_molid);
501 ASSERT ( iter != MoleculesinSceneMap.end(),
502 "GLWorldScene::removeMolecule() - to be removed molecule "+toString(_molid)
503 +" is already gone.");
504 GLMoleculeObject_molecule *molObject = iter->second;
505 // check for any atoms and bonds still attached to it
506#ifndef NDEBUG
507 foreach (QObject *obj, molObject->children()) {
508 GLMoleculeObject *meshobj = qobject_cast<GLMoleculeObject *>(obj);
509 ASSERT( meshobj == NULL,
510 "GLWorldScene::removeMolecule() - there are still atoms or bonds attached to a to molecule.");
511 }
512#endif
513 // finally, remove molecule
514 delete molObject;
515 MoleculesinSceneMap.erase(iter);
516
517 emit changed();
518 emit changeOccured();
519}
520
521void GLWorldScene::moleculesVisibilityChanged(ObservedValue_Index_t _id, bool _visible)
522{
523 MoleculeNodeMap::iterator iter = MoleculesinSceneMap.find(_id);
524 ASSERT( iter != MoleculesinSceneMap.end(),
525 "GLWorldScene::moleculeInserted() - molecule's id "
526 +toString(board->getMoleculeIdToIndex(_id))+" is unknown.");
527
528 GLMoleculeObject_molecule *molObject = iter->second;
529 molObject->setVisible(_visible);
530
531 emit changed();
532 emit changeOccured();
533}
534
535/** This converts safely index into a GLMoleculeObject_molecule.
536 *
537 * \param _MoleculesinSceneMap all present molecules
538 * \param _molid index to look for
539 * \return MolObject or NULL when not found
540 */
541GLMoleculeObject_molecule *GLWorldScene::getMoleculeObject(
542 const ObservedValue_Index_t _molid) const
543{
544 const MoleculeNodeMap::const_iterator moliter = MoleculesinSceneMap.find(_molid);
545 if (moliter != MoleculesinSceneMap.end())
546 return moliter->second;
547 else
548 return NULL;
549}
550
551/** Changes the parent of an object in the scene.
552 *
553 * \param _id index of the object whose parent to change
554 * \param _ob new parent
555 */
556void GLWorldScene::reparentAtom()
557{
558 QtObservedAtom *walker = dynamic_cast<QtObservedAtom *>(sender());
559 boost::recursive_mutex::scoped_lock lock(AtomNodeParentMap_mutex);
560 const ObservedValue_Index_t walkerid = walker->getIndex();
561 LOG(4, "DEBUG: GLWorldScene: Received signal moleculeChanged for atom "+toString(walkerid)+".");
562 AtomNodeParentMap_t::left_iterator parentiter = AtomNodeParentMap.left.find(walkerid);
563 ASSERT( parentiter != AtomNodeParentMap.left.end(),
564 "GLWorldScene::reparentAtom() - could not find object to id "+toString(walkerid));
565
566 // change parent entry
567 AtomNodeParentMap.left.erase(parentiter);
568 if (walker->getMoleculeRef() != NULL)
569 AtomNodeParentMap.left.insert( std::make_pair(walkerid, walker->getMoleculeRef()->getIndex()) );
570 else
571 AtomNodeParentMap.left.insert( std::make_pair(walkerid, (ObservedValue_Index_t)NULL) );
572 parentiter = AtomNodeParentMap.left.find(walkerid);
573
574 const AtomNodeMap::iterator atomiter = AtomsinSceneMap.find(walkerid);
575 if (atomiter != AtomsinSceneMap.end())
576 resetParent(atomiter->second, getMoleculeObject(parentiter->second));
577 // else atom does not yet exist
578}
579
580/** Changes the parent of an left-side bond in the scene.
581 *
582 */
583void GLWorldScene::reparentBondLeft()
584{
585 boost::recursive_mutex::scoped_lock lock(BondNodeParentMap_mutex);
586 QtObservedBond *bond = dynamic_cast<QtObservedBond *>(sender());
587 reparentBond(bond, bond->getLeftAtom(), GLMoleculeObject_bond::left);
588}
589
590/** Changes the parent of an right-side bond in the scene.
591 *
592 */
593void GLWorldScene::reparentBondRight()
594{
595 boost::recursive_mutex::scoped_lock lock(BondNodeParentMap_mutex);
596 QtObservedBond *bond = dynamic_cast<QtObservedBond *>(sender());
597 reparentBond(bond, bond->getRightAtom(), GLMoleculeObject_bond::right);
598}
599
600GLMoleculeObject_bond *GLWorldScene::getBondInScene(
601 const ObservedValue_Index_t _bondid,
602 GLMoleculeObject_bond::SideOfBond _side) const
603{
604 std::pair<GLWorldScene::BondNodeMap::const_iterator, GLWorldScene::BondNodeMap::const_iterator> iters =
605 BondsinSceneMap.equal_range(_bondid);
606 ASSERT( std::distance(iters.first, iters.second) == 2,
607 "GLWorldScene::getBondInScene() - not exactly two bonds of same id "
608 +toString(_bondid)+" present in scene.");
609 for (GLWorldScene::BondNodeMap::const_iterator bonditer = iters.first;
610 bonditer != iters.second; ++bonditer) {
611 if (bonditer->second->BondSide == _side)
612 return bonditer->second;
613 }
614 return NULL;
615}
616
617/** Changes the parent of an object in the scene.
618 *
619 * \param _atom atom of bond whose molecule we are associated to
620 * \param _side side of bond
621 */
622void GLWorldScene::reparentBond(
623 QtObservedBond *_bond,
624 const QtObservedAtom::ptr _atom,
625 const GLMoleculeObject_bond::SideOfBond _side)
626{
627 boost::recursive_mutex::scoped_lock lock(BondNodeParentMap_mutex);
628 const size_t dim = (_side == GLMoleculeObject_bond::left) ? 0 : 1;
629 const ObservedValue_Index_t bondid = _bond->getIndex();
630 LOG(4, "DEBUG: GLWorldScene: Received signal moleculeChanged for bond "+toString(bondid)+".");
631 BondNodeParentMap_t::left_iterator parentiter = BondNodeParentMaps[dim].left.find(bondid);
632 ASSERT( parentiter != BondNodeParentMaps[dim].left.end(),
633 "GLWorldScene::reparentBond() - could not find object to id "+toString(bondid));
634
635 // change parent entry
636 BondNodeParentMaps[dim].left.erase(bondid);
637 if ((_atom != NULL) && (_atom->getMoleculeRef() != NULL))
638 BondNodeParentMaps[dim].left.insert( std::make_pair( bondid, _atom->getMoleculeRef()->getIndex()));
639 else
640 BondNodeParentMaps[dim].left.insert( std::make_pair( bondid, (ObservedValue_Index_t)NULL) );
641 parentiter = BondNodeParentMaps[dim].left.find(bondid);
642
643 // reset parent
644 resetParent(getBondInScene(bondid, _side), getMoleculeObject(parentiter->second));
645}
646
647/** Resets the parent of an GLMoleculeObject.
648 *
649 * \param _obj object to reparent
650 * \param _molid index of parent molecule
651 */
652void GLWorldScene::resetParent(
653 GLMoleculeObject *_obj,
654 GLMoleculeObject_molecule *_molobj)
655{
656 if (_obj != NULL) {
657 QObject *parent = this;
658 if (_molobj != NULL)
659 parent = _molobj;
660 // else: molecule does not yet exist: is done when molecule is instantiated
661 LOG(5, "DEBUG: Resetting parent of " << _obj << " to " << parent);
662 _obj->setParent(parent);
663 ASSERT( _obj->parent() == parent,
664 "GLWorldScene::resetParent() - new parent "+toString(parent)+" was not set.");
665 } else
666 ELOG(1, "Object to reparent was NULL.");
667 // else object does not yet exist
668}
669
670/** Adds a shape to the scene.
671 *
672 */
673void GLWorldScene::addShape(const std::string &_name)
674{
675 Shape * const shape = ShapeRegistry::getInstance().getByName(_name);
676 if (shape != NULL) {
677 GLMoleculeObject_shape *shapeObject = new GLMoleculeObject_shape(*shape, this);
678 ShapeNodeMap::iterator iter = ShapesinSceneMap.find(_name);
679 ASSERT(iter == ShapesinSceneMap.end(),
680 "GLWorldScene::addShape() - same shape "+_name+" added again.");
681 ShapesinSceneMap.insert( make_pair(_name, shapeObject) );
682 } else
683 ELOG(2, "GLWorldScene::addShape() - shape disappeared before we could draw it.");
684
685 emit changed();
686}
687
688void GLWorldScene::removeShape(const std::string &_name)
689{
690 ShapeNodeMap::iterator iter = ShapesinSceneMap.find(_name);
691 ASSERT(iter != ShapesinSceneMap.end(),
692 "GLWorldScene::removeShape() - shape "+_name+" not in scene.");
693 ShapesinSceneMap.erase(iter);
694 delete(iter->second);
695
696 emit changed();
697}
698
699void GLWorldScene::updateSelectedShapes()
700{
701 foreach (QObject *obj, children()) {
702 GLMoleculeObject_shape *shapeobj = qobject_cast<GLMoleculeObject_shape *>(obj);
703 if (shapeobj){
704 shapeobj->enable(ShapeRegistry::getInstance().isSelected(shapeobj->getShape()));
705 }
706 }
707
708 emit changed();
709}
710
711void GLWorldScene::initialize(QGLView *view, QGLPainter *painter) const
712{
713 // Initialize all of the mesh objects that we have as children.
714 foreach (QObject *obj, children()) {
715 GLMoleculeObject *meshobj = qobject_cast<GLMoleculeObject *>(obj);
716 if (meshobj)
717 meshobj->initialize(view, painter);
718 }
719}
720
721void GLWorldScene::draw(QGLPainter *painter, const QVector4D &cameraPlane) const
722{
723 // Draw all of the mesh objects that we have as children.
724 foreach (QObject *obj, children()) {
725 GLMoleculeObject *meshobj = qobject_cast<GLMoleculeObject *>(obj);
726 if (meshobj)
727 meshobj->draw(painter, cameraPlane);
728 }
729}
730
731void GLWorldScene::setSelectionMode(SelectionModeType mode)
732{
733 selectionMode = mode;
734 // TODO send update to toolbar
735}
736
737void GLWorldScene::setSelectionModeAtom()
738{
739 setSelectionMode(SelectAtom);
740}
741
742void GLWorldScene::setSelectionModeMolecule()
743{
744 setSelectionMode(SelectMolecule);
745}
746
Note: See TracBrowser for help on using the repository browser.