source: src/menu.cpp@ a306a2

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

Small fix to delete the complete menu.

Signed-off-by: Tillmann Crueger <crueger@…>

  • Property mode set to 100644
File size: 55.5 KB
Line 
1/** \file menu.cpp
2 * The class in this file is responsible for displaying the menu and enabling choices.
3 *
4 * This class is currently being refactored. Functions were copied from builder.cpp and are
5 * to be imported into the menu class.
6 *
7 */
8
9#include "menu.hpp"
10#include "analysis_correlation.hpp"
11#include "atom.hpp"
12#include "bond.hpp"
13#include "bondgraph.hpp"
14#include "boundary.hpp"
15#include "config.hpp"
16#include "element.hpp"
17#include "ellipsoid.hpp"
18#include "helpers.hpp"
19#include "leastsquaremin.hpp"
20#include "linkedcell.hpp"
21#include "log.hpp"
22#include "memoryusageobserverunittest.hpp"
23#include "molecule.hpp"
24#include "periodentafel.hpp"
25
26#include "Menu/Menu.hpp"
27#include "Menu/TextMenu.hpp"
28#include "Menu/ActionMenuItem.hpp"
29#include "Menu/SeperatorItem.hpp"
30#include "Menu/DisplayMenuItem.hpp"
31#include "Actions/MethodAction.hpp"
32#include "Views/StreamStringView.hpp"
33#include "Views/MethodStringView.hpp"
34
35
36#include <boost/bind.hpp>
37
38/* copied methods for refactoring */
39/*TODO: Move these methods inside menu class
40 * and restructure menu class*/
41
42/********************************************* Subsubmenu routine ************************************/
43
44/** Submenu for adding atoms to the molecule.
45 * \param *periode periodentafel
46 * \param *molecule molecules with atoms
47 */
48void oldmenu::AddAtoms(periodentafel *periode, molecule *mol)
49{
50 atom *first, *second, *third, *fourth;
51 Vector **atoms;
52 Vector x,y,z,n; // coordinates for absolute point in cell volume
53 double a,b,c;
54 char choice; // menu choice char
55 bool valid;
56
57 Log() << Verbose(0) << "===========ADD ATOM============================" << endl;
58 Log() << Verbose(0) << " a - state absolute coordinates of atom" << endl;
59 Log() << Verbose(0) << " b - state relative coordinates of atom wrt to reference point" << endl;
60 Log() << Verbose(0) << " c - state relative coordinates of atom wrt to already placed atom" << endl;
61 Log() << Verbose(0) << " d - state two atoms, two angles and a distance" << endl;
62 Log() << Verbose(0) << " e - least square distance position to a set of atoms" << endl;
63 Log() << Verbose(0) << "all else - go back" << endl;
64 Log() << Verbose(0) << "===============================================" << endl;
65 Log() << Verbose(0) << "Note: Specifiy angles in degrees not multiples of Pi!" << endl;
66 Log() << Verbose(0) << "INPUT: ";
67 cin >> choice;
68
69 switch (choice) {
70 default:
71 eLog() << Verbose(2) << "Not a valid choice." << endl;
72 break;
73 case 'a': // absolute coordinates of atom
74 Log() << Verbose(0) << "Enter absolute coordinates." << endl;
75 first = new atom;
76 first->x.AskPosition(mol->cell_size, false);
77 first->type = periode->AskElement(); // give type
78 mol->AddAtom(first); // add to molecule
79 break;
80
81 case 'b': // relative coordinates of atom wrt to reference point
82 first = new atom;
83 valid = true;
84 do {
85 if (!valid) eLog() << Verbose(2) << "Resulting position out of cell." << endl;
86 Log() << Verbose(0) << "Enter reference coordinates." << endl;
87 x.AskPosition(mol->cell_size, true);
88 Log() << Verbose(0) << "Enter relative coordinates." << endl;
89 first->x.AskPosition(mol->cell_size, false);
90 first->x.AddVector((const Vector *)&x);
91 Log() << Verbose(0) << "\n";
92 } while (!(valid = mol->CheckBounds((const Vector *)&first->x)));
93 first->type = periode->AskElement(); // give type
94 mol->AddAtom(first); // add to molecule
95 break;
96
97 case 'c': // relative coordinates of atom wrt to already placed atom
98 first = new atom;
99 valid = true;
100 do {
101 if (!valid) eLog() << Verbose(2) << "Resulting position out of cell." << endl;
102 second = mol->AskAtom("Enter atom number: ");
103 Log() << Verbose(0) << "Enter relative coordinates." << endl;
104 first->x.AskPosition(mol->cell_size, false);
105 for (int i=NDIM;i--;) {
106 first->x.x[i] += second->x.x[i];
107 }
108 } while (!(valid = mol->CheckBounds((const Vector *)&first->x)));
109 first->type = periode->AskElement(); // give type
110 mol->AddAtom(first); // add to molecule
111 break;
112
113 case 'd': // two atoms, two angles and a distance
114 first = new atom;
115 valid = true;
116 do {
117 if (!valid) {
118 eLog() << Verbose(2) << "Resulting coordinates out of cell - " << first->x << endl;
119 }
120 Log() << Verbose(0) << "First, we need two atoms, the first atom is the central, while the second is the outer one." << endl;
121 second = mol->AskAtom("Enter central atom: ");
122 third = mol->AskAtom("Enter second atom (specifying the axis for first angle): ");
123 fourth = mol->AskAtom("Enter third atom (specifying a plane for second angle): ");
124 a = ask_value("Enter distance between central (first) and new atom: ");
125 b = ask_value("Enter angle between new, first and second atom (degrees): ");
126 b *= M_PI/180.;
127 bound(&b, 0., 2.*M_PI);
128 c = ask_value("Enter second angle between new and normal vector of plane defined by first, second and third atom (degrees): ");
129 c *= M_PI/180.;
130 bound(&c, -M_PI, M_PI);
131 Log() << Verbose(0) << "radius: " << a << "\t phi: " << b*180./M_PI << "\t theta: " << c*180./M_PI << endl;
132/*
133 second->Output(1,1,(ofstream *)&cout);
134 third->Output(1,2,(ofstream *)&cout);
135 fourth->Output(1,3,(ofstream *)&cout);
136 n.MakeNormalvector((const vector *)&second->x, (const vector *)&third->x, (const vector *)&fourth->x);
137 x.Copyvector(&second->x);
138 x.SubtractVector(&third->x);
139 x.Copyvector(&fourth->x);
140 x.SubtractVector(&third->x);
141
142 if (!z.SolveSystem(&x,&y,&n, b, c, a)) {
143 Log() << Verbose(0) << "Failure solving self-dependent linear system!" << endl;
144 continue;
145 }
146 Log() << Verbose(0) << "resulting relative coordinates: ";
147 z.Output();
148 Log() << Verbose(0) << endl;
149 */
150 // calc axis vector
151 x.CopyVector(&second->x);
152 x.SubtractVector(&third->x);
153 x.Normalize();
154 Log() << Verbose(0) << "x: ",
155 x.Output();
156 Log() << Verbose(0) << endl;
157 z.MakeNormalVector(&second->x,&third->x,&fourth->x);
158 Log() << Verbose(0) << "z: ",
159 z.Output();
160 Log() << Verbose(0) << endl;
161 y.MakeNormalVector(&x,&z);
162 Log() << Verbose(0) << "y: ",
163 y.Output();
164 Log() << Verbose(0) << endl;
165
166 // rotate vector around first angle
167 first->x.CopyVector(&x);
168 first->x.RotateVector(&z,b - M_PI);
169 Log() << Verbose(0) << "Rotated vector: ",
170 first->x.Output();
171 Log() << Verbose(0) << endl;
172 // remove the projection onto the rotation plane of the second angle
173 n.CopyVector(&y);
174 n.Scale(first->x.ScalarProduct(&y));
175 Log() << Verbose(0) << "N1: ",
176 n.Output();
177 Log() << Verbose(0) << endl;
178 first->x.SubtractVector(&n);
179 Log() << Verbose(0) << "Subtracted vector: ",
180 first->x.Output();
181 Log() << Verbose(0) << endl;
182 n.CopyVector(&z);
183 n.Scale(first->x.ScalarProduct(&z));
184 Log() << Verbose(0) << "N2: ",
185 n.Output();
186 Log() << Verbose(0) << endl;
187 first->x.SubtractVector(&n);
188 Log() << Verbose(0) << "2nd subtracted vector: ",
189 first->x.Output();
190 Log() << Verbose(0) << endl;
191
192 // rotate another vector around second angle
193 n.CopyVector(&y);
194 n.RotateVector(&x,c - M_PI);
195 Log() << Verbose(0) << "2nd Rotated vector: ",
196 n.Output();
197 Log() << Verbose(0) << endl;
198
199 // add the two linear independent vectors
200 first->x.AddVector(&n);
201 first->x.Normalize();
202 first->x.Scale(a);
203 first->x.AddVector(&second->x);
204
205 Log() << Verbose(0) << "resulting coordinates: ";
206 first->x.Output();
207 Log() << Verbose(0) << endl;
208 } while (!(valid = mol->CheckBounds((const Vector *)&first->x)));
209 first->type = periode->AskElement(); // give type
210 mol->AddAtom(first); // add to molecule
211 break;
212
213 case 'e': // least square distance position to a set of atoms
214 first = new atom;
215 atoms = new (Vector*[128]);
216 valid = true;
217 for(int i=128;i--;)
218 atoms[i] = NULL;
219 int i=0, j=0;
220 Log() << Verbose(0) << "Now we need at least three molecules.\n";
221 do {
222 Log() << Verbose(0) << "Enter " << i+1 << "th atom: ";
223 cin >> j;
224 if (j != -1) {
225 second = mol->FindAtom(j);
226 atoms[i++] = &(second->x);
227 }
228 } while ((j != -1) && (i<128));
229 if (i >= 2) {
230 first->x.LSQdistance((const Vector **)atoms, i);
231
232 first->x.Output();
233 first->type = periode->AskElement(); // give type
234 mol->AddAtom(first); // add to molecule
235 } else {
236 delete first;
237 Log() << Verbose(0) << "Please enter at least two vectors!\n";
238 }
239 break;
240 };
241};
242
243/** Submenu for centering the atoms in the molecule.
244 * \param *mol molecule with all the atoms
245 */
246void oldmenu::CenterAtoms(molecule *mol)
247{
248 Vector x, y, helper;
249 char choice; // menu choice char
250
251 Log() << Verbose(0) << "===========CENTER ATOMS=========================" << endl;
252 Log() << Verbose(0) << " a - on origin" << endl;
253 Log() << Verbose(0) << " b - on center of gravity" << endl;
254 Log() << Verbose(0) << " c - within box with additional boundary" << endl;
255 Log() << Verbose(0) << " d - within given simulation box" << endl;
256 Log() << Verbose(0) << "all else - go back" << endl;
257 Log() << Verbose(0) << "===============================================" << endl;
258 Log() << Verbose(0) << "INPUT: ";
259 cin >> choice;
260
261 switch (choice) {
262 default:
263 Log() << Verbose(0) << "Not a valid choice." << endl;
264 break;
265 case 'a':
266 Log() << Verbose(0) << "Centering atoms in config file on origin." << endl;
267 mol->CenterOrigin();
268 break;
269 case 'b':
270 Log() << Verbose(0) << "Centering atoms in config file on center of gravity." << endl;
271 mol->CenterPeriodic();
272 break;
273 case 'c':
274 Log() << Verbose(0) << "Centering atoms in config file within given additional boundary." << endl;
275 for (int i=0;i<NDIM;i++) {
276 Log() << Verbose(0) << "Enter axis " << i << " boundary: ";
277 cin >> y.x[i];
278 }
279 mol->CenterEdge(&x); // make every coordinate positive
280 mol->Center.AddVector(&y); // translate by boundary
281 helper.CopyVector(&y);
282 helper.Scale(2.);
283 helper.AddVector(&x);
284 mol->SetBoxDimension(&helper); // update Box of atoms by boundary
285 break;
286 case 'd':
287 Log() << Verbose(1) << "Centering atoms in config file within given simulation box." << endl;
288 for (int i=0;i<NDIM;i++) {
289 Log() << Verbose(0) << "Enter axis " << i << " boundary: ";
290 cin >> x.x[i];
291 }
292 // update Box of atoms by boundary
293 mol->SetBoxDimension(&x);
294 // center
295 mol->CenterInBox();
296 break;
297 }
298};
299
300/** Submenu for aligning the atoms in the molecule.
301 * \param *periode periodentafel
302 * \param *mol molecule with all the atoms
303 */
304void oldmenu::AlignAtoms(periodentafel *periode, molecule *mol)
305{
306 atom *first, *second, *third;
307 Vector x,n;
308 char choice; // menu choice char
309
310 Log() << Verbose(0) << "===========ALIGN ATOMS=========================" << endl;
311 Log() << Verbose(0) << " a - state three atoms defining align plane" << endl;
312 Log() << Verbose(0) << " b - state alignment vector" << endl;
313 Log() << Verbose(0) << " c - state two atoms in alignment direction" << endl;
314 Log() << Verbose(0) << " d - align automatically by least square fit" << endl;
315 Log() << Verbose(0) << "all else - go back" << endl;
316 Log() << Verbose(0) << "===============================================" << endl;
317 Log() << Verbose(0) << "INPUT: ";
318 cin >> choice;
319
320 switch (choice) {
321 default:
322 case 'a': // three atoms defining mirror plane
323 first = mol->AskAtom("Enter first atom: ");
324 second = mol->AskAtom("Enter second atom: ");
325 third = mol->AskAtom("Enter third atom: ");
326
327 n.MakeNormalVector((const Vector *)&first->x,(const Vector *)&second->x,(const Vector *)&third->x);
328 break;
329 case 'b': // normal vector of mirror plane
330 Log() << Verbose(0) << "Enter normal vector of mirror plane." << endl;
331 n.AskPosition(mol->cell_size,0);
332 n.Normalize();
333 break;
334 case 'c': // three atoms defining mirror plane
335 first = mol->AskAtom("Enter first atom: ");
336 second = mol->AskAtom("Enter second atom: ");
337
338 n.CopyVector((const Vector *)&first->x);
339 n.SubtractVector((const Vector *)&second->x);
340 n.Normalize();
341 break;
342 case 'd':
343 char shorthand[4];
344 Vector a;
345 struct lsq_params param;
346 do {
347 fprintf(stdout, "Enter the element of atoms to be chosen: ");
348 fscanf(stdin, "%3s", shorthand);
349 } while ((param.type = periode->FindElement(shorthand)) == NULL);
350 Log() << Verbose(0) << "Element is " << param.type->name << endl;
351 mol->GetAlignvector(&param);
352 for (int i=NDIM;i--;) {
353 x.x[i] = gsl_vector_get(param.x,i);
354 n.x[i] = gsl_vector_get(param.x,i+NDIM);
355 }
356 gsl_vector_free(param.x);
357 Log() << Verbose(0) << "Offset vector: ";
358 x.Output();
359 Log() << Verbose(0) << endl;
360 n.Normalize();
361 break;
362 };
363 Log() << Verbose(0) << "Alignment vector: ";
364 n.Output();
365 Log() << Verbose(0) << endl;
366 mol->Align(&n);
367};
368
369/** Submenu for mirroring the atoms in the molecule.
370 * \param *mol molecule with all the atoms
371 */
372void oldmenu::MirrorAtoms(molecule *mol)
373{
374 atom *first, *second, *third;
375 Vector n;
376 char choice; // menu choice char
377
378 Log() << Verbose(0) << "===========MIRROR ATOMS=========================" << endl;
379 Log() << Verbose(0) << " a - state three atoms defining mirror plane" << endl;
380 Log() << Verbose(0) << " b - state normal vector of mirror plane" << endl;
381 Log() << Verbose(0) << " c - state two atoms in normal direction" << endl;
382 Log() << Verbose(0) << "all else - go back" << endl;
383 Log() << Verbose(0) << "===============================================" << endl;
384 Log() << Verbose(0) << "INPUT: ";
385 cin >> choice;
386
387 switch (choice) {
388 default:
389 case 'a': // three atoms defining mirror plane
390 first = mol->AskAtom("Enter first atom: ");
391 second = mol->AskAtom("Enter second atom: ");
392 third = mol->AskAtom("Enter third atom: ");
393
394 n.MakeNormalVector((const Vector *)&first->x,(const Vector *)&second->x,(const Vector *)&third->x);
395 break;
396 case 'b': // normal vector of mirror plane
397 Log() << Verbose(0) << "Enter normal vector of mirror plane." << endl;
398 n.AskPosition(mol->cell_size,0);
399 n.Normalize();
400 break;
401 case 'c': // three atoms defining mirror plane
402 first = mol->AskAtom("Enter first atom: ");
403 second = mol->AskAtom("Enter second atom: ");
404
405 n.CopyVector((const Vector *)&first->x);
406 n.SubtractVector((const Vector *)&second->x);
407 n.Normalize();
408 break;
409 };
410 Log() << Verbose(0) << "Normal vector: ";
411 n.Output();
412 Log() << Verbose(0) << endl;
413 mol->Mirror((const Vector *)&n);
414};
415
416/** Submenu for removing the atoms from the molecule.
417 * \param *mol molecule with all the atoms
418 */
419void oldmenu::RemoveAtoms(molecule *mol)
420{
421 atom *first, *second;
422 int axis;
423 double tmp1, tmp2;
424 char choice; // menu choice char
425
426 Log() << Verbose(0) << "===========REMOVE ATOMS=========================" << endl;
427 Log() << Verbose(0) << " a - state atom for removal by number" << endl;
428 Log() << Verbose(0) << " b - keep only in radius around atom" << endl;
429 Log() << Verbose(0) << " c - remove this with one axis greater value" << endl;
430 Log() << Verbose(0) << "all else - go back" << endl;
431 Log() << Verbose(0) << "===============================================" << endl;
432 Log() << Verbose(0) << "INPUT: ";
433 cin >> choice;
434
435 switch (choice) {
436 default:
437 case 'a':
438 if (mol->RemoveAtom(mol->AskAtom("Enter number of atom within molecule: ")))
439 Log() << Verbose(1) << "Atom removed." << endl;
440 else
441 Log() << Verbose(1) << "Atom not found." << endl;
442 break;
443 case 'b':
444 second = mol->AskAtom("Enter number of atom as reference point: ");
445 Log() << Verbose(0) << "Enter radius: ";
446 cin >> tmp1;
447 first = mol->start;
448 second = first->next;
449 while(second != mol->end) {
450 first = second;
451 second = first->next;
452 if (first->x.DistanceSquared((const Vector *)&second->x) > tmp1*tmp1) // distance to first above radius ...
453 mol->RemoveAtom(first);
454 }
455 break;
456 case 'c':
457 Log() << Verbose(0) << "Which axis is it: ";
458 cin >> axis;
459 Log() << Verbose(0) << "Lower boundary: ";
460 cin >> tmp1;
461 Log() << Verbose(0) << "Upper boundary: ";
462 cin >> tmp2;
463 first = mol->start;
464 second = first->next;
465 while(second != mol->end) {
466 first = second;
467 second = first->next;
468 if ((first->x.x[axis] < tmp1) || (first->x.x[axis] > tmp2)) {// out of boundary ...
469 //Log() << Verbose(0) << "Atom " << *first << " with " << first->x.x[axis] << " on axis " << axis << " is out of bounds [" << tmp1 << "," << tmp2 << "]." << endl;
470 mol->RemoveAtom(first);
471 }
472 }
473 break;
474 };
475 //mol->Output();
476 choice = 'r';
477};
478
479/** Submenu for measuring out the atoms in the molecule.
480 * \param *periode periodentafel
481 * \param *mol molecule with all the atoms
482 */
483void oldmenu::MeasureAtoms(periodentafel *periode, molecule *mol, config *configuration)
484{
485 atom *first, *second, *third;
486 Vector x,y;
487 double min[256], tmp1, tmp2, tmp3;
488 int Z;
489 char choice; // menu choice char
490
491 Log() << Verbose(0) << "===========MEASURE ATOMS=========================" << endl;
492 Log() << Verbose(0) << " a - calculate bond length between one atom and all others" << endl;
493 Log() << Verbose(0) << " b - calculate bond length between two atoms" << endl;
494 Log() << Verbose(0) << " c - calculate bond angle" << endl;
495 Log() << Verbose(0) << " d - calculate principal axis of the system" << endl;
496 Log() << Verbose(0) << " e - calculate volume of the convex envelope" << endl;
497 Log() << Verbose(0) << " f - calculate temperature from current velocity" << endl;
498 Log() << Verbose(0) << " g - output all temperatures per step from velocities" << endl;
499 Log() << Verbose(0) << "all else - go back" << endl;
500 Log() << Verbose(0) << "===============================================" << endl;
501 Log() << Verbose(0) << "INPUT: ";
502 cin >> choice;
503
504 switch(choice) {
505 default:
506 Log() << Verbose(1) << "Not a valid choice." << endl;
507 break;
508 case 'a':
509 first = mol->AskAtom("Enter first atom: ");
510 for (int i=MAX_ELEMENTS;i--;)
511 min[i] = 0.;
512
513 second = mol->start;
514 while ((second->next != mol->end)) {
515 second = second->next; // advance
516 Z = second->type->Z;
517 tmp1 = 0.;
518 if (first != second) {
519 x.CopyVector((const Vector *)&first->x);
520 x.SubtractVector((const Vector *)&second->x);
521 tmp1 = x.Norm();
522 }
523 if ((tmp1 != 0.) && ((min[Z] == 0.) || (tmp1 < min[Z]))) min[Z] = tmp1;
524 //Log() << Verbose(0) << "Bond length between Atom " << first->nr << " and " << second->nr << ": " << tmp1 << " a.u." << endl;
525 }
526 for (int i=MAX_ELEMENTS;i--;)
527 if (min[i] != 0.) Log() << Verbose(0) << "Minimum Bond length between " << first->type->name << " Atom " << first->nr << " and next Ion of type " << (periode->FindElement(i))->name << ": " << min[i] << " a.u." << endl;
528 break;
529
530 case 'b':
531 first = mol->AskAtom("Enter first atom: ");
532 second = mol->AskAtom("Enter second atom: ");
533 for (int i=NDIM;i--;)
534 min[i] = 0.;
535 x.CopyVector((const Vector *)&first->x);
536 x.SubtractVector((const Vector *)&second->x);
537 tmp1 = x.Norm();
538 Log() << Verbose(1) << "Distance vector is ";
539 x.Output();
540 Log() << Verbose(0) << "." << endl << "Norm of distance is " << tmp1 << "." << endl;
541 break;
542
543 case 'c':
544 Log() << Verbose(0) << "Evaluating bond angle between three - first, central, last - atoms." << endl;
545 first = mol->AskAtom("Enter first atom: ");
546 second = mol->AskAtom("Enter central atom: ");
547 third = mol->AskAtom("Enter last atom: ");
548 tmp1 = tmp2 = tmp3 = 0.;
549 x.CopyVector((const Vector *)&first->x);
550 x.SubtractVector((const Vector *)&second->x);
551 y.CopyVector((const Vector *)&third->x);
552 y.SubtractVector((const Vector *)&second->x);
553 Log() << Verbose(0) << "Bond angle between first atom Nr." << first->nr << ", central atom Nr." << second->nr << " and last atom Nr." << third->nr << ": ";
554 Log() << Verbose(0) << (acos(x.ScalarProduct((const Vector *)&y)/(y.Norm()*x.Norm()))/M_PI*180.) << " degrees" << endl;
555 break;
556 case 'd':
557 Log() << Verbose(0) << "Evaluating prinicipal axis." << endl;
558 Log() << Verbose(0) << "Shall we rotate? [0/1]: ";
559 cin >> Z;
560 if ((Z >=0) && (Z <=1))
561 mol->PrincipalAxisSystem((bool)Z);
562 else
563 mol->PrincipalAxisSystem(false);
564 break;
565 case 'e':
566 {
567 Log() << Verbose(0) << "Evaluating volume of the convex envelope.";
568 class Tesselation *TesselStruct = NULL;
569 const LinkedCell *LCList = NULL;
570 LCList = new LinkedCell(mol, 10.);
571 FindConvexBorder(mol, TesselStruct, LCList, NULL);
572 double clustervolume = VolumeOfConvexEnvelope(TesselStruct, configuration);
573 Log() << Verbose(0) << "The tesselated surface area is " << clustervolume << "." << endl;\
574 delete(LCList);
575 delete(TesselStruct);
576 }
577 break;
578 case 'f':
579 mol->OutputTemperatureFromTrajectories((ofstream *)&cout, mol->MDSteps-1, mol->MDSteps);
580 break;
581 case 'g':
582 {
583 char filename[255];
584 Log() << Verbose(0) << "Please enter filename: " << endl;
585 cin >> filename;
586 Log() << Verbose(1) << "Storing temperatures in " << filename << "." << endl;
587 ofstream *output = new ofstream(filename, ios::trunc);
588 if (!mol->OutputTemperatureFromTrajectories(output, 0, mol->MDSteps))
589 Log() << Verbose(2) << "File could not be written." << endl;
590 else
591 Log() << Verbose(2) << "File stored." << endl;
592 output->close();
593 delete(output);
594 }
595 break;
596 }
597};
598
599/** Submenu for measuring out the atoms in the molecule.
600 * \param *mol molecule with all the atoms
601 * \param *configuration configuration structure for the to be written config files of all fragments
602 */
603void oldmenu::FragmentAtoms(molecule *mol, config *configuration)
604{
605 int Order1;
606 clock_t start, end;
607
608 Log() << Verbose(0) << "Fragmenting molecule with current connection matrix ..." << endl;
609 Log() << Verbose(0) << "What's the desired bond order: ";
610 cin >> Order1;
611 if (mol->first->next != mol->last) { // there are bonds
612 start = clock();
613 mol->FragmentMolecule(Order1, configuration);
614 end = clock();
615 Log() << Verbose(0) << "Clocks for this operation: " << (end-start) << ", time: " << ((double)(end-start)/CLOCKS_PER_SEC) << "s." << endl;
616 } else
617 Log() << Verbose(0) << "Connection matrix has not yet been generated!" << endl;
618};
619
620/********************************************** Submenu routine **************************************/
621
622/** Submenu for manipulating atoms.
623 * \param *periode periodentafel
624 * \param *molecules list of molecules whose atoms are to be manipulated
625 */
626void oldmenu::ManipulateAtoms(periodentafel *periode, MoleculeListClass *molecules, config *configuration)
627{
628 atom *first, *second;
629 molecule *mol = NULL;
630 Vector x,y,z,n; // coordinates for absolute point in cell volume
631 double *factor; // unit factor if desired
632 double bond, minBond;
633 char choice; // menu choice char
634 bool valid;
635
636 Log() << Verbose(0) << "=========MANIPULATE ATOMS======================" << endl;
637 Log() << Verbose(0) << "a - add an atom" << endl;
638 Log() << Verbose(0) << "r - remove an atom" << endl;
639 Log() << Verbose(0) << "b - scale a bond between atoms" << endl;
640 Log() << Verbose(0) << "u - change an atoms element" << endl;
641 Log() << Verbose(0) << "l - measure lengths, angles, ... for an atom" << endl;
642 Log() << Verbose(0) << "all else - go back" << endl;
643 Log() << Verbose(0) << "===============================================" << endl;
644 if (molecules->NumberOfActiveMolecules() > 1)
645 eLog() << Verbose(2) << "There is more than one molecule active! Atoms will be added to each." << endl;
646 Log() << Verbose(0) << "INPUT: ";
647 cin >> choice;
648
649 switch (choice) {
650 default:
651 Log() << Verbose(0) << "Not a valid choice." << endl;
652 break;
653
654 case 'a': // add atom
655 for (MoleculeList::iterator ListRunner = molecules->ListOfMolecules.begin(); ListRunner != molecules->ListOfMolecules.end(); ListRunner++)
656 if ((*ListRunner)->ActiveFlag) {
657 mol = *ListRunner;
658 Log() << Verbose(0) << "Current molecule is: " << mol->IndexNr << "\t" << mol->name << endl;
659 AddAtoms(periode, mol);
660 }
661 break;
662
663 case 'b': // scale a bond
664 for (MoleculeList::iterator ListRunner = molecules->ListOfMolecules.begin(); ListRunner != molecules->ListOfMolecules.end(); ListRunner++)
665 if ((*ListRunner)->ActiveFlag) {
666 mol = *ListRunner;
667 Log() << Verbose(0) << "Current molecule is: " << mol->IndexNr << "\t" << mol->name << endl;
668 Log() << Verbose(0) << "Scaling bond length between two atoms." << endl;
669 first = mol->AskAtom("Enter first (fixed) atom: ");
670 second = mol->AskAtom("Enter second (shifting) atom: ");
671 minBond = 0.;
672 for (int i=NDIM;i--;)
673 minBond += (first->x.x[i]-second->x.x[i])*(first->x.x[i] - second->x.x[i]);
674 minBond = sqrt(minBond);
675 Log() << Verbose(0) << "Current Bond length between " << first->type->name << " Atom " << first->nr << " and " << second->type->name << " Atom " << second->nr << ": " << minBond << " a.u." << endl;
676 Log() << Verbose(0) << "Enter new bond length [a.u.]: ";
677 cin >> bond;
678 for (int i=NDIM;i--;) {
679 second->x.x[i] -= (second->x.x[i]-first->x.x[i])/minBond*(minBond-bond);
680 }
681 //Log() << Verbose(0) << "New coordinates of Atom " << second->nr << " are: ";
682 //second->Output(second->type->No, 1);
683 }
684 break;
685
686 case 'c': // unit scaling of the metric
687 for (MoleculeList::iterator ListRunner = molecules->ListOfMolecules.begin(); ListRunner != molecules->ListOfMolecules.end(); ListRunner++)
688 if ((*ListRunner)->ActiveFlag) {
689 mol = *ListRunner;
690 Log() << Verbose(0) << "Current molecule is: " << mol->IndexNr << "\t" << mol->name << endl;
691 Log() << Verbose(0) << "Angstroem -> Bohrradius: 1.8897261\t\tBohrradius -> Angstroem: 0.52917721" << endl;
692 Log() << Verbose(0) << "Enter three factors: ";
693 factor = new double[NDIM];
694 cin >> factor[0];
695 cin >> factor[1];
696 cin >> factor[2];
697 valid = true;
698 mol->Scale((const double ** const)&factor);
699 delete[](factor);
700 }
701 break;
702
703 case 'l': // measure distances or angles
704 for (MoleculeList::iterator ListRunner = molecules->ListOfMolecules.begin(); ListRunner != molecules->ListOfMolecules.end(); ListRunner++)
705 if ((*ListRunner)->ActiveFlag) {
706 mol = *ListRunner;
707 Log() << Verbose(0) << "Current molecule is: " << mol->IndexNr << "\t" << mol->name << endl;
708 MeasureAtoms(periode, mol, configuration);
709 }
710 break;
711
712 case 'r': // remove atom
713 for (MoleculeList::iterator ListRunner = molecules->ListOfMolecules.begin(); ListRunner != molecules->ListOfMolecules.end(); ListRunner++)
714 if ((*ListRunner)->ActiveFlag) {
715 mol = *ListRunner;
716 Log() << Verbose(0) << "Current molecule is: " << mol->IndexNr << "\t" << mol->name << endl;
717 RemoveAtoms(mol);
718 }
719 break;
720
721 case 'u': // change an atom's element
722 for (MoleculeList::iterator ListRunner = molecules->ListOfMolecules.begin(); ListRunner != molecules->ListOfMolecules.end(); ListRunner++)
723 if ((*ListRunner)->ActiveFlag) {
724 int Z;
725 mol = *ListRunner;
726 Log() << Verbose(0) << "Current molecule is: " << mol->IndexNr << "\t" << mol->name << endl;
727 first = NULL;
728 do {
729 Log() << Verbose(0) << "Change the element of which atom: ";
730 cin >> Z;
731 } while ((first = mol->FindAtom(Z)) == NULL);
732 Log() << Verbose(0) << "New element by atomic number Z: ";
733 cin >> Z;
734 first->type = periode->FindElement(Z);
735 Log() << Verbose(0) << "Atom " << first->nr << "'s element is " << first->type->name << "." << endl;
736 }
737 break;
738 }
739};
740
741/** Submenu for manipulating molecules.
742 * \param *periode periodentafel
743 * \param *molecules list of molecule to manipulate
744 */
745void oldmenu::ManipulateMolecules(periodentafel *periode, MoleculeListClass *molecules, config *configuration)
746{
747 atom *first = NULL;
748 Vector x,y,z,n; // coordinates for absolute point in cell volume
749 int j, axis, count, faktor;
750 char choice; // menu choice char
751 molecule *mol = NULL;
752 element **Elements;
753 Vector **vectors;
754 MoleculeLeafClass *Subgraphs = NULL;
755
756 Log() << Verbose(0) << "=========MANIPULATE GLOBALLY===================" << endl;
757 Log() << Verbose(0) << "c - scale by unit transformation" << endl;
758 Log() << Verbose(0) << "d - duplicate molecule/periodic cell" << endl;
759 Log() << Verbose(0) << "f - fragment molecule many-body bond order style" << endl;
760 Log() << Verbose(0) << "g - center atoms in box" << endl;
761 Log() << Verbose(0) << "i - realign molecule" << endl;
762 Log() << Verbose(0) << "m - mirror all molecules" << endl;
763 Log() << Verbose(0) << "o - create connection matrix" << endl;
764 Log() << Verbose(0) << "t - translate molecule by vector" << endl;
765 Log() << Verbose(0) << "all else - go back" << endl;
766 Log() << Verbose(0) << "===============================================" << endl;
767 if (molecules->NumberOfActiveMolecules() > 1)
768 eLog() << Verbose(2) << "There is more than one molecule active! Atoms will be added to each." << endl;
769 Log() << Verbose(0) << "INPUT: ";
770 cin >> choice;
771
772 switch (choice) {
773 default:
774 Log() << Verbose(0) << "Not a valid choice." << endl;
775 break;
776
777 case 'd': // duplicate the periodic cell along a given axis, given times
778 for (MoleculeList::iterator ListRunner = molecules->ListOfMolecules.begin(); ListRunner != molecules->ListOfMolecules.end(); ListRunner++)
779 if ((*ListRunner)->ActiveFlag) {
780 mol = *ListRunner;
781 Log() << Verbose(0) << "Current molecule is: " << mol->IndexNr << "\t" << mol->name << endl;
782 Log() << Verbose(0) << "State the axis [(+-)123]: ";
783 cin >> axis;
784 Log() << Verbose(0) << "State the factor: ";
785 cin >> faktor;
786
787 mol->CountAtoms(); // recount atoms
788 if (mol->AtomCount != 0) { // if there is more than none
789 count = mol->AtomCount; // is changed becausing of adding, thus has to be stored away beforehand
790 Elements = new element *[count];
791 vectors = new Vector *[count];
792 j = 0;
793 first = mol->start;
794 while (first->next != mol->end) { // make a list of all atoms with coordinates and element
795 first = first->next;
796 Elements[j] = first->type;
797 vectors[j] = &first->x;
798 j++;
799 }
800 if (count != j)
801 eLog() << Verbose(1) << "AtomCount " << count << " is not equal to number of atoms in molecule " << j << "!" << endl;
802 x.Zero();
803 y.Zero();
804 y.x[abs(axis)-1] = mol->cell_size[(abs(axis) == 2) ? 2 : ((abs(axis) == 3) ? 5 : 0)] * abs(axis)/axis; // last term is for sign, first is for magnitude
805 for (int i=1;i<faktor;i++) { // then add this list with respective translation factor times
806 x.AddVector(&y); // per factor one cell width further
807 for (int k=count;k--;) { // go through every atom of the original cell
808 first = new atom(); // create a new body
809 first->x.CopyVector(vectors[k]); // use coordinate of original atom
810 first->x.AddVector(&x); // translate the coordinates
811 first->type = Elements[k]; // insert original element
812 mol->AddAtom(first); // and add to the molecule (which increments ElementsInMolecule, AtomCount, ...)
813 }
814 }
815 if (mol->first->next != mol->last) // if connect matrix is present already, redo it
816 mol->CreateAdjacencyList(mol->BondDistance, configuration->GetIsAngstroem(), &BondGraph::CovalentMinMaxDistance, NULL);
817 // free memory
818 delete[](Elements);
819 delete[](vectors);
820 // correct cell size
821 if (axis < 0) { // if sign was negative, we have to translate everything
822 x.Zero();
823 x.AddVector(&y);
824 x.Scale(-(faktor-1));
825 mol->Translate(&x);
826 }
827 mol->cell_size[(abs(axis) == 2) ? 2 : ((abs(axis) == 3) ? 5 : 0)] *= faktor;
828 }
829 }
830 break;
831
832 case 'f':
833 FragmentAtoms(mol, configuration);
834 break;
835
836 case 'g': // center the atoms
837 for (MoleculeList::iterator ListRunner = molecules->ListOfMolecules.begin(); ListRunner != molecules->ListOfMolecules.end(); ListRunner++)
838 if ((*ListRunner)->ActiveFlag) {
839 mol = *ListRunner;
840 Log() << Verbose(0) << "Current molecule is: " << mol->IndexNr << "\t" << mol->name << endl;
841 CenterAtoms(mol);
842 }
843 break;
844
845 case 'i': // align all atoms
846 for (MoleculeList::iterator ListRunner = molecules->ListOfMolecules.begin(); ListRunner != molecules->ListOfMolecules.end(); ListRunner++)
847 if ((*ListRunner)->ActiveFlag) {
848 mol = *ListRunner;
849 Log() << Verbose(0) << "Current molecule is: " << mol->IndexNr << "\t" << mol->name << endl;
850 AlignAtoms(periode, mol);
851 }
852 break;
853
854 case 'm': // mirror atoms along a given axis
855 for (MoleculeList::iterator ListRunner = molecules->ListOfMolecules.begin(); ListRunner != molecules->ListOfMolecules.end(); ListRunner++)
856 if ((*ListRunner)->ActiveFlag) {
857 mol = *ListRunner;
858 Log() << Verbose(0) << "Current molecule is: " << mol->IndexNr << "\t" << mol->name << endl;
859 MirrorAtoms(mol);
860 }
861 break;
862
863 case 'o': // create the connection matrix
864 for (MoleculeList::iterator ListRunner = molecules->ListOfMolecules.begin(); ListRunner != molecules->ListOfMolecules.end(); ListRunner++)
865 if ((*ListRunner)->ActiveFlag) {
866 mol = *ListRunner;
867 double bonddistance;
868 clock_t start,end;
869 Log() << Verbose(0) << "What's the maximum bond distance: ";
870 cin >> bonddistance;
871 start = clock();
872 mol->CreateAdjacencyList(bonddistance, configuration->GetIsAngstroem(), &BondGraph::CovalentMinMaxDistance, NULL);
873 end = clock();
874 Log() << Verbose(0) << "Clocks for this operation: " << (end-start) << ", time: " << ((double)(end-start)/CLOCKS_PER_SEC) << "s." << endl;
875 }
876 break;
877
878 case 't': // translate all atoms
879 for (MoleculeList::iterator ListRunner = molecules->ListOfMolecules.begin(); ListRunner != molecules->ListOfMolecules.end(); ListRunner++)
880 if ((*ListRunner)->ActiveFlag) {
881 mol = *ListRunner;
882 Log() << Verbose(0) << "Current molecule is: " << mol->IndexNr << "\t" << mol->name << endl;
883 Log() << Verbose(0) << "Enter translation vector." << endl;
884 x.AskPosition(mol->cell_size,0);
885 mol->Center.AddVector((const Vector *)&x);
886 }
887 break;
888 }
889 // Free all
890 if (Subgraphs != NULL) { // free disconnected subgraph list of DFS analysis was performed
891 while (Subgraphs->next != NULL) {
892 Subgraphs = Subgraphs->next;
893 delete(Subgraphs->previous);
894 }
895 delete(Subgraphs);
896 }
897};
898
899
900/** Submenu for creating new molecules.
901 * \param *periode periodentafel
902 * \param *molecules list of molecules to add to
903 */
904void oldmenu::EditMolecules(periodentafel *periode, MoleculeListClass *molecules)
905{
906 char choice; // menu choice char
907 Vector center;
908 int nr, count;
909 molecule *mol = NULL;
910
911 Log() << Verbose(0) << "==========EDIT MOLECULES=====================" << endl;
912 Log() << Verbose(0) << "c - create new molecule" << endl;
913 Log() << Verbose(0) << "l - load molecule from xyz file" << endl;
914 Log() << Verbose(0) << "n - change molecule's name" << endl;
915 Log() << Verbose(0) << "N - give molecules filename" << endl;
916 Log() << Verbose(0) << "p - parse atoms in xyz file into molecule" << endl;
917 Log() << Verbose(0) << "r - remove a molecule" << endl;
918 Log() << Verbose(0) << "all else - go back" << endl;
919 Log() << Verbose(0) << "===============================================" << endl;
920 Log() << Verbose(0) << "INPUT: ";
921 cin >> choice;
922
923 switch (choice) {
924 default:
925 Log() << Verbose(0) << "Not a valid choice." << endl;
926 break;
927 case 'c':
928 mol = new molecule(periode);
929 molecules->insert(mol);
930 break;
931
932 case 'l': // load from XYZ file
933 {
934 char filename[MAXSTRINGSIZE];
935 Log() << Verbose(0) << "Format should be XYZ with: ShorthandOfElement\tX\tY\tZ" << endl;
936 mol = new molecule(periode);
937 do {
938 Log() << Verbose(0) << "Enter file name: ";
939 cin >> filename;
940 } while (!mol->AddXYZFile(filename));
941 mol->SetNameFromFilename(filename);
942 // center at set box dimensions
943 mol->CenterEdge(&center);
944 mol->cell_size[0] = center.x[0];
945 mol->cell_size[1] = 0;
946 mol->cell_size[2] = center.x[1];
947 mol->cell_size[3] = 0;
948 mol->cell_size[4] = 0;
949 mol->cell_size[5] = center.x[2];
950 molecules->insert(mol);
951 }
952 break;
953
954 case 'n':
955 {
956 char filename[MAXSTRINGSIZE];
957 do {
958 Log() << Verbose(0) << "Enter index of molecule: ";
959 cin >> nr;
960 mol = molecules->ReturnIndex(nr);
961 } while (mol == NULL);
962 Log() << Verbose(0) << "Enter name: ";
963 cin >> filename;
964 strcpy(mol->name, filename);
965 }
966 break;
967
968 case 'N':
969 {
970 char filename[MAXSTRINGSIZE];
971 do {
972 Log() << Verbose(0) << "Enter index of molecule: ";
973 cin >> nr;
974 mol = molecules->ReturnIndex(nr);
975 } while (mol == NULL);
976 Log() << Verbose(0) << "Enter name: ";
977 cin >> filename;
978 mol->SetNameFromFilename(filename);
979 }
980 break;
981
982 case 'p': // parse XYZ file
983 {
984 char filename[MAXSTRINGSIZE];
985 mol = NULL;
986 do {
987 Log() << Verbose(0) << "Enter index of molecule: ";
988 cin >> nr;
989 mol = molecules->ReturnIndex(nr);
990 } while (mol == NULL);
991 Log() << Verbose(0) << "Format should be XYZ with: ShorthandOfElement\tX\tY\tZ" << endl;
992 do {
993 Log() << Verbose(0) << "Enter file name: ";
994 cin >> filename;
995 } while (!mol->AddXYZFile(filename));
996 mol->SetNameFromFilename(filename);
997 }
998 break;
999
1000 case 'r':
1001 Log() << Verbose(0) << "Enter index of molecule: ";
1002 cin >> nr;
1003 count = 1;
1004 for(MoleculeList::iterator ListRunner = molecules->ListOfMolecules.begin(); ListRunner != molecules->ListOfMolecules.end(); ListRunner++)
1005 if (nr == (*ListRunner)->IndexNr) {
1006 mol = *ListRunner;
1007 molecules->ListOfMolecules.erase(ListRunner);
1008 delete(mol);
1009 break;
1010 }
1011 break;
1012 }
1013};
1014
1015
1016/** Submenu for merging molecules.
1017 * \param *periode periodentafel
1018 * \param *molecules list of molecules to add to
1019 */
1020void oldmenu::MergeMolecules(periodentafel *periode, MoleculeListClass *molecules)
1021{
1022 char choice; // menu choice char
1023
1024 Log() << Verbose(0) << "===========MERGE MOLECULES=====================" << endl;
1025 Log() << Verbose(0) << "a - simple add of one molecule to another" << endl;
1026 Log() << Verbose(0) << "e - embedding merge of two molecules" << endl;
1027 Log() << Verbose(0) << "m - multi-merge of all molecules" << endl;
1028 Log() << Verbose(0) << "s - scatter merge of two molecules" << endl;
1029 Log() << Verbose(0) << "t - simple merge of two molecules" << endl;
1030 Log() << Verbose(0) << "all else - go back" << endl;
1031 Log() << Verbose(0) << "===============================================" << endl;
1032 Log() << Verbose(0) << "INPUT: ";
1033 cin >> choice;
1034
1035 switch (choice) {
1036 default:
1037 Log() << Verbose(0) << "Not a valid choice." << endl;
1038 break;
1039
1040 case 'a':
1041 {
1042 int src, dest;
1043 molecule *srcmol = NULL, *destmol = NULL;
1044 {
1045 do {
1046 Log() << Verbose(0) << "Enter index of destination molecule: ";
1047 cin >> dest;
1048 destmol = molecules->ReturnIndex(dest);
1049 } while ((destmol == NULL) && (dest != -1));
1050 do {
1051 Log() << Verbose(0) << "Enter index of source molecule to add from: ";
1052 cin >> src;
1053 srcmol = molecules->ReturnIndex(src);
1054 } while ((srcmol == NULL) && (src != -1));
1055 if ((src != -1) && (dest != -1))
1056 molecules->SimpleAdd(srcmol, destmol);
1057 }
1058 }
1059 break;
1060
1061 case 'e':
1062 {
1063 int src, dest;
1064 molecule *srcmol = NULL, *destmol = NULL;
1065 do {
1066 Log() << Verbose(0) << "Enter index of matrix molecule (the variable one): ";
1067 cin >> src;
1068 srcmol = molecules->ReturnIndex(src);
1069 } while ((srcmol == NULL) && (src != -1));
1070 do {
1071 Log() << Verbose(0) << "Enter index of molecule to merge into (the fixed one): ";
1072 cin >> dest;
1073 destmol = molecules->ReturnIndex(dest);
1074 } while ((destmol == NULL) && (dest != -1));
1075 if ((src != -1) && (dest != -1))
1076 molecules->EmbedMerge(destmol, srcmol);
1077 }
1078 break;
1079
1080 case 'm':
1081 {
1082 int nr;
1083 molecule *mol = NULL;
1084 do {
1085 Log() << Verbose(0) << "Enter index of molecule to merge into: ";
1086 cin >> nr;
1087 mol = molecules->ReturnIndex(nr);
1088 } while ((mol == NULL) && (nr != -1));
1089 if (nr != -1) {
1090 int N = molecules->ListOfMolecules.size()-1;
1091 int *src = new int(N);
1092 for(MoleculeList::iterator ListRunner = molecules->ListOfMolecules.begin(); ListRunner != molecules->ListOfMolecules.end(); ListRunner++)
1093 if ((*ListRunner)->IndexNr != nr)
1094 src[N++] = (*ListRunner)->IndexNr;
1095 molecules->SimpleMultiMerge(mol, src, N);
1096 delete[](src);
1097 }
1098 }
1099 break;
1100
1101 case 's':
1102 Log() << Verbose(0) << "Not implemented yet." << endl;
1103 break;
1104
1105 case 't':
1106 {
1107 int src, dest;
1108 molecule *srcmol = NULL, *destmol = NULL;
1109 {
1110 do {
1111 Log() << Verbose(0) << "Enter index of destination molecule: ";
1112 cin >> dest;
1113 destmol = molecules->ReturnIndex(dest);
1114 } while ((destmol == NULL) && (dest != -1));
1115 do {
1116 Log() << Verbose(0) << "Enter index of source molecule to merge into: ";
1117 cin >> src;
1118 srcmol = molecules->ReturnIndex(src);
1119 } while ((srcmol == NULL) && (src != -1));
1120 if ((src != -1) && (dest != -1))
1121 molecules->SimpleMerge(srcmol, destmol);
1122 }
1123 }
1124 break;
1125 }
1126};
1127
1128
1129/********************************************** Test routine **************************************/
1130
1131/** Is called always as option 'T' in the menu.
1132 * \param *molecules list of molecules
1133 */
1134void oldmenu::testroutine(MoleculeListClass *molecules)
1135{
1136 // the current test routine checks the functionality of the KeySet&Graph concept:
1137 // We want to have a multiindex (the KeySet) describing a unique subgraph
1138 int i, comp, counter=0;
1139
1140 // create a clone
1141 molecule *mol = NULL;
1142 if (molecules->ListOfMolecules.size() != 0) // clone
1143 mol = (molecules->ListOfMolecules.front())->CopyMolecule();
1144 else {
1145 eLog() << Verbose(0) << "I don't have anything to test on ... ";
1146 performCriticalExit();
1147 return;
1148 }
1149 atom *Walker = mol->start;
1150
1151 // generate some KeySets
1152 Log() << Verbose(0) << "Generating KeySets." << endl;
1153 KeySet TestSets[mol->AtomCount+1];
1154 i=1;
1155 while (Walker->next != mol->end) {
1156 Walker = Walker->next;
1157 for (int j=0;j<i;j++) {
1158 TestSets[j].insert(Walker->nr);
1159 }
1160 i++;
1161 }
1162 Log() << Verbose(0) << "Testing insertion of already present item in KeySets." << endl;
1163 KeySetTestPair test;
1164 test = TestSets[mol->AtomCount-1].insert(Walker->nr);
1165 if (test.second) {
1166 Log() << Verbose(1) << "Insertion worked?!" << endl;
1167 } else {
1168 Log() << Verbose(1) << "Insertion rejected: Present object is " << (*test.first) << "." << endl;
1169 }
1170 TestSets[mol->AtomCount].insert(mol->end->previous->nr);
1171 TestSets[mol->AtomCount].insert(mol->end->previous->previous->previous->nr);
1172
1173 // constructing Graph structure
1174 Log() << Verbose(0) << "Generating Subgraph class." << endl;
1175 Graph Subgraphs;
1176
1177 // insert KeySets into Subgraphs
1178 Log() << Verbose(0) << "Inserting KeySets into Subgraph class." << endl;
1179 for (int j=0;j<mol->AtomCount;j++) {
1180 Subgraphs.insert(GraphPair (TestSets[j],pair<int, double>(counter++, 1.)));
1181 }
1182 Log() << Verbose(0) << "Testing insertion of already present item in Subgraph." << endl;
1183 GraphTestPair test2;
1184 test2 = Subgraphs.insert(GraphPair (TestSets[mol->AtomCount],pair<int, double>(counter++, 1.)));
1185 if (test2.second) {
1186 Log() << Verbose(1) << "Insertion worked?!" << endl;
1187 } else {
1188 Log() << Verbose(1) << "Insertion rejected: Present object is " << (*(test2.first)).second.first << "." << endl;
1189 }
1190
1191 // show graphs
1192 Log() << Verbose(0) << "Showing Subgraph's contents, checking that it's sorted." << endl;
1193 Graph::iterator A = Subgraphs.begin();
1194 while (A != Subgraphs.end()) {
1195 Log() << Verbose(0) << (*A).second.first << ": ";
1196 KeySet::iterator key = (*A).first.begin();
1197 comp = -1;
1198 while (key != (*A).first.end()) {
1199 if ((*key) > comp)
1200 Log() << Verbose(0) << (*key) << " ";
1201 else
1202 Log() << Verbose(0) << (*key) << "! ";
1203 comp = (*key);
1204 key++;
1205 }
1206 Log() << Verbose(0) << endl;
1207 A++;
1208 }
1209 delete(mol);
1210};
1211
1212oldmenu::oldmenu()
1213{
1214 // TODO Auto-generated constructor stub
1215}
1216
1217oldmenu::~oldmenu()
1218{
1219 // TODO Auto-generated destructor stub
1220}
1221
1222void oldmenu::notImplementedYet() {
1223 Log() << Verbose(0) << "This method has not yet been moved to an appropriate class." << endl;
1224}
1225
1226void oldmenu::perform(MoleculeListClass *molecules, config *configuration, periodentafel *periode, char *ConfigFileName)
1227{
1228 TextMenu *main_menu = new TextMenu(Log() << Verbose(0), "Main Menu");
1229
1230 StringView *moleculeView = new StreamStringView(boost::bind(&MoleculeListClass::Enumerate,molecules,_1));
1231 new DisplayMenuItem(main_menu,moleculeView);
1232
1233 new SeperatorItem(main_menu);
1234
1235 MethodAction *setMoleculeAction = new MethodAction(boost::bind(&oldmenu::notImplementedYet,this));
1236 new ActionMenuItem('a',"set molecule (in)active",main_menu,setMoleculeAction);
1237
1238 MethodAction *editMoleculeAction = new MethodAction(boost::bind(&oldmenu::EditMolecules,this,periode, molecules));
1239 new ActionMenuItem('e',"edit molecules (load, parse, save)",main_menu,editMoleculeAction);
1240
1241 MethodAction *manipulateMoleculeAction = new MethodAction(boost::bind(&oldmenu::ManipulateMolecules,this,periode, molecules, configuration));
1242 new ActionMenuItem('g',"globally manipulate atoms in molecule",main_menu,manipulateMoleculeAction);
1243
1244 MethodAction *mergeMoleculeAction = new MethodAction(boost::bind(&oldmenu::MergeMolecules,this,periode, molecules));
1245 new ActionMenuItem('M',"Merge molecules",main_menu,mergeMoleculeAction);
1246
1247 MethodAction *manipulateAtomsAction = new MethodAction(boost::bind(&oldmenu::ManipulateAtoms,this,periode, molecules, configuration));
1248 new ActionMenuItem('m',"manipulate atoms",main_menu,manipulateAtomsAction);
1249
1250 new SeperatorItem(main_menu);
1251
1252 MethodAction *editConfigAction = new MethodAction(boost::bind(&config::Edit,configuration));
1253 new ActionMenuItem('c',"edit the current configuration",main_menu,editConfigAction);
1254
1255 new SeperatorItem(main_menu);
1256
1257 MethodAction *saveConfigAction = new MethodAction(boost::bind(&oldmenu::SaveConfig,this,ConfigFileName, configuration, periode, molecules));
1258 new ActionMenuItem('s',"save current setup to config file",main_menu,saveConfigAction);
1259
1260 MethodAction *doTestAction = new MethodAction(boost::bind(&oldmenu::testroutine,this,molecules));
1261 new ActionMenuItem('T',"call the current test routine",main_menu,doTestAction);
1262
1263 MethodAction *quitAction = new MethodAction(boost::bind(&TextMenu::doQuit,main_menu));
1264 new ActionMenuItem('q',"quit",main_menu,quitAction);
1265
1266
1267 main_menu->display();
1268
1269
1270 delete moleculeView;
1271
1272 delete setMoleculeAction;
1273 delete editMoleculeAction;
1274 delete manipulateMoleculeAction;
1275 delete mergeMoleculeAction;
1276 delete manipulateAtomsAction;
1277 delete editConfigAction;
1278 delete saveConfigAction;
1279 delete doTestAction;
1280 delete quitAction;
1281
1282 delete main_menu;
1283
1284#if 0
1285 char choice;
1286 int j;
1287 Log() << Verbose(0) << endl << "Now comes the real construction..." << endl;
1288 do {
1289 Log() << Verbose(0) << endl << endl;
1290 Log() << Verbose(0) << "============Molecule list=======================" << endl;
1291 molecules->Enumerate((ofstream *)&cout);
1292 Log() << Verbose(0) << "============Menu===============================" << endl;
1293 Log() << Verbose(0) << "a - set molecule (in)active" << endl; //to be moved
1294 Log() << Verbose(0) << "e - edit molecules (load, parse, save)" << endl; // done
1295 Log() << Verbose(0) << "g - globally manipulate atoms in molecule" << endl;
1296 Log() << Verbose(0) << "M - Merge molecules" << endl;
1297 Log() << Verbose(0) << "m - manipulate atoms" << endl;
1298 Log() << Verbose(0) << "-----------------------------------------------" << endl;
1299 Log() << Verbose(0) << "c - edit the current configuration" << endl;
1300 Log() << Verbose(0) << "-----------------------------------------------" << endl;
1301 Log() << Verbose(0) << "s - save current setup to config file" << endl;
1302 Log() << Verbose(0) << "T - call the current test routine" << endl;
1303 Log() << Verbose(0) << "q - quit" << endl; //done
1304 Log() << Verbose(0) << "===============================================" << endl;
1305 Log() << Verbose(0) << "Input: ";
1306 cin >> choice;
1307
1308 switch (choice) {
1309 case 'a': // (in)activate molecule
1310 {
1311 Log() << Verbose(0) << "Enter index of molecule: ";
1312 cin >> j;
1313 for(MoleculeList::iterator ListRunner = molecules->ListOfMolecules.begin(); ListRunner != molecules->ListOfMolecules.end(); ListRunner++)
1314 if ((*ListRunner)->IndexNr == j)
1315 (*ListRunner)->ActiveFlag = !(*ListRunner)->ActiveFlag;
1316 }
1317 break;
1318
1319 case 'c': // edit each field of the configuration
1320 configuration->Edit();
1321 break;
1322
1323 case 'e': // create molecule
1324 EditMolecules(periode, molecules);
1325 break;
1326
1327 case 'g': // manipulate molecules
1328 ManipulateMolecules(periode, molecules, configuration);
1329 break;
1330
1331 case 'M': // merge molecules
1332 MergeMolecules(periode, molecules);
1333 break;
1334
1335 case 'm': // manipulate atoms
1336 ManipulateAtoms(periode, molecules, configuration);
1337 break;
1338
1339 case 'q': // quit
1340 break;
1341
1342 case 's': // save to config file
1343 SaveConfig(ConfigFileName, configuration, periode, molecules);
1344 break;
1345
1346 case 'T':
1347 testroutine(molecules);
1348 break;
1349
1350 default:
1351 break;
1352 };
1353 } while (choice != 'q');
1354#endif
1355};
1356
1357/** Tries given filename or standard on saving the config file.
1358 * \param *ConfigFileName name of file
1359 * \param *configuration pointer to configuration structure with all the values
1360 * \param *periode pointer to periodentafel structure with all the elements
1361 * \param *molecules list of molecules structure with all the atoms and coordinates
1362 */
1363void oldmenu::SaveConfig(char *ConfigFileName, config *configuration, periodentafel *periode, MoleculeListClass *molecules)
1364{
1365 char filename[MAXSTRINGSIZE];
1366 ofstream output;
1367 molecule *mol = new molecule(periode);
1368
1369 if (!strcmp(configuration->configpath, configuration->GetDefaultPath())) {
1370 eLog() << Verbose(2) << "config is found under different path then stated in config file::defaultpath!" << endl;
1371 }
1372
1373
1374 // first save as PDB data
1375 if (ConfigFileName != NULL)
1376 strcpy(filename, ConfigFileName);
1377 else
1378 strcpy(filename,"main_pcp_linux");
1379 Log() << Verbose(0) << "Saving as pdb input ";
1380 if (configuration->SavePDB(filename, molecules))
1381 Log() << Verbose(0) << "done." << endl;
1382 else
1383 Log() << Verbose(0) << "failed." << endl;
1384
1385 // then save as tremolo data file
1386 if (ConfigFileName != NULL)
1387 strcpy(filename, ConfigFileName);
1388 else
1389 strcpy(filename,"main_pcp_linux");
1390 Log() << Verbose(0) << "Saving as tremolo data input ";
1391 if (configuration->SaveTREMOLO(filename, molecules))
1392 Log() << Verbose(0) << "done." << endl;
1393 else
1394 Log() << Verbose(0) << "failed." << endl;
1395
1396 // translate each to its center and merge all molecules in MoleculeListClass into this molecule
1397 int N = molecules->ListOfMolecules.size();
1398 int *src = new int[N];
1399 N=0;
1400 for (MoleculeList::iterator ListRunner = molecules->ListOfMolecules.begin(); ListRunner != molecules->ListOfMolecules.end(); ListRunner++) {
1401 src[N++] = (*ListRunner)->IndexNr;
1402 (*ListRunner)->Translate(&(*ListRunner)->Center);
1403 }
1404 molecules->SimpleMultiAdd(mol, src, N);
1405 delete[](src);
1406
1407 // ... and translate back
1408 for (MoleculeList::iterator ListRunner = molecules->ListOfMolecules.begin(); ListRunner != molecules->ListOfMolecules.end(); ListRunner++) {
1409 (*ListRunner)->Center.Scale(-1.);
1410 (*ListRunner)->Translate(&(*ListRunner)->Center);
1411 (*ListRunner)->Center.Scale(-1.);
1412 }
1413
1414 Log() << Verbose(0) << "Storing configuration ... " << endl;
1415 // get correct valence orbitals
1416 mol->CalculateOrbitals(*configuration);
1417 configuration->InitMaxMinStopStep = configuration->MaxMinStopStep = configuration->MaxPsiDouble;
1418 if (ConfigFileName != NULL) { // test the file name
1419 strcpy(filename, ConfigFileName);
1420 output.open(filename, ios::trunc);
1421 } else if (strlen(configuration->configname) != 0) {
1422 strcpy(filename, configuration->configname);
1423 output.open(configuration->configname, ios::trunc);
1424 } else {
1425 strcpy(filename, DEFAULTCONFIG);
1426 output.open(DEFAULTCONFIG, ios::trunc);
1427 }
1428 output.close();
1429 output.clear();
1430 Log() << Verbose(0) << "Saving of config file ";
1431 if (configuration->Save(filename, periode, mol))
1432 Log() << Verbose(0) << "successful." << endl;
1433 else
1434 Log() << Verbose(0) << "failed." << endl;
1435
1436 // and save to xyz file
1437 if (ConfigFileName != NULL) {
1438 strcpy(filename, ConfigFileName);
1439 strcat(filename, ".xyz");
1440 output.open(filename, ios::trunc);
1441 }
1442 else {
1443 strcpy(filename,"main_pcp_linux");
1444 strcat(filename, ".xyz");
1445 output.open(filename, ios::trunc);
1446 }
1447 Log() << Verbose(0) << "Saving of XYZ file ";
1448 if (mol->MDSteps <= 1) {
1449 if (mol->OutputXYZ(&output))
1450 Log() << Verbose(0) << "successful." << endl;
1451 else
1452 Log() << Verbose(0) << "failed." << endl;
1453 } else {
1454 if (mol->OutputTrajectoriesXYZ(&output))
1455 Log() << Verbose(0) << "successful." << endl;
1456 else
1457 Log() << Verbose(0) << "failed." << endl;
1458 }
1459 output.close();
1460 output.clear();
1461
1462 // and save as MPQC configuration
1463 if (ConfigFileName != NULL)
1464 strcpy(filename, ConfigFileName);
1465 else
1466 strcpy(filename,"main_pcp_linux");
1467 Log() << Verbose(0) << "Saving as mpqc input ";
1468 if (configuration->SaveMPQC(filename, mol))
1469 Log() << Verbose(0) << "done." << endl;
1470 else
1471 Log() << Verbose(0) << "failed." << endl;
1472
1473 if (!strcmp(configuration->configpath, configuration->GetDefaultPath())) {
1474 eLog() << Verbose(2) << "config is found under different path then stated in config file::defaultpath!" << endl;
1475 }
1476
1477 delete(mol);
1478};
Note: See TracBrowser for help on using the repository browser.