Source code for MDAnalysis.transformations.fit

# -*- Mode: python; tab-width: 4; indent-tabs-mode:nil; coding:utf-8 -*-
# vim: tabstop=4 expandtab shiftwidth=4 softtabstop=4
#
# MDAnalysis --- https://www.mdanalysis.org
# Copyright (c) 2006-2017 The MDAnalysis Development Team and contributors
# (see the file AUTHORS for the full list of names)
#
# Released under the GNU Public Licence, v2 or any higher version
#
# Please cite your use of MDAnalysis in published work:
#
# R. J. Gowers, M. Linke, J. Barnoud, T. J. E. Reddy, M. N. Melo, S. L. Seyler,
# D. L. Dotson, J. Domanski, S. Buchoux, I. M. Kenney, and O. Beckstein.
# MDAnalysis: A Python package for the rapid analysis of molecular dynamics
# simulations. In S. Benthall and S. Rostrup editors, Proceedings of the 15th
# Python in Science Conference, pages 102-109, Austin, TX, 2016. SciPy.
#
# N. Michaud-Agrawal, E. J. Denning, T. B. Woolf, and O. Beckstein.
# MDAnalysis: A Toolkit for the Analysis of Molecular Dynamics Simulations.
# J. Comput. Chem. 32 (2011), 2319--2327, doi:10.1002/jcc.21787
#

"""\
Fitting transformations --- :mod:`MDAnalysis.transformations.fit`
=================================================================

Translate and/or rotates the coordinates of a given trajectory to align
a given AtomGroup to a reference structure.

.. autofunction:: fit_translation

.. autofunction:: fit_rot_trans

"""
from __future__ import absolute_import

import numpy as np
from functools import partial

from ..analysis import align
from ..lib.util import get_weights
from ..lib.transformations import euler_from_matrix, euler_matrix


