Source code for MDAnalysis.transformations.nojump
# -*- 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 Lesser GNU Public Licence, v2.1 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.
# doi: 10.25080/majora-629e541a-00e
#
# 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
#
"""\
No Jump Trajectory Unwrapping --- :mod:`MDAnalysis.transformations.nojump`
=======================================================================================
Unwraps the trajectory such that atoms never move more than half a periodic box length.
The consequence of this is that particles will diffuse across periodic boundaries when
needed. This unwrapping method is suitable as a preprocessing step to calculate
molecular diffusion, or more commonly to keep multi-domain proteins whole during trajectory
analysis.
The algorithm used is based on :footcite:p:`Kulke2022`.
.. autoclass:: NoJump
"""
import numpy as np
import warnings
from .base import TransformationBase
from ..due import due, Doi
from ..exceptions import NoDataError
[docs]class NoJump(TransformationBase):
    """
    Returns transformed coordinates for the given timestep so that an atom
    does not move more than half the periodic box size between two timesteps, and will move
    across periodic boundary edges. The algorithm used is based on :footcite:p:`Kulke2022`,
    equation B6 for non-orthogonal systems, so it is general to most applications where
    molecule trajectories should not "jump" from one side of a periodic box to another.
    Note that this transformation depends on a periodic box dimension being set for every
    frame in the trajectory, and that this box dimension can be transformed to an orthonormal
    unit cell. If not, an error is emitted. Since it is typical to transform all frames
    sequentially when unwrapping trajectories, warnings are emitted if non-sequential timesteps
    are transformed using NoJump.
    Example
    -------
    Suppose you had a molecular trajectory with a protein dimer that moved across a periodic
    boundary. This will normally appear to make the dimer split apart. This transformation
    uses the molecular motions between frames in a wrapped trajectory to create an unwrapped
    trajectory where molecules never move more than half a periodic box dimension between subsequent
    frames. Thus, the normal use case is to apply the transformation to every frame of a trajectory,
    and to do so sequentially.
    .. code-block:: python
        transformation = NoJump()
        u.trajectory.add_transformations(transformation)
        for ts in u.trajectory:
            print(ts.positions)
    In this case, ``ts.positions`` will return the NoJump unwrapped trajectory, which would keep
    protein dimers together if they are together in the inital timestep. The unwrapped trajectory may
    also be useful to compute diffusion coefficients via the Einstein relation.
    To reverse the process, wrap the coordinates again.
    Returns
    -------
    MDAnalysis.coordinates.timestep.Timestep
    References
    ----------
    .. footbibliography::
    """
    @due.dcite(
        Doi("10.1021/acs.jctc.2c00327"),
        description="Works through the orthogonal case for unwrapping, "
        "and proposes the non-orthogonal approach.",
        path=__name__,
    )
    def __init__(
        self,
        check_continuity=True,
        max_threads=None,
    ):
        # NoJump transforms are inherently unparallelizable, since
        # it depends on the previous frame's unwrapped coordinates
        super().__init__(max_threads=max_threads, parallelizable=False)
        self.prev = None
        self.old_frame = 0
        self.older_frame = "A"
        self.check_c = check_continuity
    def _transform(self, ts):
        L = ts.triclinic_dimensions
        if L is None:
            msg = f"Periodic box dimensions not provided at step {ts.frame}"
            raise NoDataError(msg)
        try:
            Linverse = np.linalg.inv(L)
        except np.linalg.LinAlgError:
            msg = f"Periodic box dimensions are not invertible at step {ts.frame}"
            raise NoDataError(msg)
        if ts.frame == 0:
            # We don't need to apply the transformation here. However, we need to
            # ensure we have the 0th frame coordinates in reduced form. We also need to
            # set an an appropriate value for self.older frame. This is so that on the
            # following frame we don't return early when we check
            # `self.older_frame != "A"`. If we return early, then the transformation is
            # not applied, and any jumps across boundaries that occur at that frame will
            # not be accounted for.
            self.prev = ts.positions @ Linverse
            self.old_frame = 0
            self.older_frame = -1
            return ts
        if (
            self.check_c
            and self.older_frame != "A"
            and (self.old_frame - self.older_frame)
            != (ts.frame - self.old_frame)
        ):
            warnings.warn(
                "NoJump detected that the interval between frames is unequal."
                "We are resetting the reference frame to the current timestep.",
                UserWarning,
            )
            self.prev = ts.positions @ Linverse
            self.old_frame = ts.frame
            self.older_frame = "A"
            return ts
        if self.check_c and np.abs(self.old_frame - ts.frame) != 1:
            warnings.warn(
                "NoJump transform is only accurate when positions"
                "do not move by more than half a box length."
                "Currently jumping between frames with a step of more than 1."
                "This might be fine, but depending on the trajectory stride,"
                "this might be inaccurate.",
                UserWarning,
            )
        # Convert into reduced coordinate space
        fcurrent = ts.positions @ Linverse
        fprev = (
            self.prev
        )  # Previous unwrapped coordinates in reduced box coordinates.
        # Calculate the new positions in reduced coordinate space (Equation B6 from
        # 10.1021/acs.jctc.2c00327). As it turns out, the displacement term can
        # be moved inside the round function in this coordinate space, as the
        # difference between wrapped and unwrapped coordinates is an integer.
        newpositions = fcurrent - np.round(fcurrent - fprev)
        # Convert back into real space
        ts.positions = newpositions @ L
        # Set things we need to save for the next frame.
        self.prev = (
            newpositions  # Note that this is in reduced coordinate space.
        )
        self.older_frame = self.old_frame
        self.old_frame = ts.frame
        return ts