Source code for radtools.exchange.hamiltonian

r"""
Exchange module.

Write a tutorial with docstring here.
"""

from copy import deepcopy
from typing import Iterable, Tuple

import numpy as np

from radtools.crystal.atom import Atom
from radtools.crystal.crystal import Crystal
from radtools.exchange.parameter import ExchangeParameter
from radtools.exchange.template import ExchangeTemplate
from radtools.routines import toradians, print_2d_array


class NotationError(ValueError):
    r"""
    Raised when the notation (or individual property) is not defined.

    Gives a summary of the notation (or individual property) and how to set it.

    Parameters
    ----------
    name : str
        Name of the corresponding attribute.

    """

    def __init__(self, name):
        self.message = (
            f"\n\nNotation`s interpretation is not set for the property {name}.\n"
            + f"Set the notation first:\n"
            + f"    ExchangeHamiltonian.{name} = True  "
            + f"or  ExchangeHamiltonian.{name} = False\n\n"
            + f"Note: When the attribute is set for the first time it sets the interpretation, "
            + "afterwards it change the notation.\n\n"
            + f"If you want to set the interpretation again, use \n"
            + f"    ExchangeHamiltonian.set_interpretation({name} = True)"
            + "\nor\n"
            + f"    ExchangeHamiltonian.set_interpretation({name} = False)\n"
        )

    def __str__(self):
        return self.message


