Source code for magnopy.io._tb2j

# MAGNOPY - Python package for magnons.
# Copyright (C) 2023-2025 Magnopy Team
#
# e-mail: anry@uv.es, web: magnopy.com
#
# 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/>.


import numpy as np
from wulfric.crystal import get_distance
from wulfric.geometry import absolute_to_relative

from magnopy._parameters._p22 import from_dmi, from_iso
from magnopy._spinham._convention import Convention
from magnopy._spinham._hamiltonian import SpinHamiltonian

# Save local scope at this moment
old_dir = set(dir())
old_dir.add("old_dir")


[docs] def load_tb2j( filename, spin_values=None, g_factors=None, quiet=True ) -> SpinHamiltonian: r""" Read spin Hamiltonian from the output of |TB2J|_. Parameters ---------- filename : str Path to the |TB2J|_ output file. spin_values : (M, ) iterable of floats, optional Spin values for each atom. In the same order as in TB2J file. TB2J outputs magnetic moments and not spin values, therefore the spin values for each atom are subject to user definition. If user does not define the spin values, then magnopy set spin value as :math:`|\boldsymbol{m} / g_{factor}|` for each spin. g_factors : (M, ) iterable of floats, optional g-factors for each atom. In the same order as in TB2J file. TB2J does not provide g-factors, therefore g-factor for each atom is subject to user definition. If user does not define g-factors, then magnopy sets g-factor of each atom to :math:`2`. quiet : bool, default True Whether to suppress output. Returns ------- spinham : :py:class:`.SpinHamiltonian` Spin Hamiltonian build from |TB2J|_ file. Convention is set to the one of TB2J. Notes ----- Distances between atoms are not read from the atoms, but rather computed from the unit cell and atom positions. If the distance read from the file is different from the computed one and ``quiet=False``, the warning is printed. It is usual for the computed and read distances to differ in the last digits, since the unit cell is provided with the same precision as the distance in |TB2J|_ output file. """ major_sep = "=" * 90 minor_sep = "-" * 88 garbage = str.maketrans( {"(": None, ")": None, "[": None, "]": None, ",": None, "'": None} ) # Do not correct spelling, it is taken from TB2J. cell_flag = "Cell (Angstrom):" atoms_flag = "Atoms:" atom_end_flag = "Total" exchange_flag = "Exchange:" iso_flag = "J_iso:" aniso_flag = "J_ani:" dmi_flag = "DMI:" file = open(filename, "r") # model = SpinHamiltonian(convention="TB2J") line = True # Read everything before exchange while line: line = file.readline() # Read cell if line and cell_flag in line: a1 = file.readline().split() a2 = file.readline().split() a3 = file.readline().split() cell = np.array( [ list(map(float, a1)), list(map(float, a2)), list(map(float, a3)), ] ) # Read atoms if line and atoms_flag in line: atoms = dict(names=[], positions=[], magnetic_moments=[], charges=[]) line = file.readline() line = file.readline() line = file.readline().split() i = 0 while line and atom_end_flag not in line: try: # Slicing is not used intentionally. magmom = tuple(map(float, [line[5], line[6], line[7]])) except IndexError: magmom = float(line[5]) try: charge = float(line[4]) except IndexError: charge = None position = absolute_to_relative( basis=cell, vector=np.array(tuple(map(float, line[1:4]))) ) atoms["names"].append(line[0]) atoms["positions"].append(position) atoms["magnetic_moments"].append(magmom) atoms["charges"].append(charge) line = file.readline().split() i += 1 # Check if the exchange section is reached if line and exchange_flag in line: break # Populate g_factors of atoms if g_factors is None: g_factors = [2.0 for _ in range(len(atoms["names"]))] atoms["g_factors"] = g_factors # Create a spin Hamiltonian spinham = SpinHamiltonian( cell=cell, atoms=atoms, convention=Convention.get_predefined(name="tb2j") ) # Prepare index mapping for atom names index_mapping = {} # Names of the atoms are unique in the TB2J files for index, name in enumerate(spinham.atoms.names): index_mapping[name] = index # Read exchange (22) parameters while line: while line and minor_sep not in line: line = file.readline() line = file.readline().translate(garbage).split() atom1 = index_mapping[line[0]] atom2 = index_mapping[line[1]] ijk = tuple(map(int, line[2:5])) distance = float(line[-1]) iso = None aniso = None dmi = None while line and minor_sep not in line: line = file.readline() # Read isotropic exchange if line and iso_flag in line: iso = float(line.split()[-1]) # Read anisotropic exchange if line and aniso_flag in line: aniso = np.array( [ list(map(float, file.readline().translate(garbage).split())), list(map(float, file.readline().translate(garbage).split())), list(map(float, file.readline().translate(garbage).split())), ] ) # Read DMI if line and dmi_flag in line: dmi = tuple(map(float, line.translate(garbage).split()[-3:])) parameter = np.zeros((3, 3), dtype=float) if iso is not None: parameter = parameter + from_iso(iso=iso) if dmi is not None: parameter = parameter + from_dmi(dmi=dmi) if aniso is not None: parameter = parameter + aniso # Adding info from the exchange block to the SpinHamiltonian structure spinham.add_22( alpha=atom1, beta=atom2, nu=ijk, # Avoid passing aniso to the function as then the function make it traceless # and symmetric, potentially loosing part of the matrix. # Due to the TB2J problem: aniso not always traceless. parameter=parameter, replace=True, ) computed_distance = get_distance(spinham.cell, spinham.atoms, atom1, atom2, ijk) if abs(computed_distance - distance) > 0.001 and not quiet: print( f"\nComputed distance is a different from the read one:\n" + f" Computed: {computed_distance:.4f}\n " + f"Read: {distance:.4f}\n" ) # Populate spin_values of atoms if spin_values is not None: if len(spin_values) != spinham.M: raise ValueError( f"Expected {spinham.M} spin values, got {len(spin_values)}" ) true_spin_values = [0.0 for _ in spinham.atoms.names] for i in range(len(spin_values)): true_spin_values[spinham.map_to_all[i]] = spin_values[i] else: true_spin_values = [ float(np.linalg.norm(atoms["magnetic_moments"][alpha])) / atoms["g_factors"][alpha] for alpha in range(len(atoms["names"])) ] spinham.atoms["spins"] = true_spin_values spinham._reset_internals() return spinham
# Populate __all__ with objects defined in this file __all__ = list(set(dir()) - old_dir) # Remove all semi-private objects __all__ = [i for i in __all__ if not i.startswith("_")] del old_dir