From 13ee1ddf858469149c637237224189e58dc38eb2 Mon Sep 17 00:00:00 2001 From: Lester Hedges Date: Tue, 23 Mar 2021 09:58:02 +0000 Subject: [PATCH] Use independent hill width for each CV component. [ref #194] --- .../CollectiveVariable/_funnel.py | 36 ++++++++++++------- python/BioSimSpace/Process/_plumed.py | 14 ++++++-- 2 files changed, 35 insertions(+), 15 deletions(-) diff --git a/python/BioSimSpace/Metadynamics/CollectiveVariable/_funnel.py b/python/BioSimSpace/Metadynamics/CollectiveVariable/_funnel.py index 15502662e..9cb2a30ec 100644 --- a/python/BioSimSpace/Metadynamics/CollectiveVariable/_funnel.py +++ b/python/BioSimSpace/Metadynamics/CollectiveVariable/_funnel.py @@ -44,7 +44,8 @@ class Funnel(_CollectiveVariable): """A class for a funnel collective variable.""" - def __init__(self, atoms0, atoms1, hill_width=_Length(0.025, "nanometer"), + def __init__(self, atoms0, atoms1, + hill_width=(_Length(0.025, "nanometer"), _Length(0.05, "nanometer")), width=_Length(0.6, "nanometers"), buffer=_Length(0.15, "nanometers"), steepness=1.5, inflection=_Length(2.0, "nanometers"), lower_bound=_Bound(_Length(0.5, "nanometers"), force_constant=2000), @@ -75,8 +76,9 @@ def __init__(self, atoms0, atoms1, hill_width=_Length(0.025, "nanometer"), The inflection point as a value of the projection along the funnel axis. - hill_width : :class:`Length ` - The width of the Gaussian hill used to sample this variable. + hill_width : (:class:`Length `, "class:`Length `) + The width of the Gaussian hill used to sample each component + of the collective variable. lower_bound : :class:`Bound ` A lower bound on the value of the collective variable. This is @@ -146,7 +148,7 @@ def __str__(self): string += ", buffer=%s" % self._buffer string += ", steepness=%s" % self._steepness string += ", inflection=%s" % self._inflection - string += ", hill_width=%s" % self._hill_width + string += ", hill_width=%s" % (self._hill_width,) if self._lower_bound is not None: string += ", lower_bound=%s" % self._lower_bound if self._upper_bound is not None: @@ -371,16 +373,25 @@ def setHillWidth(self, hill_width): ---------- hill_width : :class:`Length ` - The width of the Gaussian hill. + The width of the Gaussian hill for the two commponents of the + collective variable: the distance along the funnel projection + axis, and the orthogonal extent from the axis. """ - if type(hill_width) is not _Length: - raise TypeError("'hill_width' must be of type 'BioSimSpace.Types.Length'") - if hill_width.magnitude() < 0: - raise ValueError("'hill_width' must have a magnitude of > 0") + # Convert list to tuple. + if type(hill_width) is list: + hill_width = tuple(hill_width) + + if type(hill_width) is tuple: + if len(hill_width) != 2 or not all(isinstance(x, _Length) for x in hill_width): + raise ValueError("'hill_width' must be a two-component tuple of of type 'BioSimSpace.Metadynamics.Length'") + + for width in hill_width: + if width.magnitude() < 0: + raise ValueError("'hill_width' must have a magnitude of > 0") # Convert to the internal unit. - self._hill_width = hill_width.nanometers() + self._hill_width = tuple(x.nanometers() for x in hill_width) def getHillWidth(self): """Return the width of the Gaussian hill used to bias this collective @@ -389,8 +400,9 @@ def getHillWidth(self): Returns ------- - hill_width : :class:`Length ` - The width of the Gaussian hill. + hill_width : (:class:`Length `, "class:`Length `) + The width of the Gaussian hill for each component of the + collective variable. """ return self._hill_width diff --git a/python/BioSimSpace/Process/_plumed.py b/python/BioSimSpace/Process/_plumed.py index 60966630c..36543238d 100644 --- a/python/BioSimSpace/Process/_plumed.py +++ b/python/BioSimSpace/Process/_plumed.py @@ -599,9 +599,17 @@ def createConfig(self, system, protocol, is_restart=False): # Hill width. metad_string += " SIGMA=" - for idx, colvar in enumerate(colvars): - metad_string += "%s" % colvar.getHillWidth().magnitude() - if idx < self._num_colvar - 1: + for idx0, colvar in enumerate(colvars): + hill_width = colvar.getHillWidth() + if type(hill_width) is tuple: + last_hill = len(hill_width) - 1 + for idx1, width in enumerate(hill_width): + metad_string += "%s" % width.magnitude() + if idx1 < last_hill: + metad_string += "," + else: + metad_string += "%s" % hill_width.magnitude() + if idx0 < self._num_colvar - 1: metad_string += "," # Hill height.