# RAD-tools - Sandbox (mainly condense matter plotting).
# Copyright (C) 2022-2024 Andrey Rybakov
#
# e-mail: anry@uv.es, web: rad-tools.org
#
# This program 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.
#
# This program 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 <https://www.gnu.org/licenses/>.
r"""
Spin Hamiltonian module.
Write a tutorial with docstring here.
"""
__all__ = ["SpinHamiltonian", "ExchangeHamiltonian"]
from copy import deepcopy
from typing import Iterable, Tuple
import numpy as np
from radtools.constants import TORADIANS
from radtools.crystal.atom import Atom
from radtools.crystal.crystal import Crystal
from radtools.exceptions import NotationError
from radtools.spinham.constants import PREDEFINED_NOTATIONS
from radtools.spinham.parameter import ExchangeParameter
from radtools.spinham.template import ExchangeTemplate
[docs]
class SpinHamiltonian(Crystal):
r"""
Spin Hamiltonian.
By default the notation of the spin Hamiltonian is not defined
and could be different in different context.
However, in could always be checked via :py:attr:`.notation`.
In user-specific cases it is the responsibility of the user to
set the interpretation of the Hamiltonian`s notation.
For the predefined notations see :py:meth:`.notation`.
Child of the :py:class:`.Crystal` class.
.. versionchanged:: 0.8.0 Renamed from SpinHamiltonian to SpinHamiltonian.
Notes
-----
You can still use the old name, but it is deprecated and will be removed in the future.
Parameters
----------
crystal : :py:class:`.Crystal`, optional
Crystal on which the spin Hamiltonian is build.
By default it is cubic (:math:`a=1`) lattice with no atoms.
notation : str or tuple of two bool and one float, optional
One of the predefined notations or tuple with custom notation.
See :py:attr:`.notation` for details.
**kwargs
Keyword arguments for the :py:class:`.Crystal` constructor.
"""
def __init__(self, crystal: Crystal = None, notation=None, **kwargs) -> None:
if crystal is not None:
kwargs["atoms"] = crystal.atoms
kwargs["cell"] = crystal.cell
super().__init__(**kwargs)
self._bonds = {}
# Notation settings
self._double_counting = None
self._spin_normalized = None
self._factor = None
if notation is not None:
self.notation = notation
def __len__(self):
return self._bonds.__len__()
# Notation attributes
@property
def notation(self):
r"""
Tuple of the notation.
It can be set with a
* string
One of the predefined notations: "standard", "TB2J", "SpinW"
* iterable with 3 elements
First two elements are converted to ``bool``,
third element is interpreted as a float.
Order: (double counting, spin normalized, factor).
See :ref:`library_spinham_notation` for the detailed description.
See Also
--------
double_counting
spin_normalized
factor
set_interpretation
"""
return (self.double_counting, self.spin_normalized, self.factor)
@notation.setter
def notation(self, new_notation):
# Set the notation from predefined notations
if isinstance(new_notation, str):
new_notation = new_notation.lower()
if new_notation not in PREDEFINED_NOTATIONS:
raise ValueError(
f"Predefine notations are: "
+ f"{list(PREDEFINED_NOTATIONS)}, got: {new_notation}"
)
new_notation = PREDEFINED_NOTATIONS[new_notation]
# Set the notation from three values, converted to bool
elif isinstance(new_notation, Iterable) and len(new_notation) == 3:
new_notation = (
bool(new_notation[0]),
bool(new_notation[1]),
float(new_notation[2]),
)
else:
raise ValueError(
"New notation has to be either a string "
+ "or an iterable with three elements, "
+ f"got: {new_notation}"
)
(self.double_counting, self.spin_normalized, self.factor) = new_notation
@property
def notation_string(self):
r"""
Human-readable representation of the notation.
Returns
-------
notation_string : str
Human-readable representation of the notation.
"""
text_result = "H = "
if self.factor != 1:
if self.factor == -1:
text_result += "- "
elif self.factor == int(self.factor):
text_result += f"{self.factor:.0f} "
elif self.factor == 0.5:
text_result += "1/2 "
elif self.factor == -0.5:
text_result += "-1/2 "
else:
text_result += f"{self.factor} "
text_result += "\sum_{"
if self.double_counting:
text_result += "i,j}"
else:
text_result += "i>=j}"
if self.spin_normalized:
text_result += " e_i J_{ij} e_j\n"
else:
text_result += " S_i J_{ij} S_j\n"
if self.double_counting:
text_result += "Double counting is present.\n"
else:
text_result += "No double counting.\n"
if self.spin_normalized:
text_result += "Spin vectors are normalized to 1."
else:
text_result += "Spin vectors are not normalized."
return text_result
[docs]
def set_interpretation(
self, double_counting: bool = None, spin_normalized: bool = None, factor=None
):
r"""
Set the interpretation of the Hamiltonian`s notation.
It is not changing the J values,
but rather indicate how to interpret the J values of the Hamiltonian.
Parameters
----------
double_counting : bool, optional
spin_normalized : bool, optional
factor : float or int, optional
See Also
--------
double_counting
spin_normalized
factor
notation
"""
if double_counting is not None:
self._double_counting = bool(double_counting)
if spin_normalized is not None:
self._spin_normalized = bool(spin_normalized)
if factor is not None:
self._factor = factor
@property
def double_counting(self) -> bool:
r"""
Whether double counting is present in the Hamiltonian.
Access this attribute to check the current notation of the Hamiltonian,
set this attribute to change the notation of the Hamiltonian.
Returns
-------
double_counting : bool
``True`` if double counting is present, ``False`` otherwise.
Raises
------
:py:exc:`.NotationError`
If the interpretation of the Hamiltonian`s notation is not set.
See Also
--------
set_interpretation : to change the interpretation of the Hamiltonian`s notation
without the modification of the parameters.
"""
if self._double_counting is None:
raise NotationError("double_counting")
return self._double_counting
def _ensure_double_counting(self):
r"""
Ensure double counting.
Check and fix if needed, that for each (atom1, atom2, R)
(atom2, atom1, -R) is present in the Hamiltonian.
"""
bonds = list(self)
for atom1, atom2, (i, j, k), J in bonds:
if (atom2, atom1, (-i, -j, -k)) not in self:
self.add_bond(atom2, atom1, (-i, -j, -k), J=J.T)
def _ensure_no_double_counting(self):
r"""
Ensure that there is no double counting in the model.
If the bond is (atom1, atom2, (i,j,k)), then the bond is kept in the model
if one of the conditions is true:
* i > 0
* i = 0 and j > 0
* i = 0, j = 0 and k > 0
* i = 0, j = 0, k = 0 and atom1.index <= atom2.index
"""
bonds = list(self)
for atom1, atom2, (i, j, k), J in bonds:
if (i, j, k) != (0, 0, 0) or atom1 != atom2:
if (
i > 0
or (i == 0 and j > 0)
or (i == 0 and j == 0 and k > 0)
or (i == 0 and j == 0 and k == 0 and atom1.index <= atom2.index)
):
if (atom2, atom1, (-i, -j, -k)) in self:
self.remove_bond(atom2, atom1, (-i, -j, -k))
else:
if (atom2, atom1, (-i, -j, -k)) not in self:
self.add_bond(atom2, atom1, (-i, -j, -k), J=J.T)
self.remove_bond(atom1, atom2, (i, j, k))
@double_counting.setter
def double_counting(self, new_value: bool):
# Need to be at the beginning, otherwise _ensure methods do not work
old_value = self._double_counting
self._double_counting = bool(new_value)
if old_value is not None:
factor = 1
if old_value and not new_value:
factor = 2
elif not old_value and new_value:
factor = 0.5
if factor != 1:
if new_value:
self._ensure_double_counting()
else:
self._ensure_no_double_counting()
for atom1, atom2, R, J in self:
if R != (0, 0, 0) or atom1 != atom2:
self[atom1, atom2, R] = J * factor
else:
if new_value:
self._ensure_double_counting()
else:
self._ensure_no_double_counting()
@property
def spin_normalized(self) -> bool:
r"""
Whether spin is normalized.
Access this attribute to check the current notation of the Hamiltonian,
set this attribute to change the notation of the Hamiltonian.
Returns
-------
spin_normalized : bool
``True`` if spin is normalized, ``False`` otherwise.
Raises
------
:py:exc:`.NotationError`
If the interpretation of the Hamiltonian`s notation is not set.
See Also
--------
set_interpretation : to change the interpretation of the Hamiltonian`s notation
without the modification of the parameters.
"""
if self._spin_normalized is None:
raise NotationError("spin_normalized")
return self._spin_normalized
@spin_normalized.setter
def spin_normalized(self, new_value: bool):
if self._spin_normalized is not None:
if self._spin_normalized and not new_value:
for atom1, atom2, R, J in self:
self[atom1, atom2, R] = J / atom1.spin / atom2.spin
elif not self._spin_normalized and new_value:
for atom1, atom2, R, J in self:
self[atom1, atom2, R] = J * atom1.spin * atom2.spin
self._spin_normalized = bool(new_value)
@property
def factor(self) -> float:
r"""
Whether any factor is present in the Hamiltonian.
Access this attribute to check the current notation of the Hamiltonian,
set this attribute to change the notation of the Hamiltonian.
Returns
-------
factor : float
value of the factor before the sum in the Hamiltonian.
Raises
------
:py:exc:`.NotationError`
If the interpretation of the Hamiltonian`s notation is not set.
See Also
--------
set_interpretation : to change the interpretation of the Hamiltonian`s notation
without the modification of the parameters.
"""
if self._factor is None:
raise NotationError("factor")
return self._factor
@factor.setter
def factor(self, new_factor: bool):
if self._factor is not None:
factor = self._factor / new_factor
if factor != 1:
for atom1, atom2, R, J in self:
self[atom1, atom2, R] = J * factor
self._factor = float(new_factor)
def __iter__(self):
return SpinHamiltonianIterator(self)
def __contains__(self, key):
atom1, atom2, R = key
# If atom is a string, get the atom object
if isinstance(atom1, str):
atom1 = self.get_atom(atom1)
if isinstance(atom2, str):
atom2 = self.get_atom(atom2)
key = (atom1, atom2, R)
return key in self._bonds
def __getitem__(self, key) -> ExchangeParameter:
atom1, atom2, R = key
# If atom is a string, get the atom object
if isinstance(atom1, str):
atom1 = self.get_atom(atom1)
if isinstance(atom2, str):
atom2 = self.get_atom(atom2)
key = (atom1, atom2, R)
return self._bonds[key]
def __getattr__(self, name):
# Fix copy/deepcopy RecursionError
if name in ["__setstate__"]:
raise AttributeError(name)
raise AttributeError(name)
@property
def crystal(self) -> Crystal:
r"""
Crystal of the Hamiltonian.
Return an independent instance of a crystal.
You can use it to play with the model`s crystal independently,
but it will not affect the model itself.
Returns
-------
crystal : :py:class:`.Crystal`
Crystal of the Hamiltonian.
See Also
--------
Crystal
"""
return Crystal(self.lattice, self.atoms)
@property
def cell_list(self):
r"""
List of cells from the Hamiltonian.
Return the list of R for all cell that are present in the Hamiltonian.
Not ordered.
Returns
-------
cells : (n, 3) :numpy:`ndarray`
Array of n unit cells.
"""
cells = set()
for atom1, atom2, R, J in self:
cells.add(R)
return np.array(list(cells))
@property
def magnetic_atoms(self):
r"""
Magnetic atoms of the model.
Atoms with at least one bond starting or finishing in it.
Atoms are ordered with respect to the :py:attr:`.Atom.index`.
Returns
-------
magnetic_atoms : list of :py:class:`.Atom`
List of magnetic atoms.
"""
result = set()
for atom1, atom2, R, J in self:
result.add(atom1)
result.add(atom2)
return sorted(list(result), key=lambda x: x.index)
@property
def number_spins_in_unit_cell(self):
r"""
Number of spins (or magnetic atoms) in the unit cell.
Returns
-------
number_spins_in_unit_cell : int
Number of spins (or magnetic atoms) in the unit cell.
"""
return len(self.magnetic_atoms)
@property
def space_dimensions(self):
r"""
Model minimum and maximum coordinates in real space.
Takes into account only an atom which has at least
one bond associated with it.
Returns
-------
x_min : float
Minimum x coordinate.
y_min : float
Minimum y coordinate.
z_min : float
Minimum z coordinate.
x_max : float
Maximum x coordinate.
y_max : float
Maximum y coordinate.
z_max : float
Maximum z coordinate.
"""
x_max, y_max, z_max = None, None, None
x_min, y_min, z_min = None, None, None
for atom1, atom2, R in self._bonds:
x1, y1, z1 = self.get_atom_coordinates(atom1, relative=False)
x2, y2, z2 = self.get_atom_coordinates(atom2, R, relative=False)
if x_max is None:
x_min, y_min, z_min = x1, y1, z1
x_max, y_max, z_max = x1, y1, z1
x_max = max(x1, x2, x_max)
y_max = max(y1, y2, y_max)
z_max = max(z1, z2, z_max)
x_min = min(x1, x2, x_min)
y_min = min(y1, y2, y_min)
z_min = min(z1, z2, z_min)
return x_min, y_min, z_min, x_max, y_max, z_max
def __setitem__(self, key, value):
self.add_bond(*key, value)
[docs]
def add_bond(
self, atom1: Atom, atom2: Atom, R, J: ExchangeParameter = None, **kwargs
):
r"""
Add one bond to the Hamiltonian.
More straightforward syntax is advised:
.. doctest::
>>> import radtools as rad
>>> Cr = rad.Atom("Cr")
>>> J = rad.ExchangeParameter(iso=1)
>>> model = rad.SpinHamiltonian(rad.Crystal())
>>> model[Cr, Cr, (1,0,0)] = J
>>> (Cr, Cr, (1,0,0)) in model
True
It is equivalent to
.. doctest::
>>> import radtools as rad
>>> Cr = rad.Atom("Cr")
>>> J = rad.ExchangeParameter(iso=1)
>>> model = rad.SpinHamiltonian(rad.Crystal())
>>> model.add_bond(Cr, Cr, (1,0,0), J=J)
>>> (Cr, Cr, (1,0,0)) in model
True
Parameters
----------
atom1 : :py:class:`Atom` or str
Atom object in (0, 0, 0) unit cell.
str works only if atom is already in the Hamiltonian.
atom2 : :py:class:`Atom` or str
Atom object in R unit cell.
str works only if atom is already in the Hamiltonian.
R : tuple of ints
Vector of the unit cell for atom2.
In the relative coordinates (i,j,k).
J : :py:class:`.ExchangeParameter`, optional
An instance of :py:class:`ExchangeParameter`.
** kwargs
Keyword arguments for the constructor of :py:class:`ExchangeParameter`.
Ignored if J is given.
"""
if isinstance(atom1, str):
atom1 = self.get_atom(atom1)
elif atom1 not in self.atoms:
self.add_atom(atom1)
if isinstance(atom2, str):
atom2 = self.get_atom(atom2)
elif atom2 not in self.atoms:
self.add_atom(atom2)
if J is None:
J = ExchangeParameter(**kwargs)
self._bonds[(atom1, atom2, R)] = J
# Check for double counting
double_counting = False
try:
double_counting = self.double_counting
except NotationError:
pass
i, j, k = R
if double_counting and (atom2, atom1, (-i, -j, -k)) not in self:
self._bonds[(atom2, atom1, (-i, -j, -k))] = J.T
def __delitem__(self, key):
self.remove_bond(*key)
[docs]
def remove_bond(self, atom1: Atom, atom2: Atom, R):
r"""
Remove one bond from the Hamiltonian.
More straightforward syntax is advised:
.. doctest::
>>> import radtools as rad
>>> Cr = rad.Atom("Cr")
>>> J = rad.ExchangeParameter(iso=1)
>>> model = rad.SpinHamiltonian(rad.Crystal())
>>> model[Cr, Cr, (1,0,0)] = J
>>> (Cr, Cr, (1,0,0)) in model
True
>>> del model[Cr, Cr, (1,0,0)]
>>> (Cr, Cr, (1,0,0)) in model
False
It is the same as
.. doctest::
>>> import radtools as rad
>>> Cr = rad.Atom("Cr")
>>> J = rad.ExchangeParameter(iso=1)
>>> model = rad.SpinHamiltonian(rad.Crystal())
>>> model[Cr, Cr, (1,0,0)] = J
>>> (Cr, Cr, (1,0,0)) in model
True
>>> model.remove_bond(Cr, Cr, (1,0,0))
>>> (Cr, Cr, (1,0,0)) in model
False
Parameters
----------
atom1 : py:class:`.Atom`
Atom object in (0, 0, 0) unit cell.
atom2 : py:class:`.Atom`
Atom object in R unit cell.
R : tuple of ints
Radius vector of the unit cell for atom2 (i,j,k).
"""
# If atom is a string, get the atom object
if isinstance(atom1, str):
atom1 = self.get_atom(atom1)
if isinstance(atom2, str):
atom2 = self.get_atom(atom2)
try:
del self._bonds[(atom1, atom2, R)]
except KeyError:
raise KeyError(
f"Bond ({atom2.fullname}, {atom2.fullname}, {R}) is not present in the model."
)
# Check for double counting
double_counting = False
try:
double_counting = self.double_counting
except NotationError:
pass
i, j, k = R
if double_counting and (atom2, atom1, (-i, -j, -k)) in self:
del self._bonds[(atom2, atom1, (-i, -j, -k))]
[docs]
def remove_atom(self, atom):
r"""
Remove magnetic atom from the Hamiltonian.
Note: this method will remove atom itself and all the
bonds, which starts or ends in this atom, if atom is magnetic.
Parameters
----------
atom : :py:class:`.Atom`
Atom object.
"""
# If atom given as a string, get the atom object
if isinstance(atom, str):
atom = self.get_atom(atom)
bonds_for_removal = []
for atom1, atom2, R, J in self:
if atom1 == atom or atom2 == atom:
bonds_for_removal.append((atom1, atom2, R))
for bond in bonds_for_removal:
del self[bond]
super().remove_atom(atom)
[docs]
def filter(
self, max_distance=None, min_distance=None, template=None, R_vector=None
):
r"""
Filter the exchange entries based on the given conditions.
The result will be defined by logical conjugate of the conditions.
Saying so the filtering will be performed for each given condition
one by one.
Parameters
----------
max_distance : float or int, optional
Distance for sorting, the condition is <<less or equal>>.
min_distance : float or int, optional
Distance for sorting, the condition is <<more or equal>>.
template : list or :py:class:`.ExchangeTemplate`.
List of pairs, which will remain in the Hamiltonian. ::
[(atom1, atom2, R), ...]
R_vector : tuple of ints or list of tuples of ints
Tuple of 3 integers or list of tuples, specifying the R vectors,
which will be kept after filtering.
See Also
--------
filtered : Returns new object.
Notes
-----
This method modifies the instance at which it is called.
"""
if isinstance(template, ExchangeTemplate):
template = template.get_list()
if template is not None:
template = set(template)
if R_vector is not None:
if type(R_vector) == tuple:
R_vector = {R_vector}
elif type(R_vector) == list:
R_vector = set(R_vector)
else:
raise TypeError("Type is not supported, supported: list.")
bonds_for_removal = set()
for atom1, atom2, R, J in self:
i, j, k = R
dis = self.get_distance(atom1, atom2, R)
if max_distance is not None and dis > max_distance:
bonds_for_removal.add((atom1, atom2, R))
if min_distance is not None and dis < min_distance:
bonds_for_removal.add((atom1, atom2, R))
# This behaviour should depend on the notation of the Hamiltonian
condition = R_vector is not None and R not in R_vector
try:
if self.double_counting:
condition = condition and (-i, -j, -k) not in R_vector
except NotationError:
pass
if condition:
bonds_for_removal.add((atom1, atom2, R))
# Here names, not objects are compared, because in general
# template only has information about names (or fullnames) and R.
if (
template is not None
and (atom1.name, atom2.name, R) not in template
and (atom1.fullname, atom2.fullname, R) not in template
):
bonds_for_removal.add((atom1, atom2, R))
for atom1, atom2, R in bonds_for_removal:
try:
del self[atom1, atom2, R]
except KeyError:
pass
[docs]
def filtered(
self, max_distance=None, min_distance=None, template=None, R_vector=None
):
r"""
Create filtered spin Hamiltonian based on the given conditions.
The result will be defined by logical conjugate of the conditions.
Saying so the filtering will be performed for each given condition
one by one.
Note: this method is not modifying the instance at which it is called.
It will create a new instance with sorted :py:attr:`bonds` and all
the other attributes will be copied (through :py:func:`deepcopy`).
Parameters
----------
max_distance : float or int, optional
Distance for sorting, the condition is <<less or equal>>.
min_distance : float or int, optional
Distance for sorting, the condition is <<more or equal>>.
template : list or :py:class:`.ExchangeTemplate`
List of pairs, which will remain in the Hamiltonian. ::
[(atom1, atom2, R), ...]
R_vector : tuple of ints or list of tuples of ints
Tuple of 3 integers or list of tuples, specifying the R vectors,
which will be kept after filtering.
Returns
-------
filtered_model : :py:class:`.SpinHamiltonian`
Spin Hamiltonian after filtering.
See Also
--------
filter : Modifies current object.
Notes
-----
This method is not modifying the instance at which it is called.
It creates a new instance (through deepcopy).
"""
filtered_model = deepcopy(self)
filtered_model.filter(
max_distance=max_distance,
min_distance=min_distance,
template=template,
R_vector=R_vector,
)
return filtered_model
# TODO rewrite with new template logic (J1 through getattr)
[docs]
def ferromagnetic_energy(self, theta=0, phi=0):
r"""
Compute energy of the Hamiltonian assuming ferromagnetic state.
Notes
-----
Notation of the Hamiltonian has to be known.
See :py:meth:`.set_interpretation` and :py:attr:`.notation`.
Parameters
----------
theta : float or (N,) |array_like|_, default 0
Angle between z axis and direction of the magnetization.
:math:`0^{\circ} \le \theta \le 180^{\circ}`.
phi : float or (N,) |array_like|_, default 0
angle between x axis and projection of
magnetization`s direction on xy plane.
:math:`0^{\circ} \le \phi < 360^{\circ}`.
Returns
-------
energy : float or (N,) :numpy:`ndarray`
Energy of ferromagnetic Hamiltonian with magnetization direction defined
by ``theta`` and ``phi``. In the units of J values.
"""
# Compute spin direction
theta = np.array(theta) * TORADIANS
phi = np.array(phi) * TORADIANS
if theta.shape == ():
theta = np.array([theta])
if phi.shape == ():
phi = np.array([phi])
spin_direction = np.array(
[np.cos(phi) * np.sin(theta), np.sin(phi) * np.sin(theta), np.cos(theta)]
)
energy = np.zeros((3, 3), dtype=float)
for atom1, atom2, R, J in self:
if self.spin_normalized:
energy += self.factor * J.matrix
else:
energy += self.factor * J.matrix * atom1.spin * atom2.spin
energy = np.einsum("ni,ij,jn->n", spin_direction.T, energy, spin_direction)
if len(energy) == 1:
energy = energy[0]
return energy
class ExchangeHamiltonian(SpinHamiltonian):
pass
class SpinHamiltonianIterator:
def __init__(self, exchange_model: SpinHamiltonian) -> None:
self._bonds = list(
map(
lambda x: (x[0], x[1], x[2], exchange_model._bonds[x]),
exchange_model._bonds,
)
)
self._index = 0
def __next__(self) -> Tuple[Atom, Atom, tuple, ExchangeParameter]:
if self._index < len(self._bonds):
result = self._bonds[self._index]
self._index += 1
return result
raise StopIteration
def __iter__(self):
return self