mirror of
https://github.com/rdkit/rdkit.git
synced 2026-06-03 21:44:30 +08:00
184 lines
5.3 KiB
C++
184 lines
5.3 KiB
C++
//
|
|
// Copyright (C) 2001-2021 Greg Landrum and other RDKit contributors
|
|
//
|
|
// @@ All Rights Reserved @@
|
|
// This file is part of the RDKit.
|
|
// The contents are covered by the terms of the BSD license
|
|
// which is included in the file license.txt, found at the root
|
|
// of the RDKit source tree.
|
|
//
|
|
#include <RDGeneral/export.h>
|
|
#ifndef _RD_CONFORMER_H
|
|
#define _RD_CONFORMER_H
|
|
|
|
#include <Geometry/point.h>
|
|
#include <RDGeneral/types.h>
|
|
#include <boost/smart_ptr.hpp>
|
|
#include <RDGeneral/RDProps.h>
|
|
#include <limits>
|
|
#include <utility>
|
|
|
|
namespace RDKit {
|
|
class ROMol;
|
|
|
|
//! used to indicate errors from incorrect conformer access
|
|
class RDKIT_GRAPHMOL_EXPORT ConformerException : public std::exception {
|
|
public:
|
|
//! construct with an error message
|
|
ConformerException(const char *msg) : _msg(msg) {}
|
|
//! construct with an error message
|
|
ConformerException(std::string msg) : _msg(std::move(msg)) {}
|
|
//! get the error message
|
|
const char *what() const noexcept override { return _msg.c_str(); }
|
|
~ConformerException() noexcept override = default;
|
|
|
|
private:
|
|
std::string _msg;
|
|
};
|
|
|
|
//! The class for representing 2D or 3D conformation of a molecule
|
|
/*!
|
|
This class contains
|
|
- a pointer to the owing molecule
|
|
- a vector of 3D points (positions of atoms)
|
|
*/
|
|
class RDKIT_GRAPHMOL_EXPORT Conformer : public RDProps {
|
|
public:
|
|
friend class ROMol;
|
|
|
|
//! Constructor
|
|
Conformer() { d_positions.clear(); }
|
|
|
|
//! Constructor with number of atoms specified ID specification
|
|
Conformer(unsigned int numAtoms) {
|
|
if (numAtoms) {
|
|
d_positions.resize(numAtoms, RDGeom::Point3D(0.0, 0.0, 0.0));
|
|
} else {
|
|
d_positions.resize(0);
|
|
d_positions.clear();
|
|
}
|
|
}
|
|
|
|
//! Copy Constructor: initialize from a second conformation.
|
|
Conformer(const Conformer &other);
|
|
Conformer &operator=(const Conformer &other);
|
|
Conformer(Conformer &&o) noexcept
|
|
: RDProps(std::move(o)),
|
|
df_is3D(std::move(o.df_is3D)),
|
|
d_id(std::move(o.d_id)),
|
|
dp_mol(std::move(o.dp_mol)),
|
|
d_positions(std::move(o.d_positions)){};
|
|
Conformer &operator=(Conformer &&o) noexcept {
|
|
if (this == &o) {
|
|
return *this;
|
|
}
|
|
RDProps::operator=(std::move(o));
|
|
df_is3D = std::move(o.df_is3D);
|
|
d_id = std::move(o.d_id);
|
|
dp_mol = std::move(o.dp_mol);
|
|
d_positions = std::move(o.d_positions);
|
|
return *this;
|
|
}
|
|
//! Destructor
|
|
~Conformer() = default;
|
|
|
|
//! Resize the conformer so that more atoms location can be added.
|
|
//! Useful, for e.g., when adding hydrogens
|
|
void resize(unsigned int size) { d_positions.resize(size); }
|
|
|
|
//! Reserve more space for atom position
|
|
void reserve(unsigned int size) { d_positions.reserve(size); }
|
|
|
|
//! returns whether or not this instance belongs to a molecule
|
|
bool hasOwningMol() const { return dp_mol != nullptr; }
|
|
|
|
//! Get the molecule that owns this instance
|
|
ROMol &getOwningMol() const {
|
|
PRECONDITION(dp_mol, "no owner");
|
|
return *dp_mol;
|
|
}
|
|
|
|
//! Get a const reference to the vector of atom positions
|
|
const RDGeom::POINT3D_VECT &getPositions() const;
|
|
|
|
//! Get a reference to the atom positions
|
|
RDGeom::POINT3D_VECT &getPositions();
|
|
|
|
//! Get the position of the specified atom
|
|
const RDGeom::Point3D &getAtomPos(unsigned int atomId) const;
|
|
//! overload
|
|
template <class U>
|
|
const RDGeom::Point3D &getAtomPos(U atomId) const {
|
|
return getAtomPos(rdcast<unsigned int>(atomId));
|
|
}
|
|
|
|
//! Get the position of the specified atom
|
|
RDGeom::Point3D &getAtomPos(unsigned int atomId);
|
|
//! overload
|
|
template <class U>
|
|
RDGeom::Point3D &getAtomPos(U atomId) {
|
|
return getAtomPos(rdcast<unsigned int>(atomId));
|
|
}
|
|
|
|
//! Set the position of the specified atom
|
|
inline void setAtomPos(unsigned int atomId, const RDGeom::Point3D &position) {
|
|
if (atomId == std::numeric_limits<unsigned int>::max()) {
|
|
throw ValueErrorException("atom index overflow");
|
|
}
|
|
if (atomId >= d_positions.size()) {
|
|
d_positions.resize(atomId + 1, RDGeom::Point3D(0.0, 0.0, 0.0));
|
|
}
|
|
d_positions[atomId] = position;
|
|
}
|
|
//! overload
|
|
template <class U>
|
|
void setAtomPos(U atomId, const RDGeom::Point3D &position) {
|
|
return setAtomPos(rdcast<unsigned int>(atomId), position);
|
|
}
|
|
//! get the ID of this conformer
|
|
inline unsigned int getId() const { return d_id; }
|
|
|
|
//! set the ID of this conformer
|
|
inline void setId(unsigned int id) { d_id = id; }
|
|
|
|
//! Get the number of atoms
|
|
inline unsigned int getNumAtoms() const {
|
|
return rdcast<unsigned int>(d_positions.size());
|
|
}
|
|
inline bool is3D() const { return df_is3D; }
|
|
inline void set3D(bool v) { df_is3D = v; }
|
|
|
|
protected:
|
|
//! Set owning molecule
|
|
void setOwningMol(ROMol *mol);
|
|
|
|
//! Set owning molecule
|
|
void setOwningMol(ROMol &mol);
|
|
|
|
private:
|
|
bool df_is3D{true}; // is this a 3D conformation?
|
|
unsigned int d_id{0}; // id is the conformation
|
|
ROMol *dp_mol{nullptr}; // owning molecule
|
|
RDGeom::POINT3D_VECT d_positions; // positions of the atoms
|
|
void initFromOther(const Conformer &conf);
|
|
};
|
|
|
|
typedef boost::shared_ptr<Conformer> CONFORMER_SPTR;
|
|
|
|
//! Returns true if any of the z coords are non zero, false otherwise
|
|
/*!
|
|
\param conf Conformer object to analyze
|
|
*/
|
|
inline bool hasNonZeroZCoords(const Conformer &conf) {
|
|
for (auto p : conf.getPositions()) {
|
|
if (p.z != 0.0) {
|
|
return true;
|
|
}
|
|
}
|
|
return false;
|
|
}
|
|
|
|
} // namespace RDKit
|
|
|
|
#endif
|