# -*- coding: utf-8 -*-
"""This module defines classes for handling mode data."""
import numpy as np
__all__ = ['Mode', 'Vector']
class VectorBase(object):
"""A base class for :class:`Mode` and :class:`Vector`.
This base class defines some shared methods, such as scalar multiplication
or addition of mode instances.
Defined operations are:
* Absolute value (abs(mode)) returns mode length
* Additive inverse (-mode)
* Mode addition (mode1 + mode2)
* Mode subtraction (mode1 - mode2)
* Scalar multiplication (x*mode or mode*x)
* Division by a scalar (mode/x)
* Dot product (mode1*mode2)
* Power (mode**x)"""
__slots__ = []
def __abs__(self):
return np.sqrt((self._getArray()**2).sum())
def __neg__(self):
return Vector(-self._getArray(), '-({0})'.format(str(self)),
self.is3d())
def __div__(self, other):
try:
result = self._getArray() / other
except Exception as err:
raise TypeError('{0} is not a scalar {1}'
.format(other, str(err)))
return Vector(result, '({0})/{1}'.format(str(self), other),
self.is3d())
def __idiv__(self, other):
return self.__div__(other)
def __mul__(self, other):
"""Returns scaled mode or dot product between modes."""
try:
other = other._getArray()
except AttributeError:
try:
result = other * self._getArray()
except Exception as err:
raise TypeError('{0} is not a scalar or a mode ({1})'
.format(other, str(err)))
else:
return Vector(result, '({1})*{0}'.format(other, str(self)),
self.is3d())
else:
try:
return np.dot(self._getArray(), other)
except Exception:
raise ValueError('{0} and {1} do not have same dimensions '
'({2})'.format(str(self), str(other),
str(err)))
def __rmul__(self, other):
"""Returns scaled mode or dot product between modes."""
try:
other = other._getArray()
except AttributeError:
try:
result = other * self._getArray()
except Exception as err:
raise TypeError('{0} is not a scalar or a mode ({1})'
.format(other, str(err)))
else:
return Vector(result, '{0}*({1})'.format(other, str(self)),
self.is3d())
else:
try:
return np.dot(self._getArray(), other)
except Exception:
raise ValueError('{0} and {1} do not have same dimensions '
'({2})'.format(str(self), str(other),
str(err)))
def __imul__(self, other):
return self.__mul__(other)
def __add__(self, other):
if isinstance(other, VectorBase):
if len(self) != len(other):
raise ValueError('modes do not have the same length')
return Vector(self._getArray() + other._getArray(),
'({0}) + ({1})'.format(str(self), str(other)),
self.is3d())
else:
raise TypeError('{0} is not a mode instance'.format(other))
def __radd__(self, other):
if isinstance(other, VectorBase):
if len(self) != len(other):
raise ValueError('modes do not have the same length')
return Vector(self._getArray() + other._getArray(),
'({0}) + ({1})'.format(str(other), str(self)),
self.is3d())
else:
raise TypeError('{0} is not a mode instance'.format(other))
def __iadd__(self, other):
return self.__add__(other)
def __sub__(self, other):
if isinstance(other, VectorBase):
if len(self) != len(other):
raise ValueError('modes do not have the same length')
return Vector(self._getArray() - other._getArray(),
'({0}) - ({1})'.format(str(self), str(other)),
self.is3d())
else:
raise TypeError('{0} is not a mode instance'.format(other))
def __rsub__(self, other):
if isinstance(other, VectorBase):
if len(self) != len(other):
raise ValueError('modes do not have the same length')
return Vector(other._getArray() - self._getArray(),
'({0}) - ({1})'.format(str(other), str(self)),
self.is3d())
else:
raise TypeError('{0} is not a mode instance'.format(other))
def __isub__(self, other):
return self.__sub__(other)
def __pow__(self, other):
try:
result = self._getArray() ** other
except Exception as err:
raise TypeError('{0} is not a scalar ({0})'
.format(other, str(err)))
else:
return Vector(result, '({0})**{1}'.format(str(self), other),
self.is3d())
def getArray(self):
"""Returns a copy of array."""
pass
def _getArray(self):
"""Returns array."""
pass
def numAtoms(self):
"""Returns number of atoms."""
pass
def is3d(self):
"""Returns **True** if vector is 3d."""
pass
def getArrayNx3(self):
"""Returns a copy of array with shape (N, 3)."""
if self.is3d():
return self.getArray().reshape((self.numAtoms(), 3))
else:
return self.getArray()
def _getArrayNx3(self):
"""Returns a copy of array with shape (N, 3)."""
if self.is3d():
return self._getArray().reshape((self.numAtoms(), 3))
else:
return self._getArray()
def numModes(self):
"""Returns 1."""
return 1
[docs]class Mode(VectorBase):
"""A class to provide access to and operations on mode data.
"""
__slots__ = ['_model', '_index']
def __init__(self, model, index):
"""Initialize mode object as part of an NMA model.
:arg model: a normal mode analysis instance
:type model: :class:`.NMA`, :class:`.GNM`, or :class:`.PCA`
:arg index: index of the mode
:type index: int"""
self._model = model
self._index = int(index)
def __len__(self):
return self.numEntries()
def __repr__(self):
return '<Mode: {0} from {1}>'.format(self._index + 1,
str(self._model))
def __str__(self):
return 'Mode {0} from {1}'.format(self._index+1, str(self._model))
def __int__(self):
return self._index
def __float__(self):
return self.getEigval()
[docs] def is3d(self):
"""Returns **True** if mode instance is from a 3-dimensional model."""
return self._model.is3d()
[docs] def numAtoms(self):
"""Returns number of atoms."""
return self._model.numAtoms()
[docs] def numDOF(self):
"""Returns number of degrees of freedom."""
return self._model.numDOF()
[docs] def numEntries(self):
"""Returns number of entries in the eigenvector."""
return len(self._getArray())
[docs] def getTitle(self):
"""A descriptive title for the mode instance."""
return str(self)
[docs] def getIndex(self):
"""Returns the index of the mode. Note that mode indices are
zero-based."""
return self._index
[docs] def getModel(self):
"""Returns the model that the mode instance belongs to."""
return self._model
[docs] def getArray(self):
"""Returns a copy of the normal mode array (eigenvector)."""
return self._model.getArray()[:, self._index].copy()
getEigvec = getArray
getEigvecs = getEigvec
def _getArray(self):
"""Returns a copy of the normal mode array (eigenvector)."""
return self._model._getArray()[:, self._index]
[docs] def getEigval(self):
"""Returns normal mode eigenvalue. For :class:`.PCA` and :class:`.EDA`
models built using coordinate data in Å, unit of eigenvalues is |A2|.
For :class:`.ANM` and :class:`.GNM`, on the other hand, eigenvalues
are in arbitrary or relative units but they correlate with stiffness
of the motion along associated eigenvector."""
return self._model._eigvals[self._index]
getEigvals = getEigval
[docs] def getVariance(self):
"""Returns variance along the mode. For :class:`.PCA` and :class:`.EDA`
models built using coordinate data in Å, unit of variance is |A2|. For
:class:`.ANM` and :class:`.GNM`, on the other hand, variance is the
inverse of the eigenvalue, so it has arbitrary or relative units."""
return self._model._vars[self._index]
getVariances = getVariance
[docs]class Vector(VectorBase):
"""A class to provide operations on a modified mode array. This class
holds only mode array (i.e. eigenvector) data, and has no associations
with an NMA instance. Scalar multiplication of :class:`Mode` instance
or addition of two :class:`Mode` instances results in a :class:`Vector`
instance."""
__slots__ = ['_title', '_array', '_is3d']
def __init__(self, array, title='Unknown', is3d=True):
"""Instantiate with a name, an array, and a 3d flag."""
try:
ndim, shape = array.ndim, array.shape
except AttributeError:
array = np.array(array)
ndim, shape = array.ndim, array.shape
if ndim != 1:
raise ValueError('array.ndim must be 1')
is3d = bool(is3d)
if is3d and shape[0] % 3 != 0:
raise ValueError('len(array) must be a multiple of 3')
self._title = str(title)
self._array = array
self._is3d = is3d
def __len__(self):
return len(self._array)
def __repr__(self):
return '<Vector: {0}>'.format(self._title)
def __str__(self):
return self._title
[docs] def is3d(self):
"""Returns **True** if vector instance describes a 3-dimensional
property, such as a deformation for a set of atoms."""
return self._is3d
[docs] def getTitle(self):
"""Get the descriptive title for the vector instance."""
return self._title
[docs] def setTitle(self, title):
"""Set the descriptive title for the vector instance."""
self._title = str(title)
[docs] def getArray(self):
"""Returns a copy of array."""
return self._array.copy()
def _getArray(self):
"""Returns array."""
return self._array
[docs] def getNormed(self):
"""Returns mode after normalizing it."""
return Vector(self._array/(self._array**2).sum()**0.5,
'({0})/||{0}||'.format(self._title), self._is3d)
[docs] def numDOF(self):
"""Returns number of degrees of freedom."""
return len(self._array)
[docs] def numEntries(self):
"""Returns number of entries in the vector."""
return len(self._array)
[docs] def numAtoms(self):
"""Returns number of atoms. For a 3-dimensional vector, returns length
of the vector divided by 3."""
if self._is3d:
return len(self._array)//3
else:
return len(self._array)