[docs] class ExchangeHamiltonian: r""" Exchange Hamiltonian. By default the notation of the exchange 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`: Parameters ---------- crystal : :py:class:`.Crystal`, optional Crystal on which the exchange Hamiltonian is build. By default it is orthonormal lattice (:py:class:`.CUB`, :math:`a = 1`) with no atoms. notation : str or tuple of bool, optional One of the predefined notations or list of 5 bool. See :py:attr:`.notation` for details. Attributes ---------- crystal : :py:class:`.Crystal` Crystal on which the ExchangeHamiltonian is build. """ def __init__(self, crystal: Crystal = None, notation=None) -> None: self._predefined_notations = { "standard": (True, False, False, False, True), "tb2j": (True, True, False, False, True), "spinw": (True, False, False, False, False), } self._crystal = None if crystal is None: crystal = Crystal() self.crystal = crystal self._bonds = {} # Notation settings self._double_counting = None self._spin_normalized = None self._factor_one_half = None self._factor_two = None self._minus_sign = None if notation is not None: self.notation = notation def __len__(self): return self._bonds.__len__() # Notation attributes @property def notation(self): r""" Return a string with a simple comment about the Hamiltonian notation. It can be set with a * string One of the predefined notations: "standard", "TB2J", "SpinW" * iterable with 5 elements Each element is converted to bool and the parameters are set with values. Order: (double counting, spin normalized, factor 1/2, factor 2, minus sign). Predefined notations: * Standard (True, False, False, False, True) .. math:: H = -\sum_{i,j} \hat{\boldsymbol{S}}_i \cdot \boldsymbol{J}_{i,j} \hat{\boldsymbol{S}}_j where double counting is present (:math:`ij` and :math:`ji` is in the sum). Spin vectors are **not** normalized. * |TB2J|_ (True, True, False, False, True) .. math:: H = -\sum_{i,j} \hat{\boldsymbol{S}}_i \cdot \boldsymbol{J}_{i,j} \hat{\boldsymbol{S}}_j where double counting is present (:math:`ij` and :math:`ji` is in the sum). Spin vectors are normalized to 1. * SpinW (True, False, False, False, False) .. math:: H = \sum_{i,j} \hat{\boldsymbol{S}}_i \cdot \boldsymbol{J}_{i,j} \hat{\boldsymbol{S}}_j where double counting is present (:math:`ij` and :math:`ji` is in the sum). Spin vectors are **not** normalized. Setting of this attribute either change the notation (if corresponding attributes are defined) or set the interpretation (if corresponding attribute are not defined). Returns ------- notation : (5,) tuple of bool ``True``/``False`` values of notation properties: .. code-block:: python (double_counting, spin_normalized, factor_one_half, factor_two, minus_sign) Examples -------- Setting one of the predefined notations: .. doctest:: >>> import radtools as rad >>> model = rad.ExchangeHamiltonian() >>> model.notation = "standard" >>> model.notation H = -sum_{i,j} S_i J_ij S_j Double counting is present. Spin vectors are not normalized. (True, False, False, False, True) >>> model.notation = "TB2J" >>> model.notation H = -sum_{i,j} S_i J_ij S_j Double counting is present. Spin vectors are normalized to 1. (True, True, False, False, True) >>> model.notation = "SpinW" >>> model.notation H = sum_{i,j} S_i J_ij S_j Double counting is present. Spin vectors are not normalized. (True, False, False, False, False) Setting the notation: .. doctest:: >>> import radtools as rad >>> model = rad.ExchangeHamiltonian() >>> Cr = rad.Atom("Cr", spin=1.5) >>> model.add_bond(rad.ExchangeParameter(iso=1), Cr, Cr, (1, 0, 0)) >>> model[Cr, Cr, (1, 0, 0)].iso 1.0 >>> # For the first time interpretation is set, >>> # values of exchange are not changed >>> model.notation = "standard" >>> model[Cr, Cr, (1, 0, 0)].iso 1.0 >>> # Once the notation is set the values >>> # are changing if the notation is changed again. >>> model.notation = "TB2J" >>> model[Cr, Cr, (1, 0, 0)].iso 2.25 >>> model.notation = "SpinW" >>> model[Cr, Cr, (1, 0, 0)].iso -1.0 >>> model.notation = "standard" >>> model[Cr, Cr, (1, 0, 0)].iso 1.0 Setting individual properties: .. doctest:: >>> import radtools as rad >>> model = rad.ExchangeHamiltonian() >>> Cr = rad.Atom("Cr", spin=1.5) >>> model.add_bond(rad.ExchangeParameter(iso=1), Cr, Cr, (1, 0, 0)) >>> model[Cr, Cr, (1, 0, 0)].iso 1.0 >>> # For the first time interpretation is set, >>> # values of exchange are not changed >>> model.minus_sign = True >>> model[Cr, Cr, (1, 0, 0)].iso 1.0 >>> # Once the property is set the values >>> # are changing if the property is changed again. >>> model.minus_sign = False >>> model[Cr, Cr, (1, 0, 0)].iso -1.0 Changing individual properties: .. doctest:: >>> import radtools as rad >>> model = rad.ExchangeHamiltonian() >>> Cr = rad.Atom("Cr", spin=1.5) >>> model.add_bond(rad.ExchangeParameter(iso=1), Cr, Cr, (1, 0, 0)) >>> model[Cr, Cr, (1, 0, 0)].iso 1.0 >>> model.notation = "standard" >>> model[Cr, Cr, (1, 0, 0)].iso 1.0 >>> model.double_counting, model.spin_normalized, model.factor_one_half, model.factor_two, model.minus_sign (True, False, False, False, True) >>> model.minus_sign = False >>> model[Cr, Cr, (1, 0, 0)].iso -1.0 >>> model.factor_one_half, model.factor_two (False, False) >>> model.factor_one_half = True >>> model[Cr, Cr, (1, 0, 0)].iso -2.0 >>> model.factor_one_half, model.factor_two (True, False) >>> model.factor_two = True >>> model[Cr, Cr, (1, 0, 0)].iso -1.0 >>> # Note that the values are switched to False, >>> # since factor one half and two are cancelling each other >>> model.factor_one_half, model.factor_two (False, False) >>> model.spin_normalized = True >>> model[Cr, Cr, (1, 0, 0)].iso -2.25 >>> model.double_counting = False >>> model[Cr, Cr, (1, 0, 0)].iso -4.5 See Also -------- double_counting spin_normalized factor_one_half factor_two minus_sign set_interpretation """ text_result = "H = " if self.minus_sign: text_result += "-" if self.factor_one_half and not self._factor_two: text_result += "1/2 " if self._factor_two and not self.factor_one_half: text_result += "2 " text_result += "sum_{" if self.double_counting: text_result += "i,j} " else: text_result += "i>=j} " 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." print(text_result) return ( self.double_counting, self.spin_normalized, self.factor_one_half, self.factor_two, self.minus_sign, ) @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 self._predefined_notations: raise ValueError( f"Predefine notations are: " + f"{list(self._predefined_notations)}, got: {new_notation}" ) new_notation = self._predefined_notations[new_notation] # Set the notation from five values, converted to bool elif isinstance(new_notation, Iterable) and len(new_notation) == 5: new_notation = tuple(map(bool, new_notation)) else: raise ValueError( "New notation has to be either a string " + "or an iterable with five elements, " + f"got: {new_notation}" ) ( self.double_counting, self.spin_normalized, self.factor_one_half, self.factor_two, self.minus_sign, ) = new_notation
[docs] def set_interpretation( self, double_counting: bool = None, spin_normalized: bool = None, factor_one_half: bool = None, factor_two: bool = None, minus_sign: bool = 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_one_half : bool, optional factor_two : bool, optional minus_sign : bool, optional See Also -------- double_counting spin_normalized factor_one_half factor_two minus_sign """ 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_one_half is not None: self._factor_one_half = bool(factor_one_half) if factor_two is not None: self._factor_two = bool(factor_two) if minus_sign is not None: self._minus_sign = bool(minus_sign)
@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(J.T, atom2, atom1, (-i, -j, -k)) 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 > 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(J.T, atom2, atom1, (-i, -j, -k)) self.remove_bond(atom1, atom2, (i, j, k)) @double_counting.setter def double_counting(self, new_value: bool): if self._double_counting is not None: factor = 1 if self._double_counting and not new_value: factor = 2 elif not self._double_counting 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: self[atom1, atom2, R] = J * factor else: if new_value: self._ensure_double_counting() else: self._ensure_no_double_counting() self._double_counting = bool(new_value) @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_one_half(self) -> bool: r""" Whether factor 1/2 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_one_half : bool ``True`` if factor 1/2 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._factor_one_half is None: raise NotationError("factor_one_half") return self._factor_one_half @factor_one_half.setter def factor_one_half(self, new_value: bool): if self._factor_one_half is not None: factor = 1 if self._factor_one_half and not new_value: factor = 0.5 elif not self._factor_one_half and new_value: factor = 2 if factor != 1: for atom1, atom2, R, J in self: self[atom1, atom2, R] = J * factor self._factor_one_half = bool(new_value) if self._factor_one_half and self.factor_two: self._factor_one_half = False self._factor_two = False @property def factor_two(self) -> bool: r""" Whether factor 2 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_two : bool ``True`` if factor 2 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._factor_two is None: raise NotationError("factor_two") return self._factor_two @factor_two.setter def factor_two(self, new_value: bool): if self._factor_two is not None: factor = 1 if self._factor_two and not new_value: factor = 2 elif not self._factor_two and new_value: factor = 0.5 if factor != 1: for atom1, atom2, R, J in self: self[atom1, atom2, R] = J * factor self._factor_two = bool(new_value) if self._factor_one_half and self.factor_two: self._factor_one_half = False self._factor_two = False @property def minus_sign(self) -> bool: r""" Whether the minus sign 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 ------- minus_sign : bool ``True`` if minus sign 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._minus_sign is None: raise NotationError("minus_sign") return self._minus_sign @minus_sign.setter def minus_sign(self, new_value: bool): if self._minus_sign is not None: factor = 1 if self._minus_sign ^ new_value: factor = -1 if factor != 1: for atom1, atom2, R, J in self: self[atom1, atom2, R] = J * factor self._minus_sign = bool(new_value) def __iter__(self): return ExchangeHamiltonianIterator(self) def __contains__(self, key): atom1, atom2, R = key # If atom is a string, get the atom object if isinstance(atom1, str): atom1 = self.crystal.get_atom(atom1) if isinstance(atom2, str): atom2 = self.crystal.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.crystal.get_atom(atom1) if isinstance(atom2, str): atom2 = self.crystal.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) return getattr(self.crystal, name) @property def crystal(self) -> Crystal: r""" Crystal of the Hamiltonian. Crystal, which define the structure. See :py:class:`.Crystal`. Returns ------- crystal : :py:class:`.Crystal` Crystal of the Hamiltonian. """ return self._crystal @crystal.setter def crystal(self, new_crystal: Crystal): if not isinstance(new_crystal, Crystal): raise TypeError( "New crystal is not a Crystal. " + f"Received {type(new_crystal)}." ) self._crystal = new_crystal @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 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, J in self: x1, y1, z1 = self.crystal.get_atom_coordinates(atom1) x2, y2, z2 = self.crystal.get_atom_coordinates(atom2, R) 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(value, *key)
[docs] def add_bond(self, J: ExchangeParameter, atom1: Atom, atom2: Atom, R): 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.ExchangeHamiltonian(rad.Crystal()) >>> model[Cr, Cr, (1,0,0)] = J >>> (Cr, Cr, (1,0,0)) in model True It is the same as .. doctest:: >>> import radtools as rad >>> Cr = rad.Atom("Cr") >>> J = rad.ExchangeParameter(iso=1) >>> model = rad.ExchangeHamiltonian(rad.Crystal()) >>> model.add_bond(J, Cr, Cr, (1,0,0)) >>> (Cr, Cr, (1,0,0)) in model True Parameters ---------- J : :py:class:`.ExchangeParameter` An instance of :py:class:`ExchangeParameter`. 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). """ if isinstance(atom1, str): atom1 = self.crystal.get_atom(atom1) elif atom1 not in self.crystal: self.add_atom(atom1) if isinstance(atom2, str): atom2 = self.crystal.get_atom(atom2) elif atom2 not in self.crystal: self.add_atom(atom2) self._bonds[(atom1, atom2, R)] = J
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.ExchangeHamiltonian(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.ExchangeHamiltonian(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.crystal.get_atom(atom1) if isinstance(atom2, str): atom2 = self.crystal.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." )
[docs] def add_atom(self, atom: Atom): r""" Add atom to the Hamiltonian. see :py:meth:`.Crystal.add_atom` Parameters ---------- atom : :py:class:`.Atom` Atom object. """ self.crystal.add_atom(atom)
[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.crystal.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] self.crystal.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: dis = self.crystal.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)) if R_vector is not None and R not in R_vector: 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: del self[atom1, atom2, R]
[docs] def filtered( self, max_distance=None, min_distance=None, template=None, R_vector=None ): r""" Create filtered exchange 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:`.ExchangeHamiltonian` Exchange 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 force_symmetry(self, template): r""" Force the Hamiltonian to have the symmetries of the template. Takes mean values of the parameters. For DMI interaction directions are kept the same, but the length of the DMI vector is scaled. This method modifies an instance on which it was called. Parameters ---------- template : :py:class:`.ExchangeTemplate` Template. See Also -------- forced_symmetry : Returns new object. """ if not isinstance(template, ExchangeTemplate): raise TypeError self.filter(template=template) for name in template.names: bonds = template.names[name] symm_matrix = np.zeros((3, 3), dtype=float) dmi_module = 0 for atom1, atom2, R in bonds: # Here names, not objects are obtained, because in general # template only has information about names and R. J = self[self.crystal.get_atom(atom1), self.crystal.get_atom(atom2), R] symm_matrix = symm_matrix + J.symm_matrix dmi_module = dmi_module + J.dmi_module symm_matrix = symm_matrix / len(bonds) dmi_module = dmi_module / len(bonds) for atom1, atom2, R in bonds: J = self[self.crystal.get_atom(atom1), self.crystal.get_atom(atom2), R] if J.dmi_module != 0: asymm_factor = dmi_module / J.dmi_module else: asymm_factor = 0 J.matrix = symm_matrix + J.asymm_matrix * asymm_factor
[docs] def forced_symmetry(self, template): r""" Force the Hamiltonian to have the symmetries of the template. Takes mean values of the parameters. Respect the direction of the DMI vectors. This method return a new instance. Parameters ---------- template : :py:class:`.ExchangeTemplate` Template. Returns ------- new_model : :py:class:`.ExchangeHamiltonian` Exchange Hamiltonian with forced symmetry. See Also -------- force_symmetry: Modifies current object. """ new_model = deepcopy(self) new_model.force_symmetry(template=template) return new_model
[docs] def summary_as_txt( self, template: ExchangeTemplate, decimals=4, force_symmetry=False, isotropic=False, anisotropic=False, matrix=False, dmi=False, ): r""" Return exchange Hamiltonian based on the template file in .txt format. Parameters ---------- template : :py:class:`.ExchangeTemplate` Template of the desired exchange Hamiltonian. (see :ref:`rad-make-template`) accuracy : int, default 4 Accuracy for the exchange values force_symmetry: bool, default False Whether to force the symmetry of the template on the exchange Hamiltonian. If ``False`` then each individual bond is written, otherwise exchange parameters of the template are written. isotropic : bool, default False Whether to output isotropic exchange. anisotropic : bool, default False Whether to output anisotropic exchange. matrix : bool, default False Whether to output whole matrix exchange. dmi : bool, default False Whether to output DMI exchange. Returns ------- summary : str Exchange information as a string. """ if force_symmetry: self.force_symmetry(template=template) else: self.filter(template=template) summary = "" for name in template.names: scalar_written = False for atom1, atom2, R in template.names[name]: # Get values from the model atom1 = self.crystal.get_atom(atom1) atom2 = self.crystal.get_atom(atom2) J = self[atom1, atom2, R] if not force_symmetry: # Write the values summary += ( f"{atom1:3} {atom2:3} " + f"({R[0]:2.0f}, {R[1]:2.0f}, {R[2]:2.0f})\n" ) if isotropic: summary += f" Isotropic: {J.iso:.{decimals}f}\n" if anisotropic: summary += ( f" Anisotropic:\n" + print_2d_array( J.aniso, fmt=f"{decimals+3}.{decimals}f", borders=False, shift=7, print_result=False, ) + "\n" ) if matrix: summary += ( f" Matrix:\n" + print_2d_array( J.matrix, fmt=f"{decimals+3}.{decimals}f", borders=False, shift=7, print_result=False, ) + "\n" ) if dmi: summary += ( f" |DMI|: {J.dmi_module:.{decimals}f}\n" + f" |DMI/J|: {J.rel_dmi:.{decimals}f}\n" + f" DMI: {J.dmi[0]:.{decimals}f} " + f"{J.dmi[1]:.{decimals}f} " + f"{J.dmi[2]:.{decimals}f}\n" ) summary += "\n" else: if not scalar_written: scalar_written = True summary += f"{name}\n" if isotropic: summary += f" Isotropic: {J.iso:.{decimals}f}\n" if anisotropic: summary += ( f" Anisotropic:\n" + print_2d_array( J.aniso, fmt=f"{decimals+3}.{decimals}f", borders=False, shift=7, print_result=False, ) + "\n" ) if matrix: summary += ( f" Matrix:\n" + print_2d_array( J.matrix, fmt=f"{decimals+3}.{decimals}f", borders=False, shift=7, print_result=False, ) + "\n" ) if dmi: summary += ( f" |DMI|: {J.dmi_module:.{decimals}f}\n" + f" |DMI/J|: {J.rel_dmi:.{decimals}f}\n" ) if dmi: summary += ( f" DMI: {J.dmi[0]:{decimals+3}.{decimals}f} " + f"{J.dmi[1]:{decimals+3}.{decimals}f} " + f"{J.dmi[2]:{decimals+3}.{decimals}f} " + f"({atom1:3} {atom2:3} " + f"{R[0]:2.0f} {R[1]:2.0f} {R[2]:2.0f})\n" ) if force_symmetry: summary += "\n" return summary
[docs] def ferromagnetic_energy(self, theta=0, phi=0): r""" Compute energy of the Hamiltonian assuming ferromagnetic state. Notes ----- Notation of the Hamiltonian nas 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)] ) # compute factor from notation factor = 1 if self.minus_sign: factor *= -1 if self.factor_one_half: factor /= 2 if self.factor_two: factor *= 2 energy = np.zeros((3, 3), dtype=float) for atom1, atom2, R, J in self: if self.spin_normalized: energy += factor * J.matrix else: energy += 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
[docs] def input_for_magnons(self): r""" Input from Exchange model. This function prepare the list of exchange parameters to be used as an input for magnon dispersion calculation. Returns ------- Jij : list i : list j : list dij : list """ Jij = [] i = [] j = [] dij = [] magnetic_atoms = self.magnetic_atoms atom_index = dict([(atom, i) for i, atom in enumerate(magnetic_atoms)]) for atom1, atom2, R, J in self: Jij.append(J.matrix) i.append(atom_index[atom1]) j.append(atom_index[atom2]) dij.append(self.crystal.get_vector(atom1, atom1, R)) return Jij, i, j, dij
# OLD METHODS AND ATTRIBUTES, KEPT FOR BACKWARD COMPATIBILITY # TODO, get rid of this cell @property def cell(self): r""" Matrix of lattice vectors. See :py:attr:`.Crystal.cell`. """ return self.crystal.lattice.cell @cell.setter def cell(self, new_cell): self.crystal.cell = new_cell
[docs] def get_atom_coordinates(self, atom, R=(0, 0, 0)): r""" Getter for the atom coordinates. See :py:meth:`.Crystal.get_atom_coordinates`. """ return self.crystal.get_atom_coordinates(atom, R)
[docs] def get_bond_vector(self, atom1, atom2, R=(0, 0, 0)): r""" Getter for distance between the atom1 and atom2. See :py:meth:`.Crystal.get_vector` """ return self.crystal.get_vector(atom1, atom2, R)
[docs] def get_distance(self, atom1, atom2, R=(0, 0, 0)): r""" Getter for distance between the atom1 and atom2. See :py:meth:`.Crystal.get_distance` """ return self.crystal.get_distance(atom1, atom2, R)
class ExchangeHamiltonianIterator: def __init__(self, exchange_model: ExchangeHamiltonian) -> 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 if __name__ == "__main__": model = ExchangeHamiltonian() Cr = Atom("Cr", (0, 0, 0), spin=3 / 2) model[Cr, Cr, (1, 0, 0)] = ExchangeParameter(iso=1) assert model[Cr, Cr, (1, 0, 0)].iso == 1 model.notation = "standard" assert model[Cr, Cr, (1, 0, 0)].iso == 1 model.notation = "standard" assert model[Cr, Cr, (1, 0, 0)].iso == 1 assert model.double_counting model.double_counting = False assert model[Cr, Cr, (1, 0, 0)].iso == 2 model.double_counting = True assert model[Cr, Cr, (1, 0, 0)].iso == 1