#
# Copyright (C) 2010-2022 The ESPResSo project
#
# This file is part of ESPResSo.
#
# ESPResSo is free software: you can redistribute it and/or modify
# it under the terms of the GNU General Public License as published by
# the Free Software Foundation, either version 3 of the License, or
# (at your option) any later version.
#
# ESPResSo is distributed in the hope that it will be useful,
# but WITHOUT ANY WARRANTY; without even the implied warranty of
# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
# GNU General Public License for more details.
#
# You should have received a copy of the GNU General Public License
# along with this program. If not, see <http://www.gnu.org/licenses/>.
#
from .script_interface import ScriptObjectList, ScriptInterfaceHelper, script_interface_register
import numpy as np
import itertools
[docs]
@script_interface_register
class Constraints(ScriptObjectList):
"""
List of active constraints. Add a :class:`espressomd.constraints.Constraint`
to make it active in the system, or remove it to make it inactive.
"""
_so_name = "Constraints::Constraints"
[docs]
def add(self, *args, **kwargs):
"""
Add a constraint to the list.
Parameters
----------
constraint: :class:`espressomd.constraints.Constraint`
Either a constraint object...
**kwargs
... or parameters to construct an
:class:`espressomd.constraints.ShapeBasedConstraint`
Returns
----------
constraint : :class:`espressomd.constraints.Constraint`
The added constraint
"""
if len(args) == 1:
if isinstance(args[0], Constraint):
constraint = args[0]
else:
raise TypeError(
"Either a Constraint object or key-value pairs for the parameters of a ShapeBasedConstraint object need to be passed.")
else:
constraint = ShapeBasedConstraint(**kwargs)
self.call_method("add", object=constraint)
return constraint
[docs]
def remove(self, constraint):
"""
Remove a constraint from the list.
Parameters
----------
constraint : :obj:`espressomd.constraints.Constraint`
"""
self.call_method("remove", object=constraint)
[docs]
def clear(self):
"""
Remove all constraints.
"""
self.call_method("clear")
[docs]
class Constraint(ScriptInterfaceHelper):
"""
Base class for constraints. A constraint provides a force and
an energy contribution for a single particle.
"""
_so_name = "Constraints::Constraint"
def __reduce__(self):
raise RuntimeError(
"Constraints can only be checkpointed through an ESPResSo system's constraints property")
[docs]
@script_interface_register
class ShapeBasedConstraint(Constraint):
"""
Attributes
----------
only_positive : :obj:`bool`
Act only in the direction of positive normal,
only useful if penetrable is ``True``.
particle_type : :obj:`int`
Interaction type of the constraint.
particle_velocity : array_like of :obj:`float`
Interaction velocity of the boundary
penetrable : :obj:`bool`
Whether particles are allowed to penetrate the constraint.
shape : :class:`espressomd.shapes.Shape`
One of the shapes from :mod:`espressomd.shapes`
See Also
----------
espressomd.shapes : shape module that defines mathematical surfaces
Examples
----------
>>> import espressomd
>>> import espressomd.shapes
>>> system = espressomd.System(box_l=3 * [10.])
>>>
>>> # create first a shape-object to define the constraint surface
>>> spherical_cavity = espressomd.shapes.Sphere(center=system.box_l / 2, radius=2.0, direction=-1.0)
>>>
>>> # now create an un-penetrable shape-based constraint of type 0
>>> spherical_constraint = system.constraints.add(particle_type=0, penetrable=False, shape=spherical_cavity)
>>>
>>> # place a trapped particle inside this sphere
>>> system.part.add(pos=0.51 * system.box_l, type=1)
"""
_so_name = "Constraints::ShapeBasedConstraint"
[docs]
def min_dist(self):
"""
Calculates the minimum distance to all interacting particles.
Returns
----------
:obj:`float` :
The minimum distance
"""
return self.call_method("min_dist", object=self)
[docs]
def total_force(self):
"""
Get total force acting on this constraint.
Examples
----------
>>> import espressomd
>>> import espressomd.shapes
>>> system = espressomd.System(box_l=[50., 50., 50.])
>>> system.time_step = 0.01
>>> system.thermostat.set_langevin(kT=0.0, gamma=1.0)
>>> system.cell_system.set_n_square(use_verlet_lists=False)
>>> system.non_bonded_inter[0, 0].lennard_jones.set_params(
... epsilon=1, sigma=1, cutoff=2**(1. / 6), shift="auto")
>>>
>>> floor = system.constraints.add(
... shape=espressomd.shapes.Wall(normal=[0, 0, 1], dist=0.0),
... particle_type=0, penetrable=False, only_positive=False)
>>>
>>> p = system.part.add(pos=[0,0,1.5], type=0, ext_force=[0, 0, -.1])
>>> # print the particle position as it falls
>>> # and print the force it applies on the floor
>>> for t in range(10):
... system.integrator.run(100)
... print(p.pos, floor.total_force())
"""
return self.call_method("total_force", constraint=self)
[docs]
def total_normal_force(self):
"""
Get the total summed normal force acting on this constraint.
"""
return self.call_method("total_normal_force", constraint=self)
[docs]
@script_interface_register
class HomogeneousMagneticField(Constraint):
"""
Homogeneous magnetic field :math:`\\vec{H}`.
The resulting force :math:`\\vec{F}`, torque :math:`\\vec{\\tau}`
and energy `U` on the particles are then
:math:`\\vec{F} = \\vec{0}`
:math:`\\vec{\\tau} = \\vec{\\mu} \\times \\vec{H}`
:math:`U = -\\vec{\\mu} \\cdot \\vec{H}`
where :math:`\\vec{\\mu}` is the particle dipole moment.
Attributes
----------
H : (3,) array_like of :obj:`float`
Magnetic field vector. Describes both field direction and
strength of the magnetic field (via length of the vector).
"""
_so_name = "Constraints::HomogeneousMagneticField"
class _Interpolated(Constraint):
"""
Tabulated field data.
The actual field value is calculated by linear
interpolation (force fields) or gradient linear
interpolation.
The data has to have one point of halo in each direction,
and is shifted by half a grid spacing in the +xyz direction,
so that the element (0,0,0) has coordinates -0.5 * grid_spacing.
The number of points has to be such that the data spans the whole
box, e.g. the most up right back point has to be at least at
box + 0.5 * grid_spacing. There are convenience functions on this
class that can calculate the required grid dimensions and the coordinates.
Arguments
----------
field : (M, N, O, P) array_like of :obj:`float`
The actual field on a grid of size (M, N, O) with dimension P.
grid_spacing : (3,) array_like of :obj:`float`
Spacing of the grid points.
Attributes
----------
field : (M, N, O, P) array_like of :obj:`float`
The actual field on a grid of size (M, N, O) with dimension P.
Please be aware that depending on the interpolation
order additional points are used on the boundaries.
grid_spacing : array_like of :obj:`float`
Spacing of the grid points.
origin : (3,) array_like of :obj:`float`
Coordinates of the grid origin.
"""
def __init__(self, **kwargs):
if "sip" not in kwargs:
field = kwargs.pop("field")
shape, codim = self._unpack_dims(field)
super().__init__(_field_shape=shape, _field_codim=codim,
_field_data=field.flatten(), **kwargs)
else:
super().__init__(**kwargs)
@classmethod
def required_dims(cls, box_size, grid_spacing):
"""
Calculate the grid size and origin needed for specified box size and
grid spacing. Returns the shape and origin (coordinates of [0][0][0])
needed.
Arguments
---------
box_size : (3,) array_like of obj:`float`
The box the field should be used.
grid_spacing : array_like obj:`float`
The desired grid spacing.
"""
shape = np.array(np.ceil(box_size / grid_spacing), dtype=int) + 2
origin = -0.5 * grid_spacing
return shape, origin
@classmethod
def field_from_fn(cls, box_size, grid_spacing, f, codim=None):
"""Generate field data for a desired box size and grid spacing
by evaluating a function at the coordinates.
Arguments
---------
box_size : (3,) array_like of obj:`float`
The box the field should be used.
grid_spacing : array_like obj:`float`
The desired grid spacing.
f : callable
A function that is called with the coordinates of every
grid point to populate the grid.
"""
shape, origin = cls.required_dims(box_size, grid_spacing)
if not codim:
codim = cls._codim
field = np.zeros((shape[0], shape[1], shape[2], codim))
for i in itertools.product(*map(range, shape)):
x = origin + np.array(i) * grid_spacing
field[i] = f(x)
return field
@classmethod
def field_coordinates(cls, box_size, grid_spacing):
"""Returns an array of the coordinates of the grid points required.
Arguments
---------
box_size : (3,) array_like of obj:`float`
The box the field should be used.
grid_spacing : array_like obj:`float`
The desired grid spacing.
"""
return cls.field_from_fn(box_size, grid_spacing, lambda x: x, 3)
def _unpack_dims(self, a):
s = a.shape
shape = s[:3]
codim = s[3]
return (shape, codim)
@property
def field(self):
shape = self._field_shape
return np.reshape(self._field_data,
(shape[0], shape[1], shape[2], self._field_codim))
[docs]
@script_interface_register
class ForceField(_Interpolated):
"""
A generic tabulated force field that applies a per-particle scaling factor.
Arguments
----------
field : (M, N, O, 3) array_like of :obj:`float`
Forcefield amplitude on a grid of size (M, N, O).
grid_spacing : (3,) array_like of :obj:`float`
Spacing of the grid points.
default_scale : :obj:`float`
Scaling factor for particles that have no individual scaling factor.
particle_scales : :obj:`dict`
A dictionary mapping particle ids to scaling factors.
For these particles, the interaction is scaled with
their individual scaling factor. Other particles are
scaled with the default scaling factor.
"""
_codim = 3
_so_name = "Constraints::ForceField"
[docs]
@script_interface_register
class PotentialField(_Interpolated):
"""
A generic tabulated force field that applies a per-particle
scaling factor. The forces are calculated numerically from
the data by finite differences. The potential is interpolated
from the provided data.
Arguments
----------
field : (M, N, O, 1) array_like of :obj:`float`
Potential on a grid of size (M, N, O).
grid_spacing : (3,) array_like of :obj:`float`
Spacing of the grid points.
default_scale : :obj:`float`
Scaling factor for particles that have no individual scaling factor.
particle_scales : :obj:`dict`
A dictionary mapping particle ids to scaling factors.
For these particles, the interaction is scaled with
their individual scaling factor. Other particles are
scaled with the default scaling factor.
"""
_codim = 1
_so_name = "Constraints::PotentialField"
[docs]
@script_interface_register
class Gravity(Constraint):
"""
Gravity force
:math:`\\vec{F} = m \\cdot \\vec{g}`
Arguments
----------
g : (3,) array_like of :obj:`float`
The gravitational constant.
"""
def __init__(self, **kwargs):
if "sip" not in kwargs:
kwargs["value"] = kwargs.pop("g")
super().__init__(**kwargs)
@property
def g(self):
return self.value
_so_name = "Constraints::Gravity"
[docs]
@script_interface_register
class LinearElectricPotential(Constraint):
"""
Electric potential of the form
:math:`\\phi = -\\vec{E} \\cdot \\vec{x} + \\phi_0`,
resulting in the electric field :math:`\\vec{E}` everywhere.
The resulting force on the particles are then
:math:`\\vec{F} = q \\cdot \\vec{E}`
where :math:`q` and :math:`\\vec{x}` are the particle charge and position
in folded coordinates.
This can be used to model a plate capacitor.
Arguments
----------
E : array_like of :obj:`float`
The electric field.
phi0 : :obj:`float`
The potential at the origin
"""
def __init__(self, phi0=0, **kwargs):
if "sip" not in kwargs:
kwargs["A"] = -np.array(kwargs.pop("E"))
kwargs["b"] = phi0
super().__init__(**kwargs)
@property
def E(self):
return -np.array(self.A)
@property
def phi0(self):
return np.array(self.b)
_so_name = "Constraints::LinearElectricPotential"
[docs]
@script_interface_register
class ElectricPlaneWave(Constraint):
"""
Electric field of the form
:math:`\\vec{E} = \\vec{E_0} \\cdot \\sin(\\vec{k} \\cdot \\vec{x} + \\omega \\cdot t + \\phi)`
The resulting force on the particles are then
:math:`\\vec{F} = q \\cdot \\vec{E}`
where :math:`q` and :math:`\\vec{x}` are the particle charge and position
in folded coordinates.
This can be used to generate a homogeneous AC
field by setting :math:`\\vec{k}` to the null vector.
For periodic systems, :math:`\\vec{k}` must be an integer multiple
of :math:`2\\pi \\vec{L}^{-1}` with :math:`\\vec{L}` the box length.
Arguments
----------
E0 : array_like of :obj:`float`
Amplitude of the electric field.
k : array_like of :obj:`float`
Wave vector of the wave
omega : :obj:`float`
Frequency of the wave
phi : :obj:`float`, optional
Phase
"""
_so_name = "Constraints::ElectricPlaneWave"
def __init__(self, phi=0, **kwargs):
if "sip" not in kwargs:
kwargs["amplitude"] = kwargs.pop("E0")
kwargs["wave_vector"] = kwargs.pop("k")
kwargs["frequency"] = kwargs.pop("omega")
kwargs["phase"] = phi
super().__init__(**kwargs)
@property
def E0(self):
return np.array(self.amplitude)
@property
def k(self):
return np.array(self.wave_vector)
@property
def omega(self):
return self.frequency
@property
def phi(self):
return self.phase
[docs]
@script_interface_register
class FlowField(_Interpolated):
"""
Viscous coupling to a flow field that is
interpolated from tabulated data like
:math:`\\vec{F} = -\\gamma \\cdot \\left( \\vec{u}(\\vec{x}) - \\vec{v} \\right)`
where :math:`\\vec{v}` and :math:`\\vec{x}` are the particle velocity and position
in folded coordinates, and :math:`\\vec{u}(\\vec{x})` is a 3D flow field on a grid.
Arguments
----------
field : (M, N, O, 3) array_like of :obj:`float`
Field velocity on a grid of size (M, N, O)
grid_spacing : (3,) array_like of :obj:`float`
Spacing of the grid points.
gamma : :obj:`float`
Coupling constant
"""
_codim = 3
_so_name = "Constraints::FlowField"
[docs]
@script_interface_register
class HomogeneousFlowField(Constraint):
"""
Viscous coupling to a flow field that is
constant in space with the force
:math:`\\vec{F} = -\\gamma \\cdot (\\vec{u} - \\vec{v})`
where :math:`\\vec{v}` is the velocity of the particle
and :math:`\\vec{u}` is the constant flow field.
Attributes
----------
gamma : :obj:`float`
Coupling constant
"""
def __init__(self, **kwargs):
if "sip" not in kwargs:
kwargs["value"] = kwargs.pop("u")
super().__init__(**kwargs)
@property
def u(self):
"""
Field velocity ((3,) array_like of :obj:`float`).
"""
return self.value
_so_name = "Constraints::HomogeneousFlowField"
[docs]
@script_interface_register
class ElectricPotential(_Interpolated):
"""
Electric potential interpolated from
provided data. The electric field :math:`\\vec{E}` is
calculated numerically from the potential,
and the resulting force on the particles are
:math:`\\vec{F} = q \\cdot \\vec{E}`
where :math:`q` is the charge of the particle.
Arguments
----------
field : (M, N, O, 1) array_like of :obj:`float`
Potential on a grid of size (M, N, O)
grid_spacing : (3,) array_like of :obj:`float`
Spacing of the grid points.
"""
_codim = 1
_so_name = "Constraints::ElectricPotential"