/*
* Project: MoleCuilder
* Description: creates and alters molecular systems
* Copyright (C) 2012 University of Bonn. All rights reserved.
* Copyright (C) 2013-2014 Frederik Heber. All rights reserved.
*
*
* This file is part of MoleCuilder.
*
* MoleCuilder is free software: you can redistribute it and/or modify
* it under the terms of the GNU General Public License as published by
* the Free Software Foundation, either version 2 of the License, or
* (at your option) any later version.
*
* MoleCuilder is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
* GNU General Public License for more details.
*
* You should have received a copy of the GNU General Public License
* along with MoleCuilder. If not, see .
*/
/*
* UndoRedoHelpers.cpp
*
* Created on: Apr 5, 2012
* Author: heber
*/
// include config.h
#ifdef HAVE_CONFIG_H
#include
#endif
#include "CodePatterns/MemDebug.hpp"
#include "UndoRedoHelpers.hpp"
#include
#include
#include
#include "Atom/atom.hpp"
#include "molecule.hpp"
#include "Descriptors/AtomIdDescriptor.hpp"
#include "Descriptors/MoleculeIdDescriptor.hpp"
#include "CodePatterns/Assert.hpp"
#include "CodePatterns/Log.hpp"
#include "World.hpp"
#include "WorldTime.hpp"
bool MoleCuilder::AddAtomsFromAtomicInfo(const std::vector &atoms)
{
size_t i=0;
for (; i > &mol_atoms)
{
bool status = true;
for (std::map< moleculeId_t, std::vector >::const_iterator iter = mol_atoms.begin();
iter != mol_atoms.end(); ++iter) {
// re-create the atom
LOG(3, "DEBUG: Re-adding molecule " << iter->first << ".");
molecule *mol_Walker = World::getInstance().recreateMolecule(iter->first);
status &= (mol_Walker != NULL);
// add all its atoms
status &= AddAtomsFromAtomicInfo(iter->second);
}
if (!status) {
// remove all molecules again
for (std::map< moleculeId_t, std::vector >::const_iterator iter = mol_atoms.begin();
iter != mol_atoms.end(); ++iter) {
molecule * mol = World::getInstance().getMolecule(MoleculeById(iter->first));
if (mol != NULL)
removeAtomsinMolecule(mol);
}
// and announce the failure
return false;
}
return true;
}
void MoleCuilder::RemoveAtomsFromAtomicInfo(std::vector &atoms)
{
BOOST_FOREACH(const AtomicInfo &_atom, atoms) {
World::getInstance().destroyAtom(_atom.getId());
}
}
void MoleCuilder::StoreBondInformationFromAtoms(
const std::vector &atoms,
std::vector< BondInfo > &bonds)
{
ASSERT( bonds.empty(),
"StoreBondInformationFromAtoms() - give bonds vector is not empty.");
bonds.reserve(atoms.size()*4);
for (std::vector::const_iterator atomiter = atoms.begin();
atomiter != atoms.end(); ++atomiter) {
const BondList & _atom_bonds = (*atomiter)->getListOfBonds();
for(BondList::const_iterator iter = _atom_bonds.begin(); iter != _atom_bonds.end(); ++iter)
bonds.push_back( BondInfo(*iter) );
}
}
bool MoleCuilder::AddBondsFromBondInfo(const std::vector< BondInfo > &bonds)
{
bool status = true;
std::vector< BondInfo >::const_iterator iter = bonds.begin();
for(;iter != bonds.end(); ++iter) {
if (!(*iter).RecreateBond()) {
status = false;
break;
}
}
if (!status) {
// remove all added bonds again
for(std::vector< BondInfo >::const_iterator removeiter = bonds.begin();
removeiter != iter; ++removeiter) {
removeiter->RemoveBond();
}
}
return status;
}
void MoleCuilder::SetAtomsFromAtomicInfo(const std::vector &_movedatoms)
{
BOOST_FOREACH( const AtomicInfo &_atominfo, _movedatoms) {
const atomId_t id = _atominfo.getId();
atom * const _atom = World::getInstance().getAtom(AtomById(id));
ASSERT( _atom != NULL,
"MoleCuilder::SetAtomsFromAtomicInfo() - cannot find atom with id "
+toString(id)+" in the world.");
_atominfo.setAtom( *_atom );
}
}
void MoleCuilder::SelectAtomsFromAtomicInfo(const std::vector &_movedatoms)
{
BOOST_FOREACH( const AtomicInfo &_atominfo, _movedatoms) {
const atomId_t id = _atominfo.getId();
World::getInstance().selectAtom(id);
}
}
void MoleCuilder::ResetAtomPosition(const std::vector &movedatoms, const std::vector &MovedToVector)
{
boost::function setter =
boost::bind(&atom::setPosition, _1, _2);
ResetByFunction(movedatoms, MovedToVector, setter);
}
void MoleCuilder::ResetAtomVelocity(const std::vector &movedatoms, const std::vector &VelocityVector)
{
boost::function setter =
boost::bind(&atom::setAtomicVelocity, _1, _2);
ResetByFunction(movedatoms, VelocityVector, setter);
}
void MoleCuilder::ResetAtomForce(const std::vector &movedatoms, const std::vector &ForceVector)
{
boost::function setter =
boost::bind(&atom::setAtomicForce, _1, _2);
ResetByFunction(movedatoms, ForceVector, setter);
}
void MoleCuilder::ResetByFunction(
const std::vector &movedatoms,
const std::vector &MovedToVector,
boost::function &setter)
{
std::vector::const_iterator positer = MovedToVector.begin();
ASSERT(movedatoms.size() == MovedToVector.size(),
"MoleCuilder::ResetAtomPosition() - the number of atoms "
+toString(movedatoms.size())+" and the number of positions "
+toString(MovedToVector.size())+" is not the same.");
BOOST_FOREACH( const AtomicInfo &_atominfo, movedatoms) {
const atomId_t id = _atominfo.getId();
atom * const _atom = World::getInstance().getAtom(AtomById(id));
ASSERT( _atom != NULL,
"FillSphericalSurfaceAction::performRedo() - cannot find atom with id "
+toString(id)+" in the world.");
setter(_atom, *positer );
++positer;
}
}
void MoleCuilder::RemoveMoleculesWithAtomsByIds(const std::vector &ids)
{
for (std::vector::const_iterator iter = ids.begin();
iter != ids.end(); ++iter) {
molecule * mol = World::getInstance().getMolecule(MoleculeById(*iter));
if (mol != NULL) {
removeAtomsinMolecule(mol);
// molecules are automatically removed when empty
}
}
}
void MoleCuilder::removeLastStep(const std::vector &_atoms, const unsigned int _step)
{
for (size_t i=0; i<_atoms.size(); ++i) {
atom * const _atom = World::getInstance().getAtom(AtomById(_atoms[i]));
_atom->removeStep(_step);
}
}
void MoleCuilder::addNewStep(const std::vector &_movedatoms, const unsigned int _step)
{
for(size_t i=0; i< _movedatoms.size(); ++i) {
atom * const _atom = World::getInstance().getAtom(AtomById(_movedatoms[i].getId()));
_atom->UpdateStep(_step);
}
}
void MoleCuilder::addNewStep(const std::vector &_ids, const unsigned int _step)
{
for(size_t i=0; i< _ids.size(); ++i) {
atom * const _atom = World::getInstance().getAtom(AtomById(_ids[i]));
_atom->UpdateStep(_step);
}
}
std::vector MoleCuilder::getIdsFromAtomicInfo(const std::vector &movedatoms)
{
std::vector ids(movedatoms.size(), (size_t)-1);
std::transform(
movedatoms.begin(), movedatoms.end(),
ids.begin(),
boost::bind(&AtomicInfo::getId, _1));
return ids;
}