source: src/World.cpp@ a58c16

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 Candidate_v1.7.0 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 a58c16 was 795c0f, checked in by Frederik Heber <heber@…>, 10 years ago

World::getAtoms() now has const versions as well.

  • this in turn required to adapt all AtomDescriptors and some fixes, e.g. predicate is now const member and works on const atom ptr.
  • Property mode set to 100644
File size: 28.7 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 * World.cpp
26 *
27 * Created on: Feb 3, 2010
28 * Author: crueger
29 */
30
31// include config.h
32#ifdef HAVE_CONFIG_H
33#include <config.h>
34#endif
35
36#include "CodePatterns/MemDebug.hpp"
37
38#include "World.hpp"
39
40#include <functional>
41
42#include "Actions/ActionTrait.hpp"
43#include "Actions/ManipulateAtomsProcess.hpp"
44#include "Atom/atom.hpp"
45#include "Box.hpp"
46#include "CodePatterns/Assert.hpp"
47#include "config.hpp"
48#include "Descriptors/AtomDescriptor.hpp"
49#include "Descriptors/AtomDescriptor_impl.hpp"
50#include "Descriptors/AtomIdDescriptor.hpp"
51#include "Descriptors/AtomSelectionDescriptor.hpp"
52#include "Descriptors/MoleculeDescriptor.hpp"
53#include "Descriptors/MoleculeDescriptor_impl.hpp"
54#include "Descriptors/MoleculeIdDescriptor.hpp"
55#include "Descriptors/MoleculeSelectionDescriptor.hpp"
56#include "Descriptors/SelectiveConstIterator_impl.hpp"
57#include "Descriptors/SelectiveIterator_impl.hpp"
58#include "Element/periodentafel.hpp"
59#include "Fragmentation/Homology/HomologyContainer.hpp"
60#include "Graph/BondGraph.hpp"
61#include "Graph/DepthFirstSearchAnalysis.hpp"
62#include "Helpers/defs.hpp"
63#include "LinearAlgebra/RealSpaceMatrix.hpp"
64#include "LinkedCell/LinkedCell_Controller.hpp"
65#include "LinkedCell/PointCloudAdaptor.hpp"
66#include "molecule.hpp"
67#include "MoleculeListClass.hpp"
68#include "Thermostats/ThermoStatContainer.hpp"
69#include "WorldTime.hpp"
70
71#include "IdPool_impl.hpp"
72
73#include "CodePatterns/IteratorAdaptors.hpp"
74#include "CodePatterns/Singleton_impl.hpp"
75#include "CodePatterns/Observer/Channels.hpp"
76#include "CodePatterns/Observer/ObservedContainer_impl.hpp"
77
78using namespace MoleCuilder;
79
80/******************************* Notifications ************************/
81
82
83atom* World::_lastchangedatom = NULL;
84molecule* World::_lastchangedmol = NULL;
85
86/******************************* getter and setter ************************/
87periodentafel *&World::getPeriode()
88{
89 return periode;
90}
91
92BondGraph *&World::getBondGraph()
93{
94 return BG;
95}
96
97HomologyContainer &World::getHomologies()
98{
99 return *homologies;
100}
101
102void World::resetHomologies(HomologyContainer *&_homologies)
103{
104 HomologyContainer *oldhomologies = homologies;
105
106 // install new instance, resetting given pointer
107 homologies = _homologies;
108 _homologies = NULL;
109
110 // delete old instance which also informs all observers
111 delete oldhomologies;
112}
113
114void World::setBondGraph(BondGraph *_BG){
115 delete (BG);
116 BG = _BG;
117}
118
119config *&World::getConfig(){
120 return configuration;
121}
122
123// Atoms
124
125atom* World::getAtom(AtomDescriptor descriptor){
126 return descriptor.find();
127}
128
129const atom* World::getAtom(AtomDescriptor descriptor) const{
130 return const_cast<const AtomDescriptor &>(descriptor).find();
131}
132
133World::AtomComposite World::getAllAtoms(AtomDescriptor descriptor){
134 return descriptor.findAll();
135}
136
137World::ConstAtomComposite World::getAllAtoms(AtomDescriptor descriptor) const {
138 return const_cast<const AtomDescriptor &>(descriptor).findAll();
139}
140
141World::AtomComposite World::getAllAtoms(){
142 return getAllAtoms(AllAtoms());
143}
144
145World::ConstAtomComposite World::getAllAtoms() const {
146 return getAllAtoms(AllAtoms());
147}
148
149int World::numAtoms() const {
150 return atoms.size();
151}
152
153// Molecules
154
155molecule *World::getMolecule(MoleculeDescriptor descriptor){
156 return descriptor.find();
157}
158
159const molecule *World::getMolecule(MoleculeDescriptor descriptor) const {
160 return const_cast<const MoleculeDescriptor &>(descriptor).find();
161}
162
163std::vector<molecule*> World::getAllMolecules(MoleculeDescriptor descriptor){
164 return descriptor.findAll();
165}
166
167std::vector<const molecule*> World::getAllMolecules(MoleculeDescriptor descriptor) const {
168 return const_cast<const MoleculeDescriptor &>(descriptor).findAll();
169}
170
171std::vector<molecule*> World::getAllMolecules(){
172 return getAllMolecules(AllMolecules());
173}
174
175std::vector<const molecule*> World::getAllMolecules() const {
176 return getAllMolecules(AllMolecules());
177}
178
179int World::numMolecules() const {
180 return molecules_deprecated->ListOfMolecules.size();
181}
182
183// system
184
185Box& World::getDomain() {
186 return *cell_size;
187}
188
189void World::setDomain(const RealSpaceMatrix &mat){
190 OBSERVE;
191 *cell_size = mat;
192}
193
194void World::setDomain(double * matrix)
195{
196 OBSERVE;
197 RealSpaceMatrix M = ReturnFullMatrixforSymmetric(matrix);
198 cell_size->setM(M);
199}
200
201LinkedCell::LinkedCell_View World::getLinkedCell(double distance)
202{
203 ASSERT( distance >= 0,
204 "World::getLinkedCell() - distance is not positive.");
205 if (distance < 1.) {
206 ELOG(2, "Linked cell grid with length less than 1. is very memory-intense!");
207 distance = 1.;
208 }
209 // we have to grope past the ObservedContainer mechanism and transmorph the map
210 // into a traversable list for the adaptor
211 PointCloudAdaptor< AtomSet::set_t, MapValueIterator<AtomSet::set_t::iterator> > atomset(
212 &(atoms.getContent()),
213 std::string("WorldsAtoms"));
214 return LCcontroller->getView(distance, atomset);
215}
216
217const unsigned World::getTime() const
218{
219 return WorldTime::getTime();
220}
221
222bool areBondsPresent(const unsigned int _step)
223{
224 bool status = false;
225
226 for (World::AtomConstIterator iter = const_cast<const World &>(World::getInstance()).getAtomIter();
227 (!status) && (iter != const_cast<const World &>(World::getInstance()).atomEnd()); ++iter) {
228 const atom * const Walker = *iter;
229 status |= !Walker->getListOfBondsAtStep(_step).empty();
230 }
231
232 return status;
233}
234
235void copyBondgraph(const unsigned int _srcstep, const unsigned int _deststep)
236{
237 // gather all bonds from _srcstep
238 std::set<bond *> SetOfBonds;
239 for (World::AtomConstIterator iter = const_cast<const World &>(World::getInstance()).getAtomIter();
240 iter != const_cast<const World &>(World::getInstance()).atomEnd(); ++iter) {
241 const atom * const Walker = *iter;
242 const BondList bonds = Walker->getListOfBondsAtStep(_srcstep);
243 BOOST_FOREACH( bond::ptr bondptr, bonds) {
244 SetOfBonds.insert(bondptr.get());
245 }
246 }
247 LOG(4, "DEBUG: We gathered " << SetOfBonds.size() << " bonds in total.");
248
249 // copy bond to new time step
250 for (std::set<bond *>::const_iterator bonditer = SetOfBonds.begin();
251 bonditer != SetOfBonds.end(); ++bonditer) {
252 const atom * const Walker = (*bonditer)->leftatom;
253 const atom * const OtherWalker = (*bonditer)->rightatom;
254 bond::ptr const _bond =
255 const_cast<atom *>(Walker)->addBond(_deststep, const_cast<atom *>(OtherWalker));
256 _bond->setDegree((*bonditer)->getDegree());
257 }
258}
259
260void World::setTime(const unsigned int _step)
261{
262 if (_step != WorldTime::getTime()) {
263 const unsigned int oldstep = WorldTime::getTime();
264
265 // 1. copy bond graph (such not each addBond causes GUI update)
266 if (!areBondsPresent(_step)) {
267// AtomComposite Set = getAllAtoms();
268// BG->cleanAdjacencyList(Set);
269 copyBondgraph(oldstep, _step);
270 }
271
272 // 2. set new time
273 WorldTime::getInstance().setTime(_step);
274
275 // 4. scan for connected subgraphs => molecules
276 DepthFirstSearchAnalysis DFS;
277 DFS();
278 DFS.UpdateMoleculeStructure();
279 }
280}
281
282std::string World::getDefaultName() {
283 return defaultName;
284}
285
286void World::setDefaultName(std::string name)
287{
288 OBSERVE;
289 defaultName = name;
290};
291
292class ThermoStatContainer * World::getThermostats()
293{
294 return Thermostats;
295}
296
297
298int World::getExitFlag() {
299 return ExitFlag;
300}
301
302void World::setExitFlag(int flag) {
303 if (ExitFlag < flag)
304 ExitFlag = flag;
305}
306
307/******************** Methods to change World state *********************/
308
309molecule* World::createMolecule(){
310 OBSERVE;
311 molecule *mol = NULL;
312 mol = NewMolecule();
313 moleculeId_t id = moleculeIdPool.getNextId();
314 ASSERT(!molecules.count(id),"proposed id did not specify an unused ID");
315 mol->setId(id);
316 // store the molecule by ID
317 molecules[mol->getId()] = mol;
318 _lastchangedmol = mol;
319 NOTIFY(MoleculeInserted);
320 return mol;
321}
322
323void World::destroyMolecule(molecule* mol){
324 ASSERT(mol,"Molecule that was meant to be destroyed did not exist");
325 destroyMolecule(mol->getId());
326}
327
328void World::destroyMolecule(moleculeId_t id){
329 molecule *mol = molecules[id];
330 ASSERT(mol,"Molecule id that was meant to be destroyed did not exist");
331 // give notice about immediate removal
332 {
333 OBSERVE;
334 _lastchangedmol = mol;
335 NOTIFY(MoleculeRemoved);
336 }
337 // TODO: removed when depcreated MoleculeListClass is gone
338 molecules_deprecated->erase(mol);
339 DeleteMolecule(mol);
340 if (isMoleculeSelected(id))
341 selectedMolecules.erase(id);
342 molecules.erase(id);
343 moleculeIdPool.releaseId(id);
344}
345
346atom *World::createAtom(){
347 OBSERVE;
348 atomId_t id = atomIdPool.getNextId();
349 ASSERT(!atoms.count(id),"proposed id did not specify an unused ID");
350 atom *res = NewAtom(id);
351 res->setWorld(this);
352 // store the atom by ID
353 atoms[res->getId()] = res;
354 _lastchangedatom = res;
355 NOTIFY(AtomInserted);
356 return res;
357}
358
359
360int World::registerAtom(atom *atom){
361 OBSERVE;
362 atomId_t id = atomIdPool.getNextId();
363 atom->setId(id);
364 atom->setWorld(this);
365 atoms[atom->getId()] = atom;
366 _lastchangedatom = atom;
367 NOTIFY(AtomInserted);
368 return atom->getId();
369}
370
371void World::destroyAtom(atom* atom){
372 int id = atom->getId();
373 destroyAtom(id);
374}
375
376void World::destroyAtom(atomId_t id) {
377 atom *atom = atoms[id];
378 ASSERT(atom,"Atom ID that was meant to be destroyed did not exist");
379 // give notice about immediate removal
380 {
381 OBSERVE;
382 _lastchangedatom = atom;
383 NOTIFY(AtomRemoved);
384 }
385 // check if it's the last atom
386 molecule *_mol = const_cast<molecule *>(atom->getMolecule());
387 if ((_mol == NULL) || (_mol->getAtomCount() > 1))
388 _mol = NULL;
389 DeleteAtom(atom);
390 if (isAtomSelected(id))
391 selectedAtoms.erase(id);
392 atoms.erase(id);
393 atomIdPool.releaseId(id);
394 // remove molecule if empty
395 if (_mol != NULL)
396 destroyMolecule(_mol);
397}
398
399bool World::changeAtomId(atomId_t oldId, atomId_t newId, atom* target){
400 OBSERVE;
401 // in case this call did not originate from inside the atom, we redirect it,
402 // to also let it know that it has changed
403 if(!target){
404 target = atoms[oldId];
405 ASSERT(target,"Atom with that ID not found");
406 return target->changeId(newId);
407 }
408 else{
409 if(atomIdPool.reserveId(newId)){
410 atoms.erase(oldId);
411 atoms.insert(pair<atomId_t,atom*>(newId,target));
412 return true;
413 }
414 else{
415 return false;
416 }
417 }
418}
419
420bool World::changeMoleculeId(moleculeId_t oldId, moleculeId_t newId, molecule* target){
421 OBSERVE;
422 // in case this call did not originate from inside the atom, we redirect it,
423 // to also let it know that it has changed
424 if(!target){
425 target = molecules[oldId];
426 ASSERT(target,"Molecule with that ID not found");
427 return target->changeId(newId);
428 }
429 else{
430 if(moleculeIdPool.reserveId(newId)){
431 molecules.erase(oldId);
432 molecules.insert(pair<moleculeId_t,molecule*>(newId,target));
433 return true;
434 }
435 else{
436 return false;
437 }
438 }
439}
440
441ManipulateAtomsProcess* World::manipulateAtoms(boost::function<void(atom*)> op,std::string name,AtomDescriptor descr){
442 ActionTrait manipulateTrait(name);
443 return new ManipulateAtomsProcess(op, descr,manipulateTrait);
444}
445
446ManipulateAtomsProcess* World::manipulateAtoms(boost::function<void(atom*)> op,std::string name){
447 return manipulateAtoms(op,name,AllAtoms());
448}
449
450/********************* Internal Change methods for double Callback and Observer mechanism ********/
451
452void World::doManipulate(ManipulateAtomsProcess *proc){
453 proc->signOn(this);
454 {
455 OBSERVE;
456 proc->doManipulate(this);
457 }
458 proc->signOff(this);
459}
460/******************************* Iterators ********************************/
461
462// external parts with observers
463
464CONSTRUCT_SELECTIVE_ITERATOR(atom*,World::AtomSet,AtomDescriptor)
465
466CONSTRUCT_SELECTIVE_CONST_ITERATOR(atom*,World::AtomSet,AtomDescriptor)
467
468World::AtomIterator
469World::getAtomIter(AtomDescriptor descr){
470 return AtomIterator(descr,atoms);
471}
472
473World::AtomConstIterator
474World::getAtomIter(AtomDescriptor descr) const{
475 return AtomConstIterator(descr,atoms);
476}
477
478World::AtomIterator
479World::getAtomIter(){
480 return AtomIterator(AllAtoms(),atoms);
481}
482
483World::AtomConstIterator
484World::getAtomIter() const{
485 return AtomConstIterator(AllAtoms(),atoms);
486}
487
488World::AtomIterator
489World::atomEnd(){
490 return AtomIterator(AllAtoms(),atoms,atoms.end());
491}
492
493World::AtomConstIterator
494World::atomEnd() const{
495 return AtomConstIterator(AllAtoms(),atoms,atoms.end());
496}
497
498CONSTRUCT_SELECTIVE_ITERATOR(molecule*,World::MoleculeSet,MoleculeDescriptor)
499
500CONSTRUCT_SELECTIVE_CONST_ITERATOR(molecule*,World::MoleculeSet,MoleculeDescriptor)
501
502World::MoleculeIterator
503World::getMoleculeIter(MoleculeDescriptor descr){
504 return MoleculeIterator(descr,molecules);
505}
506
507World::MoleculeConstIterator
508World::getMoleculeIter(MoleculeDescriptor descr) const{
509 return MoleculeConstIterator(descr,molecules);
510}
511
512World::MoleculeIterator
513World::getMoleculeIter(){
514 return MoleculeIterator(AllMolecules(),molecules);
515}
516
517World::MoleculeConstIterator
518World::getMoleculeIter() const{
519 return MoleculeConstIterator(AllMolecules(),molecules);
520}
521
522World::MoleculeIterator
523World::moleculeEnd(){
524 return MoleculeIterator(AllMolecules(),molecules,molecules.end());
525}
526
527World::MoleculeConstIterator
528World::moleculeEnd() const{
529 return MoleculeConstIterator(AllMolecules(),molecules,molecules.end());
530}
531
532// Internal parts, without observers
533
534// Build the AtomIterator from template
535CONSTRUCT_SELECTIVE_ITERATOR(atom*,World::AtomSet::set_t,AtomDescriptor);
536
537
538World::internal_AtomIterator
539World::getAtomIter_internal(AtomDescriptor descr){
540 return internal_AtomIterator(descr,atoms.getContent());
541}
542
543World::internal_AtomIterator
544World::atomEnd_internal(){
545 return internal_AtomIterator(AllAtoms(),atoms.getContent(),atoms.end_internal());
546}
547
548// build the MoleculeIterator from template
549CONSTRUCT_SELECTIVE_ITERATOR(molecule*,World::MoleculeSet::set_t,MoleculeDescriptor);
550
551World::internal_MoleculeIterator World::getMoleculeIter_internal(MoleculeDescriptor descr){
552 return internal_MoleculeIterator(descr,molecules.getContent());
553}
554
555World::internal_MoleculeIterator World::moleculeEnd_internal(){
556 return internal_MoleculeIterator(AllMolecules(),molecules.getContent(),molecules.end_internal());
557}
558
559/************************** Selection of Atoms and molecules ******************/
560
561// Atoms
562
563void World::clearAtomSelection(){
564 OBSERVE;
565 NOTIFY(SelectionChanged);
566 selectedAtoms.clear();
567}
568
569void World::invertAtomSelection(){
570 // get all atoms not selected
571 AtomComposite invertedSelection(getAllAtoms());
572 bool (World::*predicate)(const atom*) const = &World::isSelected; // needed for type resolution of overloaded function
573 AtomComposite::iterator iter =
574 std::remove_if(invertedSelection.begin(), invertedSelection.end(),
575 std::bind1st(std::mem_fun(predicate), this));
576 invertedSelection.erase(iter, invertedSelection.end());
577 // apply new selection
578 selectedAtoms.clear();
579 void (World::*selector)(const atom*) = &World::selectAtom; // needed for type resolution of overloaded function
580 std::for_each(invertedSelection.begin(),invertedSelection.end(),
581 std::bind1st(std::mem_fun(selector),this)); // func is select... see above
582}
583
584void World::popAtomSelection(){
585 OBSERVE;
586 NOTIFY(SelectionChanged);
587 const atomIdsVector_t atomids = selectedAtoms_Stack.top();
588 boost::function<void (const atomId_t)> IdSelector =
589 boost::bind(static_cast<void (World::*)(const atomId_t)>(&World::selectAtom), this, _1);
590 selectedAtoms.clear();
591 std::for_each(atomids.begin(),atomids.end(), IdSelector);
592 selectedAtoms_Stack.pop();
593}
594
595void World::pushAtomSelection(){
596 OBSERVE;
597 NOTIFY(SelectionChanged);
598 atomIdsVector_t atomids(countSelectedAtoms(), (atomId_t)-1);
599 std::copy(
600 MapKeyIterator<AtomSelectionConstIterator>(beginAtomSelection()),
601 MapKeyIterator<AtomSelectionConstIterator>(endAtomSelection()),
602 atomids.begin());
603 selectedAtoms_Stack.push( atomids );
604 selectedAtoms.clear();
605}
606
607void World::selectAtom(const atom *_atom){
608 OBSERVE;
609 NOTIFY(SelectionChanged);
610 // atom * is unchanged in this function, but we do store entity as changeable
611 ASSERT(_atom,"Invalid pointer in selection of atom");
612 selectedAtoms[_atom->getId()]=const_cast<atom *>(_atom);
613}
614
615void World::selectAtom(const atomId_t id){
616 OBSERVE;
617 NOTIFY(SelectionChanged);
618 ASSERT(atoms.count(id),"Atom Id selected that was not in the world");
619 selectedAtoms[id]=atoms[id];
620}
621
622void World::selectAllAtoms(AtomDescriptor descr){
623 OBSERVE;
624 NOTIFY(SelectionChanged);
625 internal_AtomIterator begin = getAtomIter_internal(descr);
626 internal_AtomIterator end = atomEnd_internal();
627 void (World::*func)(const atom*) = &World::selectAtom; // needed for type resolution of overloaded function
628 for_each(begin,end,bind1st(mem_fun(func),this)); // func is select... see above
629}
630
631void World::selectAtomsOfMolecule(const molecule *_mol){
632 OBSERVE;
633 NOTIFY(SelectionChanged);
634 ASSERT(_mol,"Invalid pointer to molecule in selection of Atoms of Molecule");
635 // need to make it const to get the fast iterators
636 const molecule *mol = _mol;
637 void (World::*func)(const atom*) = &World::selectAtom; // needed for type resolution of overloaded function
638 for_each(mol->begin(),mol->end(),bind1st(mem_fun(func),this)); // func is select... see above
639}
640
641void World::selectAtomsOfMolecule(const moleculeId_t id){
642 OBSERVE;
643 NOTIFY(SelectionChanged);
644 ASSERT(molecules.count(id),"No molecule with the given id upon Selection of atoms from molecule");
645 selectAtomsOfMolecule(molecules[id]);
646}
647
648void World::unselectAtom(const atom *_atom){
649 OBSERVE;
650 NOTIFY(SelectionChanged);
651 ASSERT(_atom,"Invalid pointer in unselection of atom");
652 unselectAtom(_atom->getId());
653}
654
655void World::unselectAtom(const atomId_t id){
656 OBSERVE;
657 NOTIFY(SelectionChanged);
658 ASSERT(atoms.count(id),"Atom Id unselected that was not in the world");
659 selectedAtoms.erase(id);
660}
661
662void World::unselectAllAtoms(AtomDescriptor descr){
663 OBSERVE;
664 NOTIFY(SelectionChanged);
665 internal_AtomIterator begin = getAtomIter_internal(descr);
666 internal_AtomIterator end = atomEnd_internal();
667 void (World::*func)(const atom*) = &World::unselectAtom; // needed for type resolution of overloaded function
668 for_each(begin,end,bind1st(mem_fun(func),this)); // func is unselect... see above
669}
670
671void World::unselectAtomsOfMolecule(const molecule *_mol){
672 OBSERVE;
673 NOTIFY(SelectionChanged);
674 ASSERT(_mol,"Invalid pointer to molecule in selection of Atoms of Molecule");
675 // need to make it const to get the fast iterators
676 const molecule *mol = _mol;
677 void (World::*func)(const atom*) = &World::unselectAtom; // needed for type resolution of overloaded function
678 for_each(mol->begin(),mol->end(),bind1st(mem_fun(func),this)); // func is unselect... see above
679}
680
681void World::unselectAtomsOfMolecule(const moleculeId_t id){
682 OBSERVE;
683 NOTIFY(SelectionChanged);
684 ASSERT(molecules.count(id),"No molecule with the given id upon Selection of atoms from molecule");
685 unselectAtomsOfMolecule(molecules[id]);
686}
687
688size_t World::countSelectedAtoms() const {
689 size_t count = 0;
690 for (AtomSet::const_iterator iter = selectedAtoms.begin(); iter != selectedAtoms.end(); ++iter)
691 count++;
692 return count;
693}
694
695bool World::isSelected(const atom *_atom) const {
696 return isAtomSelected(_atom->getId());
697}
698
699bool World::isAtomSelected(const atomId_t no) const {
700 return selectedAtoms.find(no) != selectedAtoms.end();
701}
702
703const std::vector<atom *> World::getSelectedAtoms() const {
704 std::vector<atom *> returnAtoms;
705 returnAtoms.resize(countSelectedAtoms());
706 int count = 0;
707 for (AtomSet::const_iterator iter = selectedAtoms.begin(); iter != selectedAtoms.end(); ++iter)
708 returnAtoms[count++] = iter->second;
709 return returnAtoms;
710}
711
712
713// Molecules
714
715void World::clearMoleculeSelection(){
716 OBSERVE;
717 NOTIFY(SelectionChanged);
718 selectedMolecules.clear();
719}
720
721void World::invertMoleculeSelection(){
722 // get all molecules not selected
723 typedef std::vector<molecule *> MoleculeVector_t;
724 MoleculeVector_t invertedSelection(getAllMolecules());
725 bool (World::*predicate)(const molecule*) const = &World::isSelected; // needed for type resolution of overloaded function
726 MoleculeVector_t::iterator iter =
727 std::remove_if(invertedSelection.begin(), invertedSelection.end(),
728 std::bind1st(std::mem_fun(predicate), this));
729 invertedSelection.erase(iter, invertedSelection.end());
730 // apply new selection
731 selectedMolecules.clear();
732 void (World::*selector)(const molecule*) = &World::selectMolecule; // needed for type resolution of overloaded function
733 std::for_each(invertedSelection.begin(),invertedSelection.end(),
734 std::bind1st(std::mem_fun(selector),this)); // func is select... see above
735}
736
737void World::popMoleculeSelection(){
738 OBSERVE;
739 NOTIFY(SelectionChanged);
740 const moleculeIdsVector_t moleculeids = selectedMolecules_Stack.top();
741 boost::function<void (const moleculeId_t)> IdSelector =
742 boost::bind(static_cast<void (World::*)(const moleculeId_t)>(&World::selectMolecule), this, _1);
743 selectedMolecules.clear();
744 std::for_each(moleculeids.begin(),moleculeids.end(), IdSelector);
745 selectedMolecules_Stack.pop();
746}
747
748void World::pushMoleculeSelection(){
749 OBSERVE;
750 NOTIFY(SelectionChanged);
751 moleculeIdsVector_t moleculeids(countSelectedMolecules(), (moleculeId_t)-1);
752 boost::function<moleculeId_t (const molecule*)> IdRetriever =
753 boost::bind(&molecule::getId, _1);
754 std::copy(
755 MapKeyIterator<MoleculeSelectionConstIterator>(beginMoleculeSelection()),
756 MapKeyIterator<MoleculeSelectionConstIterator>(endMoleculeSelection()),
757 moleculeids.begin());
758 selectedMolecules_Stack.push( moleculeids );
759 selectedMolecules.clear();
760}
761
762void World::selectMolecule(const molecule *_mol){
763 OBSERVE;
764 NOTIFY(SelectionChanged);
765 // molecule * is unchanged in this function, but we do store entity as changeable
766 ASSERT(_mol,"Invalid pointer to molecule in selection");
767 selectedMolecules[_mol->getId()]=const_cast<molecule *>(_mol);
768}
769
770void World::selectMolecule(const moleculeId_t id){
771 OBSERVE;
772 NOTIFY(SelectionChanged);
773 ASSERT(molecules.count(id),"Molecule Id selected that was not in the world");
774 selectedMolecules[id]=molecules[id];
775}
776
777void World::selectAllMolecules(MoleculeDescriptor descr){
778 OBSERVE;
779 NOTIFY(SelectionChanged);
780 internal_MoleculeIterator begin = getMoleculeIter_internal(descr);
781 internal_MoleculeIterator end = moleculeEnd_internal();
782 void (World::*func)(const molecule*) = &World::selectMolecule; // needed for type resolution of overloaded function
783 for_each(begin,end,bind1st(mem_fun(func),this)); // func is select... see above
784}
785
786void World::selectMoleculeOfAtom(const atom *_atom){
787 OBSERVE;
788 NOTIFY(SelectionChanged);
789 ASSERT(_atom,"Invalid atom pointer in selection of MoleculeOfAtom");
790 const molecule *mol=_atom->getMolecule();
791 // the atom might not be part of a molecule
792 if(mol){
793 selectMolecule(mol);
794 }
795}
796
797void World::selectMoleculeOfAtom(const atomId_t id){
798 OBSERVE;
799 NOTIFY(SelectionChanged);
800 ASSERT(atoms.count(id),"No such atom with given ID in selection of Molecules of Atom");\
801 selectMoleculeOfAtom(atoms[id]);
802}
803
804void World::unselectMolecule(const molecule *_mol){
805 OBSERVE;
806 NOTIFY(SelectionChanged);
807 ASSERT(_mol,"invalid pointer in unselection of molecule");
808 unselectMolecule(_mol->getId());
809}
810
811void World::unselectMolecule(const moleculeId_t id){
812 OBSERVE;
813 NOTIFY(SelectionChanged);
814 ASSERT(molecules.count(id),"No such molecule with ID in unselection");
815 selectedMolecules.erase(id);
816}
817
818void World::unselectAllMolecules(MoleculeDescriptor descr){
819 OBSERVE;
820 NOTIFY(SelectionChanged);
821 internal_MoleculeIterator begin = getMoleculeIter_internal(descr);
822 internal_MoleculeIterator end = moleculeEnd_internal();
823 void (World::*func)(const molecule*) = &World::unselectMolecule; // needed for type resolution of overloaded function
824 for_each(begin,end,bind1st(mem_fun(func),this)); // func is unselect... see above
825}
826
827void World::unselectMoleculeOfAtom(const atom *_atom){
828 OBSERVE;
829 NOTIFY(SelectionChanged);
830 ASSERT(_atom,"Invalid atom pointer in selection of MoleculeOfAtom");
831 const molecule *mol=_atom->getMolecule();
832 // the atom might not be part of a molecule
833 if(mol){
834 unselectMolecule(mol);
835 }
836}
837
838void World::unselectMoleculeOfAtom(const atomId_t id){
839 OBSERVE;
840 NOTIFY(SelectionChanged);
841 ASSERT(atoms.count(id),"No such atom with given ID in selection of Molecules of Atom");\
842 unselectMoleculeOfAtom(atoms[id]);
843}
844
845size_t World::countSelectedMolecules() const {
846 size_t count = 0;
847 for (MoleculeSet::const_iterator iter = selectedMolecules.begin(); iter != selectedMolecules.end(); ++iter)
848 count++;
849 return count;
850}
851
852bool World::isSelected(const molecule *_mol) const {
853 return isMoleculeSelected(_mol->getId());
854}
855
856bool World::isMoleculeSelected(const moleculeId_t no) const {
857 return selectedMolecules.find(no) != selectedMolecules.end();
858}
859
860std::vector<molecule *> World::getSelectedMolecules() {
861 std::vector<molecule *> returnMolecules;
862 std::transform(
863 selectedMolecules.begin(),
864 selectedMolecules.end(),
865 back_inserter(returnMolecules),
866 _take<molecule*,World::MoleculeSet::value_type>::get);
867 return returnMolecules;
868}
869
870std::vector<const molecule *> World::getSelectedMolecules() const {
871 std::vector<const molecule *> returnMolecules;
872 std::transform(
873 selectedMolecules.begin(),
874 selectedMolecules.end(),
875 back_inserter(returnMolecules),
876 _take<molecule*,World::MoleculeSet::value_type>::get);
877 return returnMolecules;
878}
879
880/******************* Iterators over Selection *****************************/
881World::AtomSelectionIterator World::beginAtomSelection(){
882 return selectedAtoms.begin();
883}
884
885World::AtomSelectionIterator World::endAtomSelection(){
886 return selectedAtoms.end();
887}
888
889World::AtomSelectionConstIterator World::beginAtomSelection() const{
890 return selectedAtoms.begin();
891}
892
893World::AtomSelectionConstIterator World::endAtomSelection() const{
894 return selectedAtoms.end();
895}
896
897
898World::MoleculeSelectionIterator World::beginMoleculeSelection(){
899 return selectedMolecules.begin();
900}
901
902World::MoleculeSelectionIterator World::endMoleculeSelection(){
903 return selectedMolecules.end();
904}
905
906World::MoleculeSelectionConstIterator World::beginMoleculeSelection() const{
907 return selectedMolecules.begin();
908}
909
910World::MoleculeSelectionConstIterator World::endMoleculeSelection() const{
911 return selectedMolecules.end();
912}
913
914/******************************* Singleton Stuff **************************/
915
916World::World() :
917 Observable("World"),
918 BG(new BondGraph(true)), // assume Angstroem for the moment
919 periode(new periodentafel(true)),
920 configuration(new config),
921 homologies(new HomologyContainer()),
922 Thermostats(new ThermoStatContainer),
923 ExitFlag(0),
924 atoms(this),
925 selectedAtoms(this),
926 atomIdPool(0, 20, 100),
927 molecules(this),
928 selectedMolecules(this),
929 moleculeIdPool(0, 20,100),
930 molecules_deprecated(new MoleculeListClass(this))
931{
932 cell_size = new Box;
933 RealSpaceMatrix domain;
934 domain.at(0,0) = 20;
935 domain.at(1,1) = 20;
936 domain.at(2,2) = 20;
937 cell_size->setM(domain);
938 LCcontroller = new LinkedCell::LinkedCell_Controller(*cell_size);
939 defaultName = "none";
940 Channels *OurChannel = new Channels;
941 NotificationChannels.insert( std::make_pair( static_cast<Observable *>(this), OurChannel) );
942 for (size_t type = 0; type < (size_t)NotificationType_MAX; ++type)
943 OurChannel->addChannel(type);
944}
945
946World::~World()
947{
948 delete LCcontroller;
949 delete cell_size;
950 delete molecules_deprecated;
951 MoleculeSet::iterator molIter;
952 for(molIter=molecules.begin();molIter!=molecules.end();++molIter){
953 DeleteMolecule((*molIter).second);
954 }
955 molecules.clear();
956 AtomSet::iterator atIter;
957 for(atIter=atoms.begin();atIter!=atoms.end();++atIter){
958 DeleteAtom((*atIter).second);
959 }
960 atoms.clear();
961
962 delete BG;
963 delete periode;
964 delete configuration;
965 delete Thermostats;
966 delete homologies;
967}
968
969// Explicit instantiation of the singleton mechanism at this point
970
971// moleculeId_t und atomId_t sind gleicher Basistyp, deswegen nur einen von beiden konstruieren
972CONSTRUCT_IDPOOL(atomId_t, uniqueId)
973CONSTRUCT_IDPOOL(moleculeId_t, continuousId)
974
975CONSTRUCT_SINGLETON(World)
976
977CONSTRUCT_OBSERVEDCONTAINER(World::AtomSTLSet)
978
979CONSTRUCT_OBSERVEDCONTAINER(World::MoleculeSTLSet)
980
981/******************************* deprecated Legacy Stuff ***********************/
982
983MoleculeListClass *&World::getMolecules() {
984 return molecules_deprecated;
985}
Note: See TracBrowser for help on using the repository browser.