source: src/Actions/FillAction/FillVolumeAction.cpp@ a844d8

Candidate_v1.6.1
Last change on this file since a844d8 was 9eb71b3, checked in by Frederik Heber <frederik.heber@…>, 8 years ago

Commented out MemDebug include and Memory::ignore.

  • MemDebug clashes with various allocation operators that use a specific placement in memory. It is so far not possible to wrap new/delete fully. Hence, we stop this effort which so far has forced us to put ever more includes (with clashes) into MemDebug and thereby bloat compilation time.
  • MemDebug does not add that much usefulness which is not also provided by valgrind.
  • Property mode set to 100644
File size: 7.5 KB
Line 
1/*
2 * Project: MoleCuilder
3 * Description: creates and alters molecular systems
4 * Copyright (C) 2014 Frederik Heber. All rights reserved.
5 *
6 *
7 * This file is part of MoleCuilder.
8 *
9 * MoleCuilder is free software: you can redistribute it and/or modify
10 * it under the terms of the GNU General Public License as published by
11 * the Free Software Foundation, either version 2 of the License, or
12 * (at your option) any later version.
13 *
14 * MoleCuilder is distributed in the hope that it will be useful,
15 * but WITHOUT ANY WARRANTY; without even the implied warranty of
16 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
17 * GNU General Public License for more details.
18 *
19 * You should have received a copy of the GNU General Public License
20 * along with MoleCuilder. If not, see <http://www.gnu.org/licenses/>.
21 */
22
23/*
24 * FillVolumeAction.cpp
25 *
26 * Created on: Sep 03, 2014
27 * Author: heber
28 */
29
30// include config.h
31#ifdef HAVE_CONFIG_H
32#include <config.h>
33#endif
34
35//#include "CodePatterns/MemDebug.hpp"
36
37#include "Actions/UndoRedoHelpers.hpp"
38#include "Atom/atom.hpp"
39#include "Atom/AtomicInfo.hpp"
40#include "Atom/CopyAtoms/CopyAtoms_withBonds.hpp"
41#include "Bond/BondInfo.hpp"
42#include "CodePatterns/Log.hpp"
43#include "Descriptors/MoleculeOrderDescriptor.hpp"
44#include "Filling/Cluster.hpp"
45#include "Filling/Filler.hpp"
46#include "Filling/Preparators/ShapeVolumeFillerPreparator.hpp"
47#include "molecule.hpp"
48#include "Parser/FormatParserInterface.hpp"
49#include "Parser/FormatParserStorage.hpp"
50#include "Shapes/Shape.hpp"
51#include "Shapes/ShapeRegistry.hpp"
52#include "Shapes/ShapeType.hpp"
53#include "World.hpp"
54
55#include <algorithm>
56#include <iostream>
57#include <string>
58#include <vector>
59
60#include "Actions/FillAction/FillVolumeAction.hpp"
61
62using namespace MoleCuilder;
63
64// and construct the stuff
65#include "FillVolumeAction.def"
66#include "Action_impl_pre.hpp"
67/** =========== define the function ====================== */
68ActionState::ptr FillVolumeAction::performCall() {
69 // get the filler molecule
70 std::vector<AtomicInfo> movedatoms;
71 const std::vector< molecule *> molecules = World::getInstance().getSelectedMolecules();
72 if (molecules.size() != 1) {
73 STATUS("No exactly one molecule selected, aborting,");
74 return Action::failure;
75 }
76 molecule *filler = *(molecules.begin());
77 for(molecule::const_iterator iter = const_cast<const molecule *>(filler)->begin();
78 iter != const_cast<const molecule *>(filler)->end();
79 ++iter)
80 movedatoms.push_back( AtomicInfo(*(*iter)) );
81 LOG(1, "INFO: Chosen molecule has " << filler->size() << " atoms.");
82
83 // center filler's tip at origin
84 filler->CenterEdge();
85
86 // prepare the filler preparator
87 if (ShapeRegistry::getInstance().countSelectedShapes() != (size_t)1) {
88 STATUS("Not exactly one shape selected.");
89 return Action::failure;
90 }
91 const std::vector<Shape*> shapes = ShapeRegistry::getInstance().getSelectedShapes();
92 const Shape &shape = **shapes.begin();
93
94 // hard check whether shape is of allowed type, not all are implemented
95 // but these only fail with an assertion, hence not with disable-debug
96 switch (shape.getType()) {
97 case NowhereType:
98 case EverywhereType:
99 case SphereType:
100 case CuboidType:
101 STATUS("The shape type "+toString(shape.getType())+" is currently not supported.");
102 return Action::failure;
103 break;
104 default:
105 break;
106 }
107
108 ShapeVolumeFillerPreparator filler_preparator(filler);
109 if (params.SphereRadius.get() != 0.) {
110 if (World::getInstance().beginAtomSelection() == World::getInstance().endAtomSelection()) {
111 STATUS("You have given a sphere radius "+toString(params.SphereRadius.get())
112 +" != 0, but have not select any atoms.");
113 return Action::failure;
114 }
115 std::vector<atom*> atoms(World::getInstance().getSelectedAtoms());
116 filler_preparator.addSurfacePredicate(
117 params.SphereRadius.get(),
118 atoms,
119 params.mindistance.get());
120 }
121 filler_preparator.addVoidPredicate(params.mindistance.get());
122 filler_preparator.addRandomInserter(
123 params.RandAtomDisplacement.get(),
124 params.RandMoleculeDisplacement.get(),
125 params.DoRotate.get());
126 filler_preparator.addShapeMesh(
127 shape,
128 params.N.get());
129 if (!filler_preparator()) {
130 STATUS("Filler was not fully constructed.");
131 return Action::failure;
132 }
133
134 // use filler
135 bool successflag = false;
136 FillVolumeState *UndoState = NULL;
137 {
138 // fill
139 Filler *fillerFunction = filler_preparator.obtainFiller();
140 // TODO: When molecule::getBoundingSphere() does not use a sphere anymore,
141 // we need to check whether we rotate the molecule randomly. For this to
142 // work we need a sphere!
143 const Shape s = filler->getBoundingSphere(params.RandAtomDisplacement.get());
144 ClusterInterface::Cluster_impl cluster( new Cluster(filler->getAtomIds(), s) );
145 CopyAtoms_withBonds copyMethod;
146 Filler::ClusterVector_t ClonedClusters;
147 successflag = (*fillerFunction)(copyMethod, cluster, ClonedClusters);
148 delete fillerFunction;
149
150 // append each cluster's atoms to clonedatoms (however not selected ones)
151 std::vector<const atom *> clonedatoms;
152 std::vector<AtomicInfo> clonedatominfos;
153 for (Filler::ClusterVector_t::const_iterator iter = ClonedClusters.begin();
154 iter != ClonedClusters.end(); ++iter) {
155 const AtomIdSet &atoms = (*iter)->getAtomIds();
156 clonedatoms.reserve(clonedatoms.size()+atoms.size());
157 for (AtomIdSet::const_iterator atomiter = atoms.begin(); atomiter != atoms.end(); ++atomiter)
158 if (!filler->containsAtom(*atomiter)) {
159 clonedatoms.push_back( *atomiter );
160 clonedatominfos.push_back( AtomicInfo(*(*atomiter)) );
161 }
162 }
163 std::vector< BondInfo > clonedbonds;
164 StoreBondInformationFromAtoms(clonedatoms, clonedbonds);
165 LOG(2, "DEBUG: There are " << clonedatominfos.size() << " newly created atoms.");
166
167 if (!successflag) {
168 STATUS("Insertion failed, removing inserted clusters, translating original one back");
169 RemoveAtomsFromAtomicInfo(clonedatominfos);
170 clonedatoms.clear();
171 SetAtomsFromAtomicInfo(movedatoms);
172 } else {
173 std::vector<Vector> MovedToVector(filler->size(), zeroVec);
174 std::transform(filler->begin(), filler->end(), MovedToVector.begin(),
175 boost::bind(&AtomInfo::getPosition, _1) );
176 UndoState = new FillVolumeState(clonedatominfos,clonedbonds,movedatoms,MovedToVector,params);
177 }
178 }
179
180 if (successflag)
181 return ActionState::ptr(UndoState);
182 else {
183 return Action::failure;
184 }
185}
186
187ActionState::ptr FillVolumeAction::performUndo(ActionState::ptr _state) {
188 FillVolumeState *state = assert_cast<FillVolumeState*>(_state.get());
189
190 // remove all created atoms
191 RemoveAtomsFromAtomicInfo(state->clonedatoms);
192 // add the original cluster
193 SetAtomsFromAtomicInfo(state->movedatoms);
194
195 return ActionState::ptr(_state);
196}
197
198ActionState::ptr FillVolumeAction::performRedo(ActionState::ptr _state){
199 FillVolumeState *state = assert_cast<FillVolumeState*>(_state.get());
200
201 // place filler cluster again at new spot
202 ResetAtomPosition(state->movedatoms, state->MovedToVector);
203
204 // re-create all clusters
205 bool statusflag = AddAtomsFromAtomicInfo(state->clonedatoms);
206
207 // re-create the bonds
208 if (statusflag)
209 AddBondsFromBondInfo(state->clonedbonds);
210 if (statusflag)
211 return ActionState::ptr(_state);
212 else {
213 STATUS("Failed re-adding filled in atoms.");
214 return Action::failure;
215 }
216}
217
218bool FillVolumeAction::canUndo() {
219 return true;
220}
221
222bool FillVolumeAction::shouldUndo() {
223 return true;
224}
225/** =========== end of function ====================== */
Note: See TracBrowser for help on using the repository browser.