[docs]def fit_translation(ag, reference, plane=None, weights=None): """Translates a given AtomGroup so that its center of geometry/mass matches the respective center of the given reference. A plane can be given by the user using the option `plane`, and will result in the removal of the translation motions of the AtomGroup over that particular plane. Example ------- Removing the translations of a given AtomGroup `ag` on the XY plane by fitting its center of mass to the center of mass of a reference `ref`: .. code-block:: python ag = u.select_atoms("protein") ref = mda.Universe("reference.pdb") transform = mda.transformations.fit_translation(ag, ref, plane="xy", weights="mass") u.trajectory.add_transformations(transform) Parameters ---------- ag : Universe or AtomGroup structure to translate, a :class:`~MDAnalysis.core.groups.AtomGroup` or a whole :class:`~MDAnalysis.core.universe.Universe` reference : Universe or AtomGroup reference structure, a :class:`~MDAnalysis.core.groups.AtomGroup` or a whole :class:`~MDAnalysis.core.universe.Universe` plane: str, optional used to define the plane on which the translations will be removed. Defined as a string of the plane. Suported planes are yz, xz and xy planes. weights : {"mass", ``None``} or array_like, optional choose weights. With ``"mass"`` uses masses as weights; with ``None`` weigh each atom equally. If a float array of the same length as `ag` is provided, use each element of the `array_like` as a weight for the corresponding atom in `ag`. Returns ------- MDAnalysis.coordinates.base.Timestep """ if plane is not None: axes = {'yz' : 0, 'xz' : 1, 'xy' : 2} try: plane = axes[plane] except (TypeError, KeyError): raise ValueError('{} is not a valid plane'.format(plane)) try: if ag.atoms.n_residues != reference.atoms.n_residues: raise ValueError("{} and {} have mismatched number of residues".format(ag,reference)) except AttributeError: raise AttributeError("{} or {} is not valid Universe/AtomGroup".format(ag,reference)) ref, mobile = align.get_matching_atoms(reference.atoms, ag.atoms) try: weights = align.get_weights(ref.atoms, weights=weights) except (ValueError, TypeError): raise ValueError("weights must be {'mass', None} or an iterable of the " "same size as the atomgroup.") ref_com = np.asarray(ref.center(weights), np.float32) def wrapped(ts): mobile_com = np.asarray(mobile.atoms.center(weights), np.float32) vector = ref_com - mobile_com if plane is not None: vector[plane] = 0 ts.positions += vector return ts return wrapped
[docs]def fit_rot_trans(ag, reference, plane=None, weights=None): """Perform a spatial superposition by minimizing the RMSD. Spatially align the group of atoms `ag` to `reference` by doing a RMSD fit. This fit works as a way to remove translations and rotations of a given AtomGroup in a trajectory. A plane can be given using the flag `plane` so that only translations and rotations in that particular plane are removed. This is useful for protein-membrane systems to where the membrane must remain in the same orientation. Example ------- Removing the translations and rotations of a given AtomGroup `ag` on the XY plane by fitting it to a reference `ref`, using the masses as weights for the RMSD fit: .. code-block:: python ag = u.select_atoms("protein") ref = mda.Universe("reference.pdb") transform = mda.transformations.fit_rot_trans(ag, ref, plane="xy", weights="mass") u.trajectory.add_transformations(transform) Parameters ---------- ag : Universe or AtomGroup structure to translate and rotate, a :class:`~MDAnalysis.core.groups.AtomGroup` or a whole :class:`~MDAnalysis.core.universe.Universe` reference : Universe or AtomGroup reference structure, a :class:`~MDAnalysis.core.groups.AtomGroup` or a whole :class:`~MDAnalysis.core.universe.Universe` plane: str, optional used to define the plane on which the rotations and translations will be removed. Defined as a string of the plane. Supported planes are "yz", "xz" and "xy" planes. weights : {"mass", ``None``} or array_like, optional choose weights. With ``"mass"`` uses masses as weights; with ``None`` weigh each atom equally. If a float array of the same length as `ag` is provided, use each element of the `array_like` as a weight for the corresponding atom in `ag`. Returns ------- MDAnalysis.coordinates.base.Timestep """ if plane is not None: axes = {'yz' : 0, 'xz' : 1, 'xy' : 2} try: plane = axes[plane] except (TypeError, KeyError): raise ValueError('{} is not a valid plane'.format(plane)) try: if ag.atoms.n_residues != reference.atoms.n_residues: raise ValueError("{} and {} have mismatched number of residues".format(ag,reference)) except AttributeError: raise AttributeError("{} or {} is not valid Universe/AtomGroup".format(ag,reference)) ref, mobile = align.get_matching_atoms(reference.atoms, ag.atoms) try: weights = align.get_weights(ref.atoms, weights=weights) except (ValueError, TypeError): raise ValueError("weights must be {'mass', None} or an iterable of the " "same size as the atomgroup.") ref_com = ref.center(weights) ref_coordinates = ref.atoms.positions - ref_com def wrapped(ts): mobile_com = mobile.atoms.center(weights) mobile_coordinates = mobile.atoms.positions - mobile_com rotation, dump = align.rotation_matrix(mobile_coordinates, ref_coordinates, weights=weights) vector = ref_com if plane is not None: matrix = np.r_[rotation, np.zeros(3).reshape(1,3)] matrix = np.c_[matrix, np.zeros(4)] euler_angs = np.asarray(euler_from_matrix(matrix, axes='sxyz'), np.float32) for i in range(0, euler_angs.size): euler_angs[i] = ( euler_angs[plane] if i == plane else 0) rotation = euler_matrix(euler_angs[0], euler_angs[1], euler_angs[2], axes='sxyz')[:3, :3] vector[plane] = mobile_com[plane] ts.positions = ts.positions - mobile_com ts.positions = np.dot(ts.positions, rotation.T) ts.positions = ts.positions + vector return ts return wrapped