| [c67518] | 1 | /* | 
|---|
|  | 2 | * Project: MoleCuilder | 
|---|
|  | 3 | * Description: creates and alters molecular systems | 
|---|
|  | 4 | * Copyright (C)  2010-2012 University of Bonn. All rights reserved. | 
|---|
| [5aaa43] | 5 | * Copyright (C)  2013 Frederik Heber. All rights reserved. | 
|---|
| [94d5ac6] | 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/>. | 
|---|
| [c67518] | 22 | */ | 
|---|
|  | 23 |  | 
|---|
|  | 24 | /* | 
|---|
|  | 25 | * GLMoleculeObject_molecule.cpp | 
|---|
|  | 26 | * | 
|---|
|  | 27 | *  Created on: Mar 30, 2012 | 
|---|
|  | 28 | *      Author: ankele | 
|---|
|  | 29 | */ | 
|---|
|  | 30 |  | 
|---|
|  | 31 |  | 
|---|
|  | 32 |  | 
|---|
|  | 33 |  | 
|---|
|  | 34 |  | 
|---|
|  | 35 | // include config.h | 
|---|
|  | 36 | #ifdef HAVE_CONFIG_H | 
|---|
|  | 37 | #include <config.h> | 
|---|
|  | 38 | #endif | 
|---|
|  | 39 |  | 
|---|
|  | 40 | #include "GLMoleculeObject_molecule.hpp" | 
|---|
|  | 41 |  | 
|---|
|  | 42 | #include <Qt3D/qglscenenode.h> | 
|---|
| [34e7fdb] | 43 | #include <Qt3D/qglbuilder.h> | 
|---|
| [c67518] | 44 |  | 
|---|
|  | 45 | #include "CodePatterns/MemDebug.hpp" | 
|---|
|  | 46 |  | 
|---|
|  | 47 | #include "CodePatterns/Assert.hpp" | 
|---|
|  | 48 | #include "CodePatterns/Log.hpp" | 
|---|
|  | 49 | #include "CodePatterns/Observer/Notification.hpp" | 
|---|
| [856d05] | 50 | #include "CodePatterns/Observer/ObserverLog.hpp" | 
|---|
| [c67518] | 51 |  | 
|---|
|  | 52 | #include "Atom/atom.hpp" | 
|---|
|  | 53 | #include "molecule.hpp" | 
|---|
|  | 54 | #include "Descriptors/AtomIdDescriptor.hpp" | 
|---|
| [704d59] | 55 | #include "Descriptors/MoleculeIdDescriptor.hpp" | 
|---|
| [c67518] | 56 | #include "Element/element.hpp" | 
|---|
|  | 57 | #include "LinearAlgebra/Vector.hpp" | 
|---|
| [34e7fdb] | 58 | #include "LinkedCell/PointCloudAdaptor.hpp" | 
|---|
|  | 59 | #include "LinkedCell/linkedcell.hpp" | 
|---|
|  | 60 | #include "Tesselation/tesselation.hpp" | 
|---|
|  | 61 | #include "Tesselation/BoundaryLineSet.hpp" | 
|---|
|  | 62 | #include "Tesselation/BoundaryTriangleSet.hpp" | 
|---|
|  | 63 | #include "Tesselation/CandidateForTesselation.hpp" | 
|---|
|  | 64 | #include "Atom/TesselPoint.hpp" | 
|---|
| [c67518] | 65 | #include "World.hpp" | 
|---|
|  | 66 |  | 
|---|
| [8c001a] | 67 | #include "GLMoleculeObject_atom.hpp" | 
|---|
|  | 68 |  | 
|---|
| [704d59] | 69 | static QGLSceneNode *createMoleculeMesh(const moleculeId_t molid, QObject *parent) | 
|---|
| [34e7fdb] | 70 | { | 
|---|
| [63fb7a] | 71 | const molecule *molref = const_cast<const World &>(World::getInstance()). | 
|---|
|  | 72 | getMolecule(MoleculeById(molid)); | 
|---|
| [704d59] | 73 | if (molref == NULL) { | 
|---|
|  | 74 | ELOG(1, "Could not createMoleculeMesh, molecule with id " << molid << " already gone."); | 
|---|
|  | 75 | return NULL; | 
|---|
|  | 76 | } | 
|---|
| [34e7fdb] | 77 | //  Shape shape = molref->getBoundingSphere(); | 
|---|
|  | 78 | double minradius = 2.; // TODO: set to maximum bond length value | 
|---|
|  | 79 | LOG(3, "DEBUG: Molecule fits into sphere of radius " << minradius); | 
|---|
| [b9b49e] | 80 | // check minimum bond radius in molecule | 
|---|
|  | 81 | double minlength = std::numeric_limits<double>::max(); | 
|---|
|  | 82 | for (molecule::const_iterator iter = molref->begin(); | 
|---|
|  | 83 | iter != molref->end(); ++iter) { | 
|---|
|  | 84 | const BondList &ListOfBonds = (*iter)->getListOfBonds(); | 
|---|
|  | 85 | for (BondList::const_iterator bonditer = ListOfBonds.begin(); | 
|---|
|  | 86 | bonditer != ListOfBonds.end(); ++bonditer) { | 
|---|
|  | 87 | const double bond_distance = (*bonditer)->GetDistance(); | 
|---|
|  | 88 | minlength = std::min(bond_distance, minlength); | 
|---|
|  | 89 | } | 
|---|
|  | 90 | } | 
|---|
|  | 91 | minradius = std::max( std::max(minradius, minlength), 1.); | 
|---|
| [34e7fdb] | 92 |  | 
|---|
|  | 93 | QGeometryData geo; | 
|---|
|  | 94 | // we need at least three points for tesselation | 
|---|
|  | 95 | if (molref->getAtomCount() >= 3) { | 
|---|
|  | 96 | // Tesselate the points. | 
|---|
|  | 97 | Tesselation T; | 
|---|
|  | 98 | PointCloudAdaptor<molecule> cloud(const_cast<molecule *>(molref), molref->getName()); | 
|---|
|  | 99 | T(cloud, minradius); | 
|---|
|  | 100 |  | 
|---|
|  | 101 | // Fill the points into a Qt geometry. | 
|---|
|  | 102 | LinkedCell_deprecated LinkedList(cloud, minradius); | 
|---|
|  | 103 | std::map<int, int> indices; | 
|---|
|  | 104 | std::map<int, Vector> normals; | 
|---|
|  | 105 | int index = 0; | 
|---|
|  | 106 | for (PointMap::const_iterator piter = T.PointsOnBoundary.begin(); | 
|---|
|  | 107 | piter != T.PointsOnBoundary.end(); ++piter) { | 
|---|
|  | 108 | const Vector &point = piter->second->getPosition(); | 
|---|
|  | 109 | // add data to the primitive | 
|---|
|  | 110 | geo.appendVertex(QVector3D(point[0], point[1], point[2])); | 
|---|
|  | 111 | Vector normalvector; | 
|---|
|  | 112 | for (LineMap::const_iterator lineiter = piter->second->lines.begin(); | 
|---|
|  | 113 | lineiter != piter->second->lines.end(); ++lineiter) | 
|---|
|  | 114 | for (TriangleMap::const_iterator triangleiter = lineiter->second->triangles.begin(); | 
|---|
|  | 115 | triangleiter != lineiter->second->triangles.end(); ++triangleiter) | 
|---|
|  | 116 | normalvector += | 
|---|
|  | 117 | triangleiter->second->NormalVector; | 
|---|
|  | 118 | normalvector.Normalize(); | 
|---|
|  | 119 | geo.appendNormal(QVector3D(normalvector[0], normalvector[1], normalvector[2])); | 
|---|
|  | 120 | geo.appendColor(QColor(1, 1, 1, 1)); | 
|---|
|  | 121 | geo.appendTexCoord(QVector2D(0, 0)); | 
|---|
|  | 122 | indices.insert( std::make_pair( piter->second->getNr(), index++)); | 
|---|
|  | 123 | } | 
|---|
|  | 124 |  | 
|---|
|  | 125 | // Fill the tesselated triangles into the geometry. | 
|---|
|  | 126 | for (TriangleMap::const_iterator runner = T.TrianglesOnBoundary.begin(); | 
|---|
|  | 127 | runner != T.TrianglesOnBoundary.end(); runner++) { | 
|---|
|  | 128 | int v[3]; | 
|---|
|  | 129 | for (size_t i=0; i<3; ++i) | 
|---|
|  | 130 | v[i] = runner->second->endpoints[i]->getNr(); | 
|---|
|  | 131 |  | 
|---|
|  | 132 | // Sort the vertices so the triangle is clockwise (relative to the normal vector). | 
|---|
|  | 133 | Vector cross = T.PointsOnBoundary[v[1]]->getPosition() - T.PointsOnBoundary[v[0]]->getPosition(); | 
|---|
|  | 134 | cross.VectorProduct(T.PointsOnBoundary[v[2]]->getPosition() - T.PointsOnBoundary[v[0]]->getPosition()); | 
|---|
|  | 135 | if (cross.ScalarProduct(runner->second->NormalVector) > 0) | 
|---|
|  | 136 | geo.appendIndices(indices[v[0]], indices[v[1]], indices[v[2]]); | 
|---|
|  | 137 | else | 
|---|
|  | 138 | geo.appendIndices(indices[v[0]], indices[v[2]], indices[v[1]]); | 
|---|
|  | 139 | } | 
|---|
|  | 140 | } | 
|---|
|  | 141 |  | 
|---|
|  | 142 | // Build a mesh from the geometry. | 
|---|
|  | 143 | QGLBuilder builder; | 
|---|
|  | 144 | builder.addTriangles(geo); | 
|---|
|  | 145 | QGLSceneNode *mesh = builder.finalizedSceneNode(); | 
|---|
|  | 146 | return mesh; | 
|---|
|  | 147 | } | 
|---|
|  | 148 |  | 
|---|
| [704d59] | 149 | GLMoleculeObject_molecule::GLMoleculeObject_molecule(QObject *parent, const moleculeId_t molid) : | 
|---|
|  | 150 | GLMoleculeObject(createMoleculeMesh(molid, parent), parent), | 
|---|
|  | 151 | Observer(std::string("GLMoleculeObject_molecule")+toString(molid)), | 
|---|
| [2b596f] | 152 | isBoundingBoxUptodate(true), | 
|---|
| [34e7fdb] | 153 | isSignedOn(false), | 
|---|
| [704d59] | 154 | moleculeid(molid), | 
|---|
| [7b5984] | 155 | TesselationHullUptodate(true), | 
|---|
| [704d59] | 156 | hoverAtomId(-1) | 
|---|
| [34e7fdb] | 157 | { | 
|---|
|  | 158 | setMaterial(getMaterial(1)); | 
|---|
|  | 159 | updateBoundingBox(); | 
|---|
|  | 160 |  | 
|---|
|  | 161 | // initially, atoms and bonds should be visible | 
|---|
|  | 162 | m_visible = false; | 
|---|
|  | 163 |  | 
|---|
| [2b596f] | 164 | connect (this, SIGNAL(hoverChanged(GLMoleculeObject *)), this, SLOT(hoverChangedSignalled(GLMoleculeObject *))); | 
|---|
|  | 165 | connect (this, SIGNAL(hoverChanged(GLMoleculeObject *)), this, SIGNAL(changed())); | 
|---|
| [9a7ef9] | 166 |  | 
|---|
|  | 167 | connect( this, SIGNAL(clicked()), this, SLOT(wasClicked())); | 
|---|
| [34e7fdb] | 168 | } | 
|---|
|  | 169 |  | 
|---|
| [704d59] | 170 | GLMoleculeObject_molecule::GLMoleculeObject_molecule(QGLSceneNode *mesh[], QObject *parent, const moleculeId_t molid) : | 
|---|
| [bca99d] | 171 | GLMoleculeObject(mesh, parent), | 
|---|
| [704d59] | 172 | Observer(std::string("GLMoleculeObject_molecule")+toString(molid)), | 
|---|
| [2b596f] | 173 | isBoundingBoxUptodate(true), | 
|---|
| [34e7fdb] | 174 | isSignedOn(false), | 
|---|
| [704d59] | 175 | moleculeid(molid), | 
|---|
| [7b5984] | 176 | TesselationHullUptodate(true), | 
|---|
| [704d59] | 177 | hoverAtomId(-1) | 
|---|
| [c67518] | 178 | { | 
|---|
| [3b229e] | 179 | setMaterial(getMaterial(1)); | 
|---|
| [d6203a] | 180 | updateBoundingBox(); | 
|---|
| [8c001a] | 181 |  | 
|---|
| [739ee9] | 182 | // initially, atoms and bonds should be visible | 
|---|
|  | 183 | m_visible = false; | 
|---|
|  | 184 |  | 
|---|
| [2b596f] | 185 | connect (this, SIGNAL(hoverChanged(GLMoleculeObject *)), this, SLOT(hoverChangedSignalled(GLMoleculeObject *))); | 
|---|
|  | 186 | connect (this, SIGNAL(hoverChanged(GLMoleculeObject *)), this, SIGNAL(changed())); | 
|---|
| [9a7ef9] | 187 |  | 
|---|
|  | 188 | connect( this, SIGNAL(clicked()), this, SLOT(wasClicked())); | 
|---|
| [c67518] | 189 | } | 
|---|
|  | 190 |  | 
|---|
|  | 191 | GLMoleculeObject_molecule::~GLMoleculeObject_molecule() | 
|---|
|  | 192 | { | 
|---|
| [34e7fdb] | 193 | if (isSignedOn) { | 
|---|
| [63fb7a] | 194 | const molecule *_molecule = const_cast<const World &>(World::getInstance()). | 
|---|
|  | 195 | getMolecule(MoleculeById(moleculeid)); | 
|---|
| [704d59] | 196 | if (_molecule != NULL) { | 
|---|
|  | 197 | _molecule->signOff(this, molecule::AtomInserted); | 
|---|
|  | 198 | _molecule->signOff(this, molecule::AtomRemoved); | 
|---|
|  | 199 | _molecule->signOff(this, molecule::AtomMoved); | 
|---|
|  | 200 | } else { | 
|---|
|  | 201 | ELOG(1, "GLMoleculeObject_molecule cannot sign off, molecule with " | 
|---|
|  | 202 | << moleculeid << " has disappeared."); | 
|---|
|  | 203 | } | 
|---|
| [34e7fdb] | 204 | } | 
|---|
| [c67518] | 205 | /*_atom->signOff(this, AtomObservable::IndexChanged); | 
|---|
|  | 206 | _atom->signOff(this, AtomObservable::PositionChanged); | 
|---|
|  | 207 | _atom->signOff(this, AtomObservable::ElementChanged); | 
|---|
|  | 208 | _atom->signOff(this, AtomObservable::BondsAdded);*/ | 
|---|
|  | 209 | World::getInstance().signOff(this, World::SelectionChanged); | 
|---|
|  | 210 | } | 
|---|
|  | 211 |  | 
|---|
| [73b13c] | 212 | void GLMoleculeObject_molecule::activateObserver() | 
|---|
|  | 213 | { | 
|---|
|  | 214 | // sign on as observer (obtain non-const instance before) | 
|---|
|  | 215 | const molecule *_molecule = const_cast<const World &>(World::getInstance()). | 
|---|
|  | 216 | getMolecule(MoleculeById(moleculeid)); | 
|---|
|  | 217 | if (_molecule != NULL) { | 
|---|
|  | 218 | _molecule->signOn(this, molecule::AtomInserted); | 
|---|
|  | 219 | _molecule->signOn(this, molecule::AtomRemoved); | 
|---|
|  | 220 | _molecule->signOn(this, molecule::AtomMoved); | 
|---|
|  | 221 | isSignedOn = true; | 
|---|
|  | 222 | } else { | 
|---|
|  | 223 | ELOG(1, "GLMoleculeObject_molecule() - added null object for not present mol id " << moleculeid); | 
|---|
|  | 224 | } | 
|---|
|  | 225 | /*molref->signOn(this, AtomObservable::IndexChanged); | 
|---|
|  | 226 | molref->signOn(this, AtomObservable::PositionChanged); | 
|---|
|  | 227 | molref->signOn(this, AtomObservable::ElementChanged); | 
|---|
|  | 228 | molref->signOn(this, AtomObservable::BondsAdded);*/ | 
|---|
|  | 229 | World::getInstance().signOn(this, World::SelectionChanged); | 
|---|
|  | 230 | } | 
|---|
|  | 231 |  | 
|---|
| [8c001a] | 232 | void GLMoleculeObject_molecule::addAtomBonds( | 
|---|
|  | 233 | const bond::ptr &_bond, | 
|---|
|  | 234 | const GLMoleculeObject_bond::SideOfBond _side | 
|---|
|  | 235 | ) | 
|---|
|  | 236 | { | 
|---|
|  | 237 | bool bond_present = false; | 
|---|
|  | 238 | const BondIds ids = getBondIds(_bond, _side); | 
|---|
|  | 239 | // check whether bond is not present already | 
|---|
|  | 240 | bond_present = BondsinSceneMap.count(ids); | 
|---|
|  | 241 | if (!bond_present) | 
|---|
|  | 242 | bondInserted(_bond, _side); | 
|---|
|  | 243 | else { | 
|---|
|  | 244 | BondsinSceneMap[ids]->resetPosition(); | 
|---|
|  | 245 | BondsinSceneMap[ids]->resetWidth(); | 
|---|
|  | 246 | } | 
|---|
|  | 247 | } | 
|---|
|  | 248 |  | 
|---|
|  | 249 | void GLMoleculeObject_molecule::addAtomBonds( | 
|---|
|  | 250 | const atom *_atom) | 
|---|
|  | 251 | { | 
|---|
|  | 252 | const bool atom_present = AtomsinSceneMap.count(_atom->getId()); | 
|---|
|  | 253 | const BondList &bondlist = _atom->getListOfBonds(); | 
|---|
|  | 254 | for (BondList::const_iterator bonditer = bondlist.begin(); | 
|---|
|  | 255 | (bonditer != bondlist.end()) && atom_present; | 
|---|
|  | 256 | ++bonditer) { | 
|---|
|  | 257 | const bond::ptr _bond = *bonditer; | 
|---|
|  | 258 | // check if OtherAtom's sphere is already present | 
|---|
|  | 259 | const atom *OtherAtom = _bond->GetOtherAtom(_atom); | 
|---|
|  | 260 | const bool otheratom_present = AtomsinSceneMap.count(OtherAtom->getId()); | 
|---|
|  | 261 | if (otheratom_present && atom_present) { | 
|---|
|  | 262 | const GLMoleculeObject_bond::SideOfBond side = (_bond->leftatom == _atom) ? | 
|---|
|  | 263 | GLMoleculeObject_bond::left : GLMoleculeObject_bond::right; | 
|---|
|  | 264 | const GLMoleculeObject_bond::SideOfBond otherside = (_bond->leftatom == _atom) ? | 
|---|
|  | 265 | GLMoleculeObject_bond::right : GLMoleculeObject_bond::left; | 
|---|
|  | 266 | addAtomBonds(_bond, side); | 
|---|
|  | 267 | addAtomBonds(_bond, otherside); | 
|---|
|  | 268 | } | 
|---|
|  | 269 | } | 
|---|
|  | 270 | } | 
|---|
|  | 271 |  | 
|---|
| [d6203a] | 272 | void GLMoleculeObject_molecule::updateBoundingBox() | 
|---|
|  | 273 | { | 
|---|
| [2b596f] | 274 | isBoundingBoxUptodate = true; | 
|---|
| [63fb7a] | 275 | const molecule * const _molecule = const_cast<const World &>(World::getInstance()). | 
|---|
|  | 276 | getMolecule(MoleculeById(moleculeid)); | 
|---|
| [704d59] | 277 | if (_molecule == NULL) { | 
|---|
|  | 278 | ELOG(1, "GLMoleculeObject_molecule cannot updateBoundingBox, molecule with " | 
|---|
|  | 279 | << moleculeid << " has disappeared."); | 
|---|
|  | 280 | return; | 
|---|
|  | 281 | } | 
|---|
| [aeb694] | 282 | Shape shape = _molecule->getBoundingSphere(); | 
|---|
| [d6203a] | 283 | Vector v = shape.getCenter(); | 
|---|
|  | 284 | setPosition(QVector3D(v[0], v[1], v[2])); | 
|---|
|  | 285 | setScale(shape.getRadius() + 0.3); // getBoundingShape() only sees atoms as points, so make the box a bit bigger | 
|---|
|  | 286 | } | 
|---|
|  | 287 |  | 
|---|
| [c67518] | 288 | void GLMoleculeObject_molecule::update(Observable *publisher) | 
|---|
|  | 289 | { | 
|---|
|  | 290 | #ifdef LOG_OBSERVER | 
|---|
| [a2a2f7] | 291 | const molecule *_mol = static_cast<molecule *>(publisher); | 
|---|
|  | 292 | observerLog().addMessage() << "++ Update of Observer " << observerLog().getName(static_cast<Observer *>(this)) << " from molecule "+toString(_mol->getId())+"."; | 
|---|
| [c67518] | 293 | #endif | 
|---|
|  | 294 | } | 
|---|
|  | 295 |  | 
|---|
|  | 296 | void GLMoleculeObject_molecule::subjectKilled(Observable *publisher) | 
|---|
| [34e7fdb] | 297 | { | 
|---|
|  | 298 | isSignedOn = false; | 
|---|
|  | 299 | } | 
|---|
| [c67518] | 300 |  | 
|---|
|  | 301 | void GLMoleculeObject_molecule::recieveNotification(Observable *publisher, Notification_ptr notification) | 
|---|
|  | 302 | { | 
|---|
| [63fb7a] | 303 | const molecule * const _molecule = const_cast<const World &>(World::getInstance()). | 
|---|
|  | 304 | getMolecule(MoleculeById(moleculeid)); | 
|---|
| [c67518] | 305 | if (publisher == dynamic_cast<const Observable*>(_molecule)){ | 
|---|
|  | 306 | // notofication from atom | 
|---|
|  | 307 | #ifdef LOG_OBSERVER | 
|---|
| [708277] | 308 | observerLog().addMessage() << "++ Update of Observer "<< observerLog().getName(static_cast<Observer *>(this)) | 
|---|
| [c67518] | 309 | << " received notification from molecule " << _molecule->getId() << " for channel " | 
|---|
|  | 310 | << notification->getChannelNo() << "."; | 
|---|
|  | 311 | #endif | 
|---|
| [8c001a] | 312 | switch (notification->getChannelNo()) { | 
|---|
|  | 313 | case molecule::AtomInserted: | 
|---|
|  | 314 | { | 
|---|
|  | 315 | const atomId_t _id = _molecule->lastChanged()->getId(); | 
|---|
|  | 316 | #ifdef LOG_OBSERVER | 
|---|
|  | 317 | observerLog().addMessage() << "++ Observer " << observerLog().getName(static_cast<Observer *>(this)) << " received notification that atom "+toString(_id)+" has been inserted."; | 
|---|
|  | 318 | #endif | 
|---|
| [9c259e] | 319 | QMetaObject::invokeMethod(this,        // pointer to a QObject | 
|---|
|  | 320 | "atomInserted",       // member name (no parameters here) | 
|---|
|  | 321 | Qt::QueuedConnection,     // connection type | 
|---|
|  | 322 | Q_ARG(atomId_t, _id));     // parameters | 
|---|
| [8c001a] | 323 | break; | 
|---|
|  | 324 | } | 
|---|
|  | 325 | case World::AtomRemoved: | 
|---|
|  | 326 | { | 
|---|
|  | 327 | const atomId_t _id = _molecule->lastChanged()->getId(); | 
|---|
|  | 328 | #ifdef LOG_OBSERVER | 
|---|
|  | 329 | observerLog().addMessage() << "++ Observer " << observerLog().getName(static_cast<Observer *>(this)) << " received notification that atom "+toString(_id)+" has been removed."; | 
|---|
|  | 330 | #endif | 
|---|
| [9c259e] | 331 | QMetaObject::invokeMethod(this,        // pointer to a QObject | 
|---|
|  | 332 | "atomRemoved",       // member name (no parameters here) | 
|---|
|  | 333 | Qt::QueuedConnection,     // connection type | 
|---|
| [2f76d2] | 334 | Q_ARG(const atomId_t, _id));     // parameters | 
|---|
| [8c001a] | 335 | break; | 
|---|
|  | 336 | } | 
|---|
| [7b5984] | 337 | case molecule::AtomMoved: | 
|---|
|  | 338 | { | 
|---|
|  | 339 | #ifdef LOG_OBSERVER | 
|---|
| [2b596f] | 340 | const atomId_t _id = _molecule->lastChanged()->getId(); | 
|---|
| [7b5984] | 341 | observerLog().addMessage() << "++ Observer " << observerLog().getName(static_cast<Observer *>(this)) << " received notification that atom "+toString(_id)+" has been inserted."; | 
|---|
|  | 342 | #endif | 
|---|
|  | 343 | TesselationHullUptodate = false; | 
|---|
| [2b596f] | 344 | isBoundingBoxUptodate = false; | 
|---|
| [7b5984] | 345 | break; | 
|---|
|  | 346 | } | 
|---|
| [8c001a] | 347 | default: | 
|---|
|  | 348 | break; | 
|---|
|  | 349 | } | 
|---|
| [c67518] | 350 | }else{ | 
|---|
|  | 351 | // notification from world | 
|---|
|  | 352 | #ifdef LOG_OBSERVER | 
|---|
| [708277] | 353 | observerLog().addMessage() << "++ Update of Observer "<< observerLog().getName(static_cast<Observer *>(this)) | 
|---|
| [c67518] | 354 | << " received notification from world for channel " | 
|---|
|  | 355 | << notification->getChannelNo() << "."; | 
|---|
|  | 356 | #endif | 
|---|
|  | 357 | switch (notification->getChannelNo()) { | 
|---|
|  | 358 | case World::SelectionChanged: | 
|---|
| [704d59] | 359 | if (_molecule != NULL) { | 
|---|
|  | 360 | setSelected(World::getInstance().isSelected(_molecule)); | 
|---|
|  | 361 | } else { | 
|---|
|  | 362 | ELOG(1, "GLMoleculeObject_molecule cannot change selection, molecule with " | 
|---|
|  | 363 | << moleculeid << " has disappeared."); | 
|---|
|  | 364 | } | 
|---|
| [c67518] | 365 | break; | 
|---|
|  | 366 | default: | 
|---|
|  | 367 | break; | 
|---|
|  | 368 | } | 
|---|
|  | 369 | } | 
|---|
|  | 370 | } | 
|---|
|  | 371 |  | 
|---|
| [8c001a] | 372 | void GLMoleculeObject_molecule::initialize(QGLView *view, QGLPainter *painter) | 
|---|
|  | 373 | { | 
|---|
|  | 374 | // Initialize all of the mesh objects that we have as children. | 
|---|
| [2b596f] | 375 | if (m_visible) { | 
|---|
|  | 376 | GLMoleculeObject::initialize(view, painter); | 
|---|
|  | 377 | } else { | 
|---|
| [8c001a] | 378 | foreach (QObject *obj, children()) { | 
|---|
|  | 379 | GLMoleculeObject *meshobj = qobject_cast<GLMoleculeObject *>(obj); | 
|---|
|  | 380 | if (meshobj) | 
|---|
|  | 381 | meshobj->initialize(view, painter); | 
|---|
|  | 382 | } | 
|---|
| [2b596f] | 383 | } | 
|---|
| [8c001a] | 384 | } | 
|---|
|  | 385 |  | 
|---|
|  | 386 | void GLMoleculeObject_molecule::draw(QGLPainter *painter, const QVector4D &cameraPlane) | 
|---|
|  | 387 | { | 
|---|
| [739ee9] | 388 | // draw either molecule's mesh or all atoms and bonds | 
|---|
|  | 389 | if (m_visible) { | 
|---|
| [7b5984] | 390 | updateTesselationHull(); | 
|---|
|  | 391 |  | 
|---|
| [34e7fdb] | 392 | painter->modelViewMatrix().push(); | 
|---|
|  | 393 |  | 
|---|
|  | 394 | // Apply the material and effect to the painter. | 
|---|
|  | 395 | QGLMaterial *material; | 
|---|
|  | 396 | if (m_hovering) | 
|---|
|  | 397 | material = m_hoverMaterial; | 
|---|
|  | 398 | else if (m_selected) | 
|---|
|  | 399 | material = m_selectionMaterial; | 
|---|
|  | 400 | else | 
|---|
|  | 401 | material = m_material; | 
|---|
|  | 402 |  | 
|---|
|  | 403 | ASSERT(material, "GLMoleculeObject::draw: chosen material is NULL"); | 
|---|
|  | 404 |  | 
|---|
|  | 405 | painter->setColor(material->diffuseColor()); | 
|---|
|  | 406 | painter->setFaceMaterial(QGL::AllFaces, material); | 
|---|
|  | 407 | if (m_effect) | 
|---|
|  | 408 | painter->setUserEffect(m_effect); | 
|---|
|  | 409 | else | 
|---|
|  | 410 | painter->setStandardEffect(QGL::LitMaterial); | 
|---|
|  | 411 |  | 
|---|
|  | 412 | // Mark the object for object picking purposes. | 
|---|
|  | 413 | int prevObjectId = painter->objectPickId(); | 
|---|
|  | 414 | if (m_objectId != -1) | 
|---|
|  | 415 | painter->setObjectPickId(m_objectId); | 
|---|
|  | 416 |  | 
|---|
|  | 417 | m_mesh[0]->draw(painter); | 
|---|
|  | 418 |  | 
|---|
|  | 419 | // Turn off the user effect, if present. | 
|---|
|  | 420 | if (m_effect) | 
|---|
|  | 421 | painter->setStandardEffect(QGL::LitMaterial); | 
|---|
|  | 422 |  | 
|---|
|  | 423 | // Revert to the previous object identifier. | 
|---|
|  | 424 | painter->setObjectPickId(prevObjectId); | 
|---|
|  | 425 |  | 
|---|
|  | 426 | // Restore the modelview matrix. | 
|---|
|  | 427 | painter->modelViewMatrix().pop(); | 
|---|
|  | 428 |  | 
|---|
|  | 429 | //    GLMoleculeObject::draw(painter, cameraPlane); | 
|---|
| [739ee9] | 430 | } else { | 
|---|
|  | 431 | // Draw all of the mesh objects that we have as children. | 
|---|
|  | 432 | foreach (QObject *obj, children()) { | 
|---|
|  | 433 | GLMoleculeObject *meshobj = qobject_cast<GLMoleculeObject *>(obj); | 
|---|
|  | 434 | if (meshobj) | 
|---|
|  | 435 | meshobj->draw(painter, cameraPlane); | 
|---|
|  | 436 | } | 
|---|
| [2b596f] | 437 |  | 
|---|
|  | 438 | // update bounding box prior to selection | 
|---|
|  | 439 | if (!isBoundingBoxUptodate) | 
|---|
|  | 440 | updateBoundingBox(); | 
|---|
|  | 441 |  | 
|---|
|  | 442 | painter->modelViewMatrix().push(); | 
|---|
|  | 443 | painter->modelViewMatrix().translate(m_position); | 
|---|
|  | 444 | if (m_rotationAngle != 0.0f) | 
|---|
|  | 445 | painter->modelViewMatrix().rotate(m_rotationAngle, m_rotationVector); | 
|---|
| [f47efd4] | 446 | if ((m_scaleX != 1.0f) || (m_scaleY != 1.0f) || (m_scaleZ != 1.0f)) | 
|---|
|  | 447 | painter->modelViewMatrix().scale(m_scaleX, m_scaleY, m_scaleZ); | 
|---|
| [2b596f] | 448 |  | 
|---|
|  | 449 | // Draw a box around the mesh, if selected. | 
|---|
|  | 450 | if (m_selected) | 
|---|
|  | 451 | drawSelectionBox(painter); | 
|---|
|  | 452 |  | 
|---|
|  | 453 | // Restore the modelview matrix. | 
|---|
|  | 454 | painter->modelViewMatrix().pop(); | 
|---|
| [739ee9] | 455 | } | 
|---|
| [8c001a] | 456 | } | 
|---|
|  | 457 |  | 
|---|
|  | 458 | /** Adds an atom of this molecule to the scene. | 
|---|
|  | 459 | * | 
|---|
|  | 460 | * @param _atom atom to add | 
|---|
|  | 461 | */ | 
|---|
| [2f76d2] | 462 | void GLMoleculeObject_molecule::atomInserted(const atomId_t _id) | 
|---|
| [8c001a] | 463 | { | 
|---|
| [9c259e] | 464 | LOG(3, "INFO: GLMoleculeObject_molecule: Received signal atomInserted for atom "+toString(_id)+"."); | 
|---|
|  | 465 |  | 
|---|
|  | 466 | TesselationHullUptodate = false; | 
|---|
|  | 467 | isBoundingBoxUptodate = false; | 
|---|
|  | 468 |  | 
|---|
| [8923ad8] | 469 | GLMoleculeObject_atom *atomObject = new GLMoleculeObject_atom(GLMoleculeObject::meshSphere, this, _id); | 
|---|
|  | 470 | ASSERT( atomObject != NULL, | 
|---|
|  | 471 | "GLMoleculeObject_molecule::atomInserted - could not create atom object for "+toString(_id)); | 
|---|
|  | 472 | AtomNodeMap::iterator iter = AtomsinSceneMap.find(_id); | 
|---|
|  | 473 | ASSERT(iter == AtomsinSceneMap.end(), | 
|---|
|  | 474 | "GLMoleculeObject_molecule::atomInserted - same atom with id "+toString(_id)+" added again."); | 
|---|
|  | 475 | AtomsinSceneMap.insert( make_pair(_id, atomObject) ); | 
|---|
|  | 476 |  | 
|---|
|  | 477 | qRegisterMetaType<atomId_t>("atomId_t"); | 
|---|
|  | 478 | qRegisterMetaType<bond::ptr>("bond::ptr"); | 
|---|
|  | 479 | qRegisterMetaType<GLMoleculeObject_bond::SideOfBond>("GLMoleculeObject_bond::SideOfBond"); | 
|---|
|  | 480 | connect (atomObject, SIGNAL(clicked(atomId_t)), this, SIGNAL(atomClicked(atomId_t))); | 
|---|
|  | 481 | connect (atomObject, SIGNAL(changed()), this, SIGNAL(changed())); | 
|---|
|  | 482 | connect (atomObject, SIGNAL(hoverChanged(GLMoleculeObject *)), this, SIGNAL(changed())); | 
|---|
|  | 483 | connect (atomObject, SIGNAL(hoverChanged(GLMoleculeObject *)), this, SLOT(hoverChangedSignalled(GLMoleculeObject *))); | 
|---|
|  | 484 | connect (atomObject, SIGNAL(selectionChanged()), this, SIGNAL(changed())); | 
|---|
|  | 485 | connect (atomObject, SIGNAL(BondsInserted(const bond::ptr , const GLMoleculeObject_bond::SideOfBond)), this, SLOT(bondInserted(const bond::ptr , const GLMoleculeObject_bond::SideOfBond))); | 
|---|
|  | 486 | connect (atomObject, SIGNAL(indexChanged(GLMoleculeObject_atom*, int, int)), this, SLOT(changeAtomId(GLMoleculeObject_atom*, int, int))); | 
|---|
|  | 487 |  | 
|---|
|  | 488 | isBoundingBoxUptodate = false; | 
|---|
|  | 489 |  | 
|---|
|  | 490 | if (m_objectId  == -1) | 
|---|
|  | 491 | setObjectId(_id); | 
|---|
| [8c001a] | 492 |  | 
|---|
| [52cd7b] | 493 | // add all bonds | 
|---|
| [f01769] | 494 | const atom * const Walker = const_cast<const World &>(World::getInstance()). | 
|---|
|  | 495 | getAtom(AtomById(_id)); | 
|---|
| [704d59] | 496 | if (Walker != NULL) | 
|---|
|  | 497 | addAtomBonds(Walker); | 
|---|
|  | 498 | else | 
|---|
|  | 499 | ELOG(1, "GLMoleculeObject_atom disappeared while about to add bonds."); | 
|---|
| [52cd7b] | 500 |  | 
|---|
| [8c001a] | 501 | emit changeOccured(); | 
|---|
|  | 502 | } | 
|---|
|  | 503 |  | 
|---|
|  | 504 | /** Removes an atom of this molecule from the scene. | 
|---|
|  | 505 | * | 
|---|
|  | 506 | * We just the id as the atom might have already been destroyed. | 
|---|
|  | 507 | * | 
|---|
|  | 508 | * @param _id id of atom to remove | 
|---|
|  | 509 | */ | 
|---|
| [2f76d2] | 510 | void GLMoleculeObject_molecule::atomRemoved(const atomId_t _id) | 
|---|
| [8c001a] | 511 | { | 
|---|
| [9c259e] | 512 | LOG(3, "INFO: GLMoleculeObject_molecule: Received signal atomRemoved for atom "+toString(_id)+"."); | 
|---|
| [8c001a] | 513 | // bonds are removed by signal coming from ~bond | 
|---|
| [2b596f] | 514 |  | 
|---|
| [9c259e] | 515 | TesselationHullUptodate = false; | 
|---|
|  | 516 | isBoundingBoxUptodate = false; | 
|---|
|  | 517 |  | 
|---|
| [704d59] | 518 | if ((unsigned int)m_objectId == _id) | 
|---|
| [2b596f] | 519 | setObjectId(-1); | 
|---|
|  | 520 |  | 
|---|
| [8c001a] | 521 | // remove atoms | 
|---|
|  | 522 | AtomNodeMap::iterator iter = AtomsinSceneMap.find(_id); | 
|---|
|  | 523 | ASSERT(iter != AtomsinSceneMap.end(), | 
|---|
| [73b13c] | 524 | "GLMoleculeObject_molecule::atomRemoved() - atom "+toString(_id)+" not on display."); | 
|---|
| [8c001a] | 525 | GLMoleculeObject_atom *atomObject = iter->second; | 
|---|
|  | 526 | AtomsinSceneMap.erase(iter); | 
|---|
| [704d59] | 527 | atomObject->disconnect(); | 
|---|
| [8c001a] | 528 | delete atomObject; | 
|---|
|  | 529 |  | 
|---|
| [2b596f] | 530 | isBoundingBoxUptodate = false; | 
|---|
| [8c001a] | 531 |  | 
|---|
|  | 532 | emit changeOccured(); | 
|---|
|  | 533 | } | 
|---|
|  | 534 |  | 
|---|
|  | 535 | void GLMoleculeObject_molecule::hoverChangedSignalled(GLMoleculeObject *ob) | 
|---|
|  | 536 | { | 
|---|
|  | 537 | // Find the atom, ob corresponds to. | 
|---|
| [704d59] | 538 | hoverAtomId = -1; | 
|---|
| [8c001a] | 539 | GLMoleculeObject_atom *atomObject = dynamic_cast<GLMoleculeObject_atom *>(ob); | 
|---|
|  | 540 | if (atomObject){ | 
|---|
|  | 541 | for (AtomNodeMap::iterator iter = AtomsinSceneMap.begin();iter != AtomsinSceneMap.end(); ++ iter){ | 
|---|
|  | 542 | if (iter->second == atomObject) | 
|---|
| [704d59] | 543 | hoverAtomId = iter->first; | 
|---|
| [8c001a] | 544 | } | 
|---|
|  | 545 |  | 
|---|
| [2b596f] | 546 | // Propagate signal. | 
|---|
| [704d59] | 547 | emit hoverChanged(hoverAtomId); | 
|---|
| [2b596f] | 548 | } else { | 
|---|
|  | 549 | // Find the atom, ob corresponds to. | 
|---|
|  | 550 | GLMoleculeObject_molecule *moleculeObject = dynamic_cast<GLMoleculeObject_molecule *>(ob); | 
|---|
|  | 551 | if (moleculeObject == this){ | 
|---|
|  | 552 | // Propagate signal. | 
|---|
| [704d59] | 553 | emit hoverChanged(moleculeid, 0); | 
|---|
| [2b596f] | 554 | } | 
|---|
|  | 555 | } | 
|---|
| [8c001a] | 556 | } | 
|---|
|  | 557 |  | 
|---|
|  | 558 |  | 
|---|
|  | 559 | /** Helper function to get bond ids in the correct order for BondNodeMap. | 
|---|
|  | 560 | * | 
|---|
|  | 561 | * \return pair of ids in correct order. | 
|---|
|  | 562 | */ | 
|---|
|  | 563 | GLMoleculeObject_molecule::BondIds GLMoleculeObject_molecule::getBondIds( | 
|---|
|  | 564 | const bond::ptr _bond, | 
|---|
|  | 565 | const enum GLMoleculeObject_bond::SideOfBond _side) | 
|---|
|  | 566 | { | 
|---|
|  | 567 | BondIds ids; | 
|---|
|  | 568 | switch (_side) { | 
|---|
|  | 569 | case GLMoleculeObject_bond::left: | 
|---|
|  | 570 | ids = std::make_pair(_bond->leftatom->getId(), _bond->rightatom->getId()); | 
|---|
|  | 571 | break; | 
|---|
|  | 572 | case GLMoleculeObject_bond::right: | 
|---|
|  | 573 | ids = std::make_pair(_bond->rightatom->getId(), _bond->leftatom->getId()); | 
|---|
|  | 574 | break; | 
|---|
|  | 575 | } | 
|---|
|  | 576 | return ids; | 
|---|
|  | 577 | } | 
|---|
|  | 578 |  | 
|---|
|  | 579 | /** Adds a bond to the scene. | 
|---|
|  | 580 | * | 
|---|
|  | 581 | * @param _bond bond to add | 
|---|
|  | 582 | * @param side which side of the bond (left or right) | 
|---|
|  | 583 | */ | 
|---|
|  | 584 | void GLMoleculeObject_molecule::bondInserted(const bond::ptr _bond, const enum GLMoleculeObject_bond::SideOfBond _side) | 
|---|
|  | 585 | { | 
|---|
|  | 586 | LOG(3, "INFO: GLWorldScene::bondInserted() - Adding bond "+toString(*_bond)+"."); | 
|---|
|  | 587 | //LOG(4, "INFO: Currently present bonds " << BondsinSceneMap << "."); | 
|---|
|  | 588 |  | 
|---|
|  | 589 | const BondIds ids = getBondIds(_bond, _side); | 
|---|
|  | 590 | BondNodeMap::iterator iter = BondsinSceneMap.find(ids); | 
|---|
|  | 591 | if (iter == BondsinSceneMap.end()) { | 
|---|
|  | 592 | GLMoleculeObject_bond * bondObject = | 
|---|
| [009e2e2] | 593 | new GLMoleculeObject_bond(GLMoleculeObject::meshCylinder, this, ids, _side); | 
|---|
| [8c001a] | 594 | connect ( | 
|---|
|  | 595 | bondObject, SIGNAL(BondRemoved(const atomId_t, const atomId_t)), | 
|---|
|  | 596 | this, SLOT(bondRemoved(const atomId_t, const atomId_t))); | 
|---|
|  | 597 | connect (bondObject, SIGNAL(changed()), this, SIGNAL(changed())); | 
|---|
|  | 598 | BondsinSceneMap.insert( make_pair(ids, bondObject) ); | 
|---|
|  | 599 | //    BondIdsinSceneMap.insert( Leftids ); | 
|---|
|  | 600 | } else { | 
|---|
|  | 601 | iter->second->resetPosition(); | 
|---|
|  | 602 | iter->second->resetWidth(); | 
|---|
|  | 603 | } | 
|---|
|  | 604 | emit changeOccured(); | 
|---|
|  | 605 | } | 
|---|
|  | 606 |  | 
|---|
|  | 607 | /** Removes a bond from the scene. | 
|---|
|  | 608 | * | 
|---|
|  | 609 | * @param _bond bond to remove | 
|---|
|  | 610 | */ | 
|---|
|  | 611 | void GLMoleculeObject_molecule::bondRemoved(const atomId_t leftnr, const atomId_t rightnr) | 
|---|
|  | 612 | { | 
|---|
|  | 613 | LOG(3, "INFO: GLWorldScene::bondRemoved() - Removing bond between "+toString(leftnr)+" and "+toString(rightnr)+"."); | 
|---|
|  | 614 | { | 
|---|
|  | 615 | // left bond | 
|---|
|  | 616 | const BondIds Leftids( make_pair(leftnr, rightnr) ); | 
|---|
|  | 617 | BondNodeMap::iterator leftiter = BondsinSceneMap.find( Leftids ); | 
|---|
|  | 618 | ASSERT(leftiter != BondsinSceneMap.end(), | 
|---|
|  | 619 | "GLWorldScene::bondRemoved() - bond "+toString(leftnr)+"-" | 
|---|
|  | 620 | +toString(rightnr)+" not on display."); | 
|---|
|  | 621 | GLMoleculeObject_bond *bondObject = leftiter->second; | 
|---|
|  | 622 | bondObject->disconnect(); | 
|---|
|  | 623 | BondsinSceneMap.erase(leftiter); | 
|---|
|  | 624 | delete bondObject; // is done by signal from bond itself | 
|---|
|  | 625 | //LOG(4, "INFO: Still present bonds " << BondsinSceneMap << "."); | 
|---|
|  | 626 | } | 
|---|
|  | 627 |  | 
|---|
|  | 628 | emit changeOccured(); | 
|---|
|  | 629 | } | 
|---|
|  | 630 |  | 
|---|
| [34e7fdb] | 631 | void GLMoleculeObject_molecule::setVisible(bool value) | 
|---|
|  | 632 | { | 
|---|
|  | 633 | // first update the mesh if we are going to be visible now | 
|---|
|  | 634 | if (value) | 
|---|
| [7b5984] | 635 | updateTesselationHull(); | 
|---|
| [34e7fdb] | 636 | // then emit onward | 
|---|
|  | 637 | GLMoleculeObject::setVisible(value); | 
|---|
|  | 638 | } | 
|---|
|  | 639 |  | 
|---|
| [7b5984] | 640 | void GLMoleculeObject_molecule::updateTesselationHull() | 
|---|
|  | 641 | { | 
|---|
|  | 642 | if (!TesselationHullUptodate) { | 
|---|
| [704d59] | 643 | updateMesh(createMoleculeMesh(moleculeid, parent())); | 
|---|
| [7b5984] | 644 | TesselationHullUptodate = true; | 
|---|
|  | 645 | } | 
|---|
|  | 646 | } | 
|---|
|  | 647 |  | 
|---|
| [8c001a] | 648 | std::ostream &operator<<(std::ostream &ost, const GLMoleculeObject_molecule::BondIds &t) | 
|---|
|  | 649 | { | 
|---|
|  | 650 | ost << t.first << "," << t.second; | 
|---|
|  | 651 | return ost; | 
|---|
|  | 652 | } | 
|---|
| [34e7fdb] | 653 |  | 
|---|
| [9a7ef9] | 654 | void GLMoleculeObject_molecule::wasClicked() | 
|---|
|  | 655 | { | 
|---|
| [704d59] | 656 | LOG(4, "INFO: GLMoleculeObject_molecule: atom " << moleculeid << " has been clicked"); | 
|---|
|  | 657 | emit moleculeClicked(moleculeid); | 
|---|
| [9a7ef9] | 658 | } | 
|---|
| [8d3ee6] | 659 |  | 
|---|
|  | 660 | void GLMoleculeObject_molecule::changeAtomId(GLMoleculeObject_atom *ob, int oldId, int newId) | 
|---|
|  | 661 | { | 
|---|
|  | 662 | LOG(3, "INFO: GLMoleculeObject_molecule - change atom id " << oldId << " to " << newId << "."); | 
|---|
|  | 663 |  | 
|---|
|  | 664 | // Remove from map. | 
|---|
|  | 665 | AtomNodeMap::iterator iter = AtomsinSceneMap.find(oldId); | 
|---|
|  | 666 | ASSERT(iter != AtomsinSceneMap.end(), | 
|---|
|  | 667 | "GLMoleculeObject_molecule::changeAtomId() - atom with old id "+toString(oldId)+" not on display."); | 
|---|
|  | 668 | ASSERT(iter->second == ob, | 
|---|
|  | 669 | "GLMoleculeObject_molecule::changeAtomId() - atom with id " | 
|---|
|  | 670 | +toString(oldId)+" does not match with object in AtomsinSceneMap."); | 
|---|
|  | 671 | AtomsinSceneMap.erase(iter); | 
|---|
|  | 672 |  | 
|---|
|  | 673 | // Reinsert with new id. | 
|---|
|  | 674 | { | 
|---|
|  | 675 | AtomNodeMap::iterator iter = AtomsinSceneMap.find(newId); | 
|---|
|  | 676 | ASSERT(iter == AtomsinSceneMap.end(), | 
|---|
|  | 677 | "GLMoleculeObject_molecule::changeAtomId() - atom with new id "+toString(newId)+" already known."); | 
|---|
|  | 678 | } | 
|---|
|  | 679 | AtomsinSceneMap.insert( make_pair(newId, ob) ); | 
|---|
|  | 680 | } | 
|---|