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

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 649f59 was 649f59, checked in by Frederik Heber <heber@…>, 9 years ago

GLWorldScene removes stored observed values only after both parent is NULL and atomRemoved() came in.

  • this is to ensure that lifetime of the observed value instance is as long as required.
  • Property mode set to 100644
File size: 31.3 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 {
166#ifndef NDEBUG
167 std::pair< ObservedAtoms_t::iterator, bool > inserter =
168#endif
169 ObservedAtoms.insert( std::make_pair(_atom.get(), _atom) );
170 ASSERT( inserter.second,
171 "GLWorldScene::connectAtom() - observed atom "+toString(_atom)+" already stored?");
172#ifndef NDEBUG
173 std::pair< ObservedValueIndexLookup_t::iterator, bool > indexinserter =
174#endif
175 ObservedValueIndexLookup.insert( std::make_pair(_atom->getIndex(), _atom.get()) );
176 ASSERT( indexinserter.second,
177 "GLWorldScene::connectAtom() - observed atom's index "
178 +toString(_atom->getIndex())+" already stored?");
179 }
180 {
181 const ObservedValue_Index_t atomid = _atom->getIndex();
182 boost::recursive_mutex::scoped_lock lock(AtomNodeParentMap_mutex);
183 AtomNodeParentMap.left.insert( std::make_pair(atomid, (ObservedValue_Index_t)NULL) );
184 }
185
186 emit atomConnected(_atom);
187}
188
189/** Adds an atom to the scene.
190 *
191 * @param _atom atom to add
192 */
193void GLWorldScene::insertAtom(QtObservedAtom::ptr _atom)
194{
195 LOG(3, "INFO: GLWorldScene: Received signal atomInserted for atom "
196 << _atom->getAtomIndex());
197
198 const ObservedValue_Index_t atomid = _atom->getIndex();
199 QObject *parent = static_cast<QObject *>(this);
200 GLMoleculeObject_atom *atomObject = NULL;
201 {
202 boost::recursive_mutex::scoped_lock lock(AtomNodeParentMap_mutex);
203 AtomNodeParentMap_t::left_iterator parentiter = AtomNodeParentMap.left.find(atomid);
204 ASSERT (parentiter != AtomNodeParentMap.left.end(),
205 "GLWorldScene::insertAtom() - parent to atom id "+toString(atomid)+" unknown?");
206 const ObservedValue_Index_t parentindex = parentiter->second;
207 if (parentindex != (ObservedValue_Index_t)NULL) {
208 const MoleculeNodeMap::iterator moliter = MoleculesinSceneMap.find(parentindex);
209 if (moliter != MoleculesinSceneMap.end())
210 parent = moliter->second;
211 }
212
213 atomObject = new GLMoleculeObject_atom(
214 GLMoleculeObject::meshSphere,
215 parent,
216 _atom);
217 ASSERT( atomObject != NULL,
218 "GLWorldScene::atomInserted - could not create atom object for "
219 +toString(_atom->getAtomIndex()));
220 }
221 AtomNodeMap::iterator iter = AtomsinSceneMap.find(atomid);
222 ASSERT(iter == AtomsinSceneMap.end(),
223 "GLWorldScene::insertAtom - same atom with id "
224 +toString(_atom->getAtomIndex())+" added again.");
225 AtomsinSceneMap.insert( make_pair(atomid, atomObject) );
226
227 connect (atomObject, SIGNAL(clicked(atomId_t)), this, SLOT(clickAtom(atomId_t)));
228 connect (atomObject, SIGNAL(changed()), this, SIGNAL(changed()));
229 connect (atomObject, SIGNAL(hoverChanged(GLMoleculeObject *)), this, SIGNAL(changed()));
230 connect (atomObject, SIGNAL(hoverChanged(GLMoleculeObject *)), this, SLOT(hoverChangedSignalled(GLMoleculeObject *)));
231
232 emit changed();
233 emit changeOccured();
234}
235
236template <class T>
237void removeStoredObservedValue(
238 GLWorldScene::ObservedValueIndexLookup_t &_ObservedValueIndexLookup,
239 T &_ObservedValues,
240 ObservedValue_Index_t _id)
241{
242 // get QObject* from lookup
243 const GLWorldScene::ObservedValueIndexLookup_t::iterator iter =
244 _ObservedValueIndexLookup.find(_id);
245 ASSERT( iter != _ObservedValueIndexLookup.end(),
246 "GLWorldScene::removeAtom() - could not find index "+toString(_id)
247 +" to stored observed value.");
248 QObject * sender = iter->second;
249 const size_t erased = _ObservedValues.erase(sender);
250 ASSERT( erased == 1,
251 "GLWorldScene::removeStoredObservedValue() - could not erase stored observed value "
252 +toString(_id));
253 _ObservedValueIndexLookup.erase(iter);
254}
255
256/** Removes an atom.
257 *
258 * @param _atomid index of the atom that is removed
259 */
260void GLWorldScene::removeAtom(ObservedValue_Index_t _atomid)
261{
262 LOG(3, "INFO: GLWorldScene: Received signal atomRemoved for atom "+toString(_atomid)+".");
263 // bonds are removed by signal coming from ~bond
264
265 // remove atoms
266 {
267 AtomNodeMap::iterator iter = AtomsinSceneMap.find(_atomid);
268 ASSERT(iter != AtomsinSceneMap.end(),
269 "GLWorldScene::removeAtom() - atom "+toString(_atomid)+" not on display.");
270 GLMoleculeObject_atom *atomObject = iter->second;
271 AtomsinSceneMap.erase(iter);
272 delete atomObject;
273 }
274 {
275 boost::recursive_mutex::scoped_lock lock(AtomNodeParentMap_mutex);
276 AtomNodeParentMap_t::left_iterator iter = AtomNodeParentMap.left.find(_atomid);
277 ASSERT( iter != AtomNodeParentMap.left.end(),
278 "GLWorldScene::removeAtom() - missing parent entry for "+toString(_atomid));
279 if (iter->second == (ObservedValue_Index_t)NULL) {
280 LOG(4, "DEBUG: Found parent of atom as GLWorldScene, removing.");
281 removeStoredObservedValue(ObservedValueIndexLookup, ObservedAtoms, _atomid);
282 }
283 AtomNodeParentMap.left.erase(iter);
284 }
285
286 emit changed();
287 emit changeOccured();
288}
289
290/** Prepares adding a bond to the scene
291 *
292 * We need to listen to moleculeChanged() in order to properly re-parent()
293 * this QObject
294 *
295 * @param _bond bond to connect
296 */
297void GLWorldScene::connectBond(QtObservedBond::ptr _bond)
298{
299 connect(_bond.get(), SIGNAL(leftmoleculeChanged()), this, SLOT(reparentBondLeft()));
300 connect(_bond.get(), SIGNAL(rightmoleculeChanged()), this, SLOT(reparentBondRight()));
301
302 // store the object, as we need it on reparenting
303 {
304#ifndef NDEBUG
305 std::pair< ObservedBonds_t::iterator, bool > inserter =
306#endif
307 ObservedBonds.insert( std::make_pair(_bond.get(), _bond) );
308 ASSERT( inserter.second,
309 "GLWorldScene::connectBond() - observed bond "+toString(_bond)+" already stored?");
310#ifndef NDEBUG
311 std::pair< ObservedValueIndexLookup_t::iterator, bool > indexinserter =
312#endif
313 ObservedValueIndexLookup.insert( std::make_pair(_bond->getIndex(), _bond.get()) );
314 ASSERT( indexinserter.second,
315 "GLWorldScene::connectBond() - observed bond's index "
316 +toString(_bond->getIndex())+" already stored?");
317 }
318 {
319 const ObservedValue_Index_t bondid = _bond->getIndex();
320 boost::recursive_mutex::scoped_lock lock(BondNodeParentMap_mutex);
321 if ((_bond->getLeftAtom() != NULL) && (_bond->getLeftAtom()->getMoleculeRef() != NULL))
322 BondNodeParentMaps[0].left.insert(
323 std::make_pair(bondid, _bond->getLeftAtom()->getMoleculeRef()->getIndex()) );
324 else
325 BondNodeParentMaps[0].left.insert(
326 std::make_pair(bondid, (ObservedValue_Index_t)NULL) );
327 if ((_bond->getRightAtom() != NULL) && (_bond->getRightAtom()->getMoleculeRef() != NULL))
328 BondNodeParentMaps[1].left.insert(
329 std::make_pair(bondid, _bond->getRightAtom()->getMoleculeRef()->getIndex()) );
330 else
331 BondNodeParentMaps[1].left.insert(
332 std::make_pair(bondid, (ObservedValue_Index_t)NULL) );
333 }
334
335 emit bondConnected(_bond);
336}
337
338/** Adds a bond to the scene.
339 *
340 * @param _bond bond to add
341 */
342void GLWorldScene::insertBond(QtObservedBond::ptr _bond)
343{
344 static const std::vector< GLMoleculeObject_bond::SideOfBond > bondsides =
345 boost::assign::list_of<GLMoleculeObject_bond::SideOfBond>
346 (GLMoleculeObject_bond::left)
347 (GLMoleculeObject_bond::right);
348 LOG(3, "INFO: GLWorldScene::insertBond() - Adding bonds " << _bond->getBondIndex());
349
350 const ObservedValue_Index_t bondid = _bond->getIndex();
351#ifdef NDEBUG
352 BondNodeMap::iterator iter = BondsinSceneMap.find(bondid);
353 ASSERT( iter == BondsinSceneMap.end(),
354 "GLWorldScene::insertBond() - bond "+toString(bondid)+" is already known.");
355#endif
356 {
357 boost::recursive_mutex::scoped_lock lock(BondNodeParentMap_mutex);
358 for (size_t i=0;i<2;++i) {
359 BondNodeParentMap_t::left_iterator parentiter = BondNodeParentMaps[i].left.find(bondid);
360 ASSERT (parentiter != BondNodeParentMaps[i].left.end(),
361 "GLWorldScene::insertBond() - parent to bond id "+toString(bondid)+" unknown?");
362 QObject *parent = this;
363 if (parentiter->second != (ObservedValue_Index_t)NULL) {
364 const MoleculeNodeMap::iterator moliter = MoleculesinSceneMap.find(parentiter->second);
365 if (moliter != MoleculesinSceneMap.end())
366 parent = moliter->second;
367 }
368
369 GLMoleculeObject_bond *bondObject = new GLMoleculeObject_bond(
370 GLMoleculeObject::meshCylinder,
371 parent,
372 _bond,
373 bondsides[i]);
374 connect (bondObject, SIGNAL(changed()), this, SIGNAL(changed()));
375 BondsinSceneMap.insert( std::make_pair(bondid, bondObject) );
376 }
377 }
378
379 emit changed();
380 emit changeOccured();
381}
382
383/** Removes a bond.
384 *
385 * @param _bondid id of bond to remove
386 */
387void GLWorldScene::removeBond(ObservedValue_Index_t _bondid)
388{
389 LOG(3, "INFO: GLWorldScene::removedBond() - Removing bond to id " << _bondid);
390
391 // left bond
392 {
393 std::pair<BondNodeMap::iterator, BondNodeMap::iterator> iters =
394 BondsinSceneMap.equal_range(_bondid);
395 ASSERT( iters.first != iters.second,
396 "GLWorldScene::removedBond() - could not find bond to id "+toString(_bondid));
397 for (BondNodeMap::iterator iter = iters.first; iter != iters.second; ++iter) {
398 GLMoleculeObject_bond *bondObject = iter->second;
399 delete bondObject; // is done by signal from bond itself
400 //LOG(4, "INFO: Still present bonds " << BondsinSceneMap << ".");
401 }
402 BondsinSceneMap.erase(_bondid);
403 }
404 {
405 boost::recursive_mutex::scoped_lock lock(BondNodeParentMap_mutex);
406 for (size_t i=0;i<2;++i) {
407 BondNodeParentMap_t::left_iterator iter = BondNodeParentMaps[i].left.find(_bondid);
408 ASSERT( iter != BondNodeParentMaps[i].left.end(),
409 "GLWorldScene::removeBond() - missing parent entry for "+toString(_bondid));
410 if (iter->second == (ObservedValue_Index_t)NULL) {
411 LOG(4, "DEBUG: Found parent of bond as GLWorldScene, removing.");
412 removeStoredObservedValue(ObservedValueIndexLookup, ObservedBonds, _bondid);
413 }
414 BondNodeParentMaps[i].left.erase(iter);
415 }
416 }
417
418 emit changed();
419 emit changeOccured();
420}
421
422void GLWorldScene::hoverChangedSignalled(GLMoleculeObject *ob)
423{
424 // Find the atom, ob corresponds to.
425 hoverAtomId = -1;
426 GLMoleculeObject_atom *atomObject = dynamic_cast<GLMoleculeObject_atom *>(ob);
427 if (atomObject){
428 for (AtomNodeMap::iterator iter = AtomsinSceneMap.begin();iter != AtomsinSceneMap.end(); ++ iter){
429 if (iter->second == atomObject)
430 hoverAtomId = iter->second->objectId();
431 }
432
433 // Propagate signal.
434 emit hoverChanged(hoverAtomId);
435 } else {
436 // Find the atom, ob corresponds to.
437 GLMoleculeObject_molecule *moleculeObject = dynamic_cast<GLMoleculeObject_molecule *>(ob);
438 if (moleculeObject){
439 // Propagate signal.
440 emit hoverChanged(moleculeObject->objectId(), 0);
441 }
442 }
443}
444
445void GLWorldScene::clickMolecule(moleculeId_t no)
446{
447 LOG(3, "INFO: GLMoleculeObject_molecule - mol " << no << " has been clicked.");
448 const molecule * const mol= const_cast<const World &>(World::getInstance()).
449 getMolecule(MoleculeById(no));
450 ASSERT(mol, "Atom without molecule has been clicked.");
451 molids_t ids(1, mol->getId());
452 if (!World::getInstance().isSelected(mol))
453 SelectionMoleculeById(ids);
454 else
455 SelectionNotMoleculeById(ids);
456 emit clicked(no);
457}
458
459/** Inserts a molecule into the scene.
460 *
461 * @param _mol molecule to insert
462 */
463void GLWorldScene::insertMolecule(QtObservedMolecule::ptr _mol)
464{
465 const ObservedValue_Index_t molid = _mol->getIndex();
466 LOG(3, "INFO: GLWorldScene: Received signal moleculeInserted for molecule "
467 << _mol->getMolIndex());
468
469 MoleculeNodeMap::const_iterator iter = MoleculesinSceneMap.find(molid);
470 ASSERT( iter == MoleculesinSceneMap.end(),
471 "GLWorldScene::insertMolecule() - molecule's id "+toString(molid)
472 +" already present.");
473
474 // add new object
475 LOG(1, "DEBUG: Adding GLMoleculeObject_molecule to id " << molid);
476 GLMoleculeObject_molecule *molObject =
477 new GLMoleculeObject_molecule(
478 GLMoleculeObject::meshEmpty,
479 this,
480 _mol);
481 ASSERT( molObject != NULL,
482 "GLWorldScene::insertMolecule - could not create molecule object for "
483 +toString(molid));
484#ifndef NDEBUG
485 foreach (QObject *obj, molObject->children()) {
486 GLMoleculeObject *meshobj = qobject_cast<GLMoleculeObject *>(obj);
487 ASSERT( meshobj == NULL,
488 "GLWorldScene::insertMolecule() - there are already atoms or bonds attached to a to molecule.");
489 }
490#endif
491
492 // check all atoms for not yet assigned parents
493 {
494 boost::recursive_mutex::scoped_lock lock(AtomNodeParentMap_mutex);
495 std::pair<AtomNodeParentMap_t::right_const_iterator, AtomNodeParentMap_t::right_const_iterator> iters =
496 AtomNodeParentMap.right.equal_range(molid);
497 for (AtomNodeParentMap_t::right_const_iterator iter = iters.first;
498 iter != iters.second; ++iter) {
499 AtomNodeMap::const_iterator atomiter = AtomsinSceneMap.find(iter->second);
500 if (atomiter != AtomsinSceneMap.end())
501 resetParent(atomiter->second, molObject);
502 }
503 }
504 // check all bonds for not yet assigned parents
505 {
506 boost::recursive_mutex::scoped_lock lock(BondNodeParentMap_mutex);
507 for (size_t i=0;i<2;++i) {
508 std::pair<BondNodeParentMap_t::right_const_iterator, BondNodeParentMap_t::right_const_iterator> iters =
509 BondNodeParentMaps[i].right.equal_range(molid);
510 for (BondNodeParentMap_t::right_const_iterator iter = iters.first;
511 iter != iters.second; ++iter) {
512 BondNodeMap::const_iterator bonditer = BondsinSceneMap.find(iter->second);
513 if (bonditer != BondsinSceneMap.end())
514 resetParent(bonditer->second, molObject);
515 }
516 }
517 }
518
519#ifndef NDEBUG
520 std::pair<MoleculeNodeMap::iterator, bool> inserter =
521#endif
522 MoleculesinSceneMap.insert( make_pair(molid, molObject) );
523 ASSERT(inserter.second,
524 "GLWorldScene::insertMolecule() - molecule "+toString(_mol->getMolIndex())
525 +" already present in scene.");
526
527 connect (molObject, SIGNAL(changed()), this, SIGNAL(changed()));
528 connect (molObject, SIGNAL(changeOccured()), this, SIGNAL(changeOccured()));
529
530 emit changed();
531 emit changeOccured();
532}
533
534/** Removes a molecule from the scene.
535 *
536 * @param _molid index of the molecule to remove
537 */
538void GLWorldScene::removeMolecule(ObservedValue_Index_t _molid)
539{
540 LOG(3, "INFO: GLWorldScene: Received signal moleculeRemoved for molecule "+toString(_molid)+".");
541
542 MoleculeNodeMap::iterator iter = MoleculesinSceneMap.find(_molid);
543 ASSERT ( iter != MoleculesinSceneMap.end(),
544 "GLWorldScene::removeMolecule() - to be removed molecule "+toString(_molid)
545 +" is already gone.");
546 GLMoleculeObject_molecule *molObject = iter->second;
547 // check for any atoms and bonds still attached to it
548#ifndef NDEBUG
549 foreach (QObject *obj, molObject->children()) {
550 GLMoleculeObject *meshobj = qobject_cast<GLMoleculeObject *>(obj);
551 ASSERT( meshobj == NULL,
552 "GLWorldScene::removeMolecule() - there are still atoms or bonds attached to a to molecule.");
553 }
554#endif
555 // finally, remove molecule
556 delete molObject;
557 MoleculesinSceneMap.erase(iter);
558
559 emit changed();
560 emit changeOccured();
561}
562
563void GLWorldScene::moleculesVisibilityChanged(ObservedValue_Index_t _id, bool _visible)
564{
565 MoleculeNodeMap::iterator iter = MoleculesinSceneMap.find(_id);
566 ASSERT( iter != MoleculesinSceneMap.end(),
567 "GLWorldScene::moleculeInserted() - molecule's id "
568 +toString(board->getMoleculeIdToIndex(_id))+" is unknown.");
569
570 GLMoleculeObject_molecule *molObject = iter->second;
571 molObject->setVisible(_visible);
572
573 emit changed();
574 emit changeOccured();
575}
576
577/** This converts safely index into a GLMoleculeObject_molecule.
578 *
579 * \param _MoleculesinSceneMap all present molecules
580 * \param _molid index to look for
581 * \return MolObject or NULL when not found
582 */
583GLMoleculeObject_molecule *GLWorldScene::getMoleculeObject(
584 const ObservedValue_Index_t _molid) const
585{
586 const MoleculeNodeMap::const_iterator moliter = MoleculesinSceneMap.find(_molid);
587 if (moliter != MoleculesinSceneMap.end())
588 return moliter->second;
589 else
590 return NULL;
591}
592
593/** Changes the parent of an object in the scene.
594 *
595 * \param _id index of the object whose parent to change
596 * \param _ob new parent
597 */
598void GLWorldScene::reparentAtom()
599{
600 boost::recursive_mutex::scoped_lock lock(AtomNodeParentMap_mutex);
601 QObject * origin = sender();
602 ObservedAtoms_t::const_iterator iter = ObservedAtoms.find(origin);
603 ASSERT( iter != ObservedAtoms.end(),
604 "GLWorldScene::reparentAtom() - atom's "+toString(origin)+" is no longer stored?");
605 QtObservedAtom::ptr walker = iter->second;
606 const ObservedValue_Index_t walkerid = walker->getIndex();
607 LOG(4, "DEBUG: GLWorldScene: Received signal moleculeChanged for atom "+toString(walkerid)+".");
608 AtomNodeParentMap_t::left_iterator parentiter = AtomNodeParentMap.left.find(walkerid);
609 ASSERT( parentiter != AtomNodeParentMap.left.end(),
610 "GLWorldScene::reparentAtom() - could not find object to id "+toString(walkerid));
611
612 // change parent entry
613 AtomNodeParentMap.left.erase(parentiter);
614 if (walker->getMoleculeRef() != NULL)
615 AtomNodeParentMap.left.insert( std::make_pair(walkerid, walker->getMoleculeRef()->getIndex()) );
616 else
617 AtomNodeParentMap.left.insert( std::make_pair(walkerid, (ObservedValue_Index_t)NULL) );
618 parentiter = AtomNodeParentMap.left.find(walkerid);
619
620 const AtomNodeMap::iterator atomiter = AtomsinSceneMap.find(walkerid);
621 if (atomiter != AtomsinSceneMap.end())
622 resetParent(atomiter->second, getMoleculeObject(parentiter->second));
623 // else atom does not yet exist
624
625 // check whether node is still shown, otherwise remove from lists completely
626 {
627 AtomNodeMap::iterator iter = AtomsinSceneMap.find(walkerid);
628 if (iter == AtomsinSceneMap.end()) {
629 LOG(3, "DEBUG: Reparented atom that is no longer on display, removing.");
630 removeStoredObservedValue(ObservedValueIndexLookup, ObservedAtoms, walkerid);
631 }
632 }
633}
634
635/** Changes the parent of an left-side bond in the scene.
636 *
637 */
638void GLWorldScene::reparentBondLeft()
639{
640 boost::recursive_mutex::scoped_lock lock(BondNodeParentMap_mutex);
641 QObject * origin = sender();
642 ObservedBonds_t::const_iterator iter = ObservedBonds.find(origin);
643 ASSERT( iter != ObservedBonds.end(),
644 "GLWorldScene::reparentBondLeft() - bond "+toString(origin)+" is no longer stored?");
645 QtObservedBond::ptr bond = iter->second;
646 reparentBond(bond, bond->getLeftAtom(), GLMoleculeObject_bond::left);
647
648 // check whether node is still shown, otherwise remove from lists completely
649 {
650 const ObservedValue_Index_t bondid = bond->getIndex();
651 BondNodeMap::iterator iter = BondsinSceneMap.find(bondid);
652 if (iter == BondsinSceneMap.end()) {
653 // check whether both parents are now NULL
654 BondNodeParentMap_t::left_iterator leftiter = BondNodeParentMaps[0].left.find(bondid);
655 BondNodeParentMap_t::left_iterator rightiter = BondNodeParentMaps[1].left.find(bondid);
656 ASSERT( leftiter != BondNodeParentMaps[0].left.end(),
657 "GLWorldScene::reparentBondRight() - left parent to index "+toString(bondid)+" missing?");
658 ASSERT( rightiter != BondNodeParentMaps[1].left.end(),
659 "GLWorldScene::reparentBondRight() - right parent to index "+toString(bondid)+" missing?");
660 if ((leftiter->second == (ObservedValue_Index_t)NULL)
661 && (rightiter->second == (ObservedValue_Index_t)NULL)) {
662 LOG(3, "DEBUG: Reparented both left and right bond that is also no longer on display, removing.");
663 removeStoredObservedValue(ObservedValueIndexLookup, ObservedBonds, bondid);
664 }
665 }
666 }
667}
668
669/** Changes the parent of an right-side bond in the scene.
670 *
671 */
672void GLWorldScene::reparentBondRight()
673{
674 boost::recursive_mutex::scoped_lock lock(BondNodeParentMap_mutex);
675 QObject * origin = sender();
676 ObservedBonds_t::const_iterator iter = ObservedBonds.find(origin);
677 ASSERT( iter != ObservedBonds.end(),
678 "GLWorldScene::reparentBondRight() - bond "+toString(origin)+" is no longer stored?");
679 QtObservedBond::ptr bond = iter->second;
680 reparentBond(bond, bond->getRightAtom(), GLMoleculeObject_bond::right);
681
682 // check whether node is still shown, otherwise remove from lists completely
683 {
684 const ObservedValue_Index_t bondid = bond->getIndex();
685 BondNodeMap::iterator iter = BondsinSceneMap.find(bondid);
686 if (iter == BondsinSceneMap.end()) {
687 // check whether both parents are now NULL
688 BondNodeParentMap_t::left_iterator leftiter = BondNodeParentMaps[0].left.find(bondid);
689 BondNodeParentMap_t::left_iterator rightiter = BondNodeParentMaps[1].left.find(bondid);
690 ASSERT( leftiter != BondNodeParentMaps[0].left.end(),
691 "GLWorldScene::reparentBondRight() - left parent to index "+toString(bondid)+" missing?");
692 ASSERT( rightiter != BondNodeParentMaps[1].left.end(),
693 "GLWorldScene::reparentBondRight() - right parent to index "+toString(bondid)+" missing?");
694 if ((leftiter->second == (ObservedValue_Index_t)NULL)
695 && (rightiter->second == (ObservedValue_Index_t)NULL)) {
696 LOG(3, "DEBUG: Reparented both left and right bond that is also no longer on display, removing.");
697 removeStoredObservedValue(ObservedValueIndexLookup, ObservedBonds, bondid);
698 }
699 }
700 }
701}
702
703GLMoleculeObject_bond *GLWorldScene::getBondInScene(
704 const ObservedValue_Index_t _bondid,
705 GLMoleculeObject_bond::SideOfBond _side) const
706{
707 std::pair<GLWorldScene::BondNodeMap::const_iterator, GLWorldScene::BondNodeMap::const_iterator> iters =
708 BondsinSceneMap.equal_range(_bondid);
709 ASSERT( std::distance(iters.first, iters.second) == 2,
710 "GLWorldScene::getBondInScene() - not exactly two bonds of same id "
711 +toString(_bondid)+" present in scene.");
712 for (GLWorldScene::BondNodeMap::const_iterator bonditer = iters.first;
713 bonditer != iters.second; ++bonditer) {
714 if (bonditer->second->BondSide == _side)
715 return bonditer->second;
716 }
717 return NULL;
718}
719
720/** Changes the parent of an object in the scene.
721 *
722 * \param _atom atom of bond whose molecule we are associated to
723 * \param _side side of bond
724 */
725void GLWorldScene::reparentBond(
726 const QtObservedBond::ptr _bond,
727 const QtObservedAtom::ptr _atom,
728 const GLMoleculeObject_bond::SideOfBond _side)
729{
730 boost::recursive_mutex::scoped_lock lock(BondNodeParentMap_mutex);
731 const size_t dim = (_side == GLMoleculeObject_bond::left) ? 0 : 1;
732 const ObservedValue_Index_t bondid = _bond->getIndex();
733 LOG(4, "DEBUG: GLWorldScene: Received signal moleculeChanged for bond "+toString(bondid)+".");
734 BondNodeParentMap_t::left_iterator parentiter = BondNodeParentMaps[dim].left.find(bondid);
735 ASSERT( parentiter != BondNodeParentMaps[dim].left.end(),
736 "GLWorldScene::reparentBond() - could not find object to id "+toString(bondid));
737
738 // change parent entry
739 BondNodeParentMaps[dim].left.erase(bondid);
740 if ((_atom != NULL) && (_atom->getMoleculeRef() != NULL))
741 BondNodeParentMaps[dim].left.insert( std::make_pair( bondid, _atom->getMoleculeRef()->getIndex()));
742 else
743 BondNodeParentMaps[dim].left.insert( std::make_pair( bondid, (ObservedValue_Index_t)NULL) );
744 parentiter = BondNodeParentMaps[dim].left.find(bondid);
745
746 // reset parent
747 resetParent(getBondInScene(bondid, _side), getMoleculeObject(parentiter->second));
748}
749
750/** Resets the parent of an GLMoleculeObject.
751 *
752 * \param _obj object to reparent
753 * \param _molid index of parent molecule
754 */
755void GLWorldScene::resetParent(
756 GLMoleculeObject *_obj,
757 GLMoleculeObject_molecule *_molobj)
758{
759 if (_obj != NULL) {
760 QObject *parent = this;
761 if (_molobj != NULL)
762 parent = _molobj;
763 // else: molecule does not yet exist: is done when molecule is instantiated
764 LOG(5, "DEBUG: Resetting parent of " << _obj << " to " << parent);
765 _obj->setParent(parent);
766 ASSERT( _obj->parent() == parent,
767 "GLWorldScene::resetParent() - new parent "+toString(parent)+" was not set.");
768 } else
769 ELOG(1, "Object to reparent was NULL.");
770 // else object does not yet exist
771}
772
773/** Adds a shape to the scene.
774 *
775 */
776void GLWorldScene::addShape(const std::string &_name)
777{
778 Shape * const shape = ShapeRegistry::getInstance().getByName(_name);
779 if (shape != NULL) {
780 GLMoleculeObject_shape *shapeObject = new GLMoleculeObject_shape(*shape, this);
781 ShapeNodeMap::iterator iter = ShapesinSceneMap.find(_name);
782 ASSERT(iter == ShapesinSceneMap.end(),
783 "GLWorldScene::addShape() - same shape "+_name+" added again.");
784 ShapesinSceneMap.insert( make_pair(_name, shapeObject) );
785 } else
786 ELOG(2, "GLWorldScene::addShape() - shape disappeared before we could draw it.");
787
788 emit changed();
789}
790
791void GLWorldScene::removeShape(const std::string &_name)
792{
793 ShapeNodeMap::iterator iter = ShapesinSceneMap.find(_name);
794 ASSERT(iter != ShapesinSceneMap.end(),
795 "GLWorldScene::removeShape() - shape "+_name+" not in scene.");
796 ShapesinSceneMap.erase(iter);
797 delete(iter->second);
798
799 emit changed();
800}
801
802void GLWorldScene::updateSelectedShapes()
803{
804 foreach (QObject *obj, children()) {
805 GLMoleculeObject_shape *shapeobj = qobject_cast<GLMoleculeObject_shape *>(obj);
806 if (shapeobj){
807 shapeobj->enable(ShapeRegistry::getInstance().isSelected(shapeobj->getShape()));
808 }
809 }
810
811 emit changed();
812}
813
814void GLWorldScene::initialize(QGLView *view, QGLPainter *painter) const
815{
816 // Initialize all of the mesh objects that we have as children.
817 foreach (QObject *obj, children()) {
818 GLMoleculeObject *meshobj = qobject_cast<GLMoleculeObject *>(obj);
819 if (meshobj)
820 meshobj->initialize(view, painter);
821 }
822}
823
824void GLWorldScene::draw(QGLPainter *painter, const QVector4D &cameraPlane) const
825{
826 // Draw all of the mesh objects that we have as children.
827 foreach (QObject *obj, children()) {
828 GLMoleculeObject *meshobj = qobject_cast<GLMoleculeObject *>(obj);
829 if (meshobj)
830 meshobj->draw(painter, cameraPlane);
831 }
832}
833
834void GLWorldScene::setSelectionMode(SelectionModeType mode)
835{
836 selectionMode = mode;
837 // TODO send update to toolbar
838}
839
840void GLWorldScene::setSelectionModeAtom()
841{
842 setSelectionMode(SelectAtom);
843}
844
845void GLWorldScene::setSelectionModeMolecule()
846{
847 setSelectionMode(SelectMolecule);
848}
849
Note: See TracBrowser for help on using the repository browser.