Source code for prody.dynamics.imanm

# -*- coding: utf-8 -*-
"""This module defines a class and a function for explicit membrane ANM calculations."""

import numpy as np

from prody import LOGGER
from prody.utilities import importLA, checkCoords, copy
from numpy import sqrt, zeros, ones, array, ceil, dot

from .anm import ANMBase
from .rtb import RTB

__all__ = ['imANM']

[docs]class imANM(RTB): """Class for implicit ANM (imANM) method ([FT00]_). .. [TL12] Lezon TR, Bahar I, Constraints Imposed by the Membrane Selectively Guide the Alternating Access Dynamics of the Glutamate Transporter GltPh """ def __init__(self, name='Unknown'): super(imANM, self).__init__(name)
[docs] def buildHessian(self, coords, blocks, cutoff=15., gamma=1., **kwargs): """Build Hessian matrix for given coordinate set. :arg coords: a coordinate set or an object with ``getCoords`` method :type coords: :class:`numpy.ndarray` :arg blocks: a list or array of block identifiers :type blocks: list, :class:`numpy.ndarray` :arg cutoff: cutoff distance (Å) for pairwise interactions, default is 15.0 Å :type cutoff: float :arg gamma: spring constant, default is 1.0 :type gamma: float :arg scale: scaling factor for force constant along Z-direction, default is 16.0 :type scale: float :arg membrane_low: minimum z-coordinate at which membrane scaling is applied default is 1.0 :type membrane_low: float :arg membrane_high: maximum z-coordinate at which membrane scaling is applied. If membrane_high < membrane_low, scaling will be applied to the entire structure default is -1.0 :type membrane_high: float """ scale = kwargs.pop('scale', 16.0) depth = kwargs.pop('depth', None) h = depth / 2 if depth is not None else None h = kwargs.pop('h', h) if h is not None: h = float(h) hu = h hl = -h else: hu = kwargs.pop('membrane_high', 13.0) hu = kwargs.pop('high', hu) hu = float(hu) hl = kwargs.pop('membrane_low', -13.0) hl = kwargs.pop('low', hl) hl = float(hl) try: coords = (coords._getCoords() if hasattr(coords, '_getCoords') else coords.getCoords()) except AttributeError: try: checkCoords(coords) except TypeError: raise TypeError('coords must be a Numpy array or an object ' 'with `getCoords` method') ANMBase.buildHessian(self, coords, cutoff=cutoff, gamma=gamma, **kwargs) ## Scale horizontal spring constants ## natm = self._n_atoms H = self._hessian s = sqrt(sqrt(scale)) S0 = array([[s*s, s*s, s], [s*s, s*s, s], [s , s, 1]], dtype=float) super_element = lambda i, j: H[i*3:(i+1)*3, j*3:(j+1)*3] scaler = lambda coords: S0 if coords[2] < hu and coords[2] > hl else ones((3, 3), dtype=float) for i in range(natm): Si = scaler(coords[i]) for j in range(i+1, natm): Sj = scaler(coords[j]) S = Si * Sj Hij = super_element(i, j) if np.any(Hij != 0) and np.any(S != 1): # update off-diagonal super-element Hij Hij *= S # use Hij's twin, Hji, which is yet to be changed to obtain the difference Hji = super_element(j, i) D = Hji - Hij # update Hji Hji *= S # update diagonal super-elements Hii = super_element(i, i) Hii += D Hjj = super_element(j, j) Hjj += D self.calcProjection(coords, blocks, **kwargs)
def test(pdb='2nwl-mem.pdb', blk='2nwl.blk'): from prody import parsePDB from numpy import zeros, linalg pdb = parsePDB(pdb, subset='ca') pdb.setData('block', zeros(len(pdb), int)) with open(blk) as inp: for line in inp: if line.startswith('BLOCK'): _, b, n1, c1, r1, n2, c2, r2 = line.split() sel = pdb.select('chain {} and resnum {} to {}' .format(c1, r1, r2)) if sel: sel.setData('block', int(b)) pdb.setBetas(pdb.getData('block')) coords = pdb.getCoords() blocks = pdb.getBetas() from prody import writePDB writePDB('pdb2gb1_truncated.pdb', pdb) anm = imANM('2nwl') anm.buildHessian(coords, blocks, scale=64) #anm.calcModes() return anm