diff --git a/CHANGELOG.md b/CHANGELOG.md index 016a252..e5ba49e 100644 --- a/CHANGELOG.md +++ b/CHANGELOG.md @@ -4,6 +4,14 @@ All notable changes to this project will be documented in this file. The format is based on [Keep a Changelog](https://keepachangelog.com/en/1.0.0/), and this project adheres to [Semantic Versioning](https://semver.org/spec/v2.0.0.html). +## [Unreleased] +### Changed +- vdW radii scaling parameter can now be adjusted via `mindlessgen.toml` or CLI + +### Fixed +- Unit conversion for (currenly unused) vdW radii from the original Fortran project +- minor print output issues (no new line breaks...) + ## [0.4.0] - 2024-09-19 ### Changed - Default file name of `.xyz` file contains prefix `mlm_` diff --git a/mindlessgen.toml b/mindlessgen.toml index 6c17afb..c0e2031 100644 --- a/mindlessgen.toml +++ b/mindlessgen.toml @@ -29,6 +29,8 @@ init_scaling = 3.0 increase_scaling_factor = 1.3 # > Distance threshold for the inital, randomly generated coordinates. Options: dist_threshold = 1.2 +# > Scaling factor for the employed van der Waals radii. Options: +scale_vdw_radii = 1.3333 # > Atom types and their minimum and maximum occurrences. Format: ":-" # > Elements that are not specified are only added by random selection. # > A star sign (*) can be used as a wildcard for integer value. diff --git a/src/mindlessgen/cli/cli_parser.py b/src/mindlessgen/cli/cli_parser.py index 3b5fff3..860935c 100644 --- a/src/mindlessgen/cli/cli_parser.py +++ b/src/mindlessgen/cli/cli_parser.py @@ -86,6 +86,12 @@ def cli_parser(argv: Sequence[str] | None = None) -> dict: required=False, help="Do not write the molecules to xyz files.", ) + parser.add_argument( + "--scale-vdw-radii", + type=float, + required=False, + help="Scaling factor for van der Waals radii.", + ) ### Molecule generation arguments ### parser.add_argument( @@ -266,6 +272,7 @@ def cli_parser(argv: Sequence[str] | None = None) -> dict: "dist_threshold": args_dict["dist_threshold"], "element_composition": args_dict["element_composition"], "forbidden_elements": args_dict["forbidden_elements"], + "scale_vdw_radii": args_dict["scale_vdw_radii"], } # XTB specific arguments rev_args_dict["xtb"] = {"xtb_path": args_dict["xtb_path"]} diff --git a/src/mindlessgen/molecules/molecule.py b/src/mindlessgen/molecules/molecule.py index 7a62e18..e14f418 100644 --- a/src/mindlessgen/molecules/molecule.py +++ b/src/mindlessgen/molecules/molecule.py @@ -185,7 +185,7 @@ def __str__(self) -> str: if self._atlist.size: if not first_line: returnstr += "\n" - returnstr += f"atomic numbers: {self.atlist}" + returnstr += f"atomic numbers: {self.atlist}\n" returnstr += f"sum formula: {self.sum_formula()}" first_line = False if self._xyz.size: diff --git a/src/mindlessgen/molecules/refinement.py b/src/mindlessgen/molecules/refinement.py index 442ef4c..638df9b 100644 --- a/src/mindlessgen/molecules/refinement.py +++ b/src/mindlessgen/molecules/refinement.py @@ -12,6 +12,10 @@ from .miscellaneous import set_random_charge COV_RADII = "pyykko" +BOHR2AA = ( + 0.529177210544 # taken from https://physics.nist.gov/cgi-bin/cuu/Value?bohrrada0 +) +AA2BOHR = 1 / BOHR2AA def iterative_optimization( @@ -44,7 +48,11 @@ def iterative_optimization( print(f"Optimized molecule in cycle {cycle + 1}:\n{rev_mol.xyz}") # Detect fragments from the optimized molecule - fragmols = detect_fragments(rev_mol, verbosity) + fragmols = detect_fragments( + mol=rev_mol, + vdw_scaling=config_generate.scale_vdw_radii, + verbosity=verbosity, + ) # Extract the number of atoms from each fragment for comparison current_atom_counts = [fragmol.num_atoms for fragmol in fragmols] @@ -127,7 +135,9 @@ def iterative_optimization( return rev_mol -def detect_fragments(mol: Molecule, verbosity: int = 1) -> list[Molecule]: +def detect_fragments( + mol: Molecule, vdw_scaling: float, verbosity: int = 1 +) -> list[Molecule]: """ Detects fragments in a molecular system based on atom positions and covalent radii. @@ -152,7 +162,16 @@ def detect_fragments(mol: Molecule, verbosity: int = 1) -> list[Molecule]: sum_radii = get_cov_radii(mol.ati[i], COV_RADII) + get_cov_radii( mol.ati[j], COV_RADII ) - if distance <= sum_radii * 1: + if verbosity > 2: + print(f"Distance between atom {i} and {j}: {distance:6.3f}") + print( + f"Covalent radii of atom {i} and {j}, " + + "and the effective threshold: " + + f"{get_cov_radii(mol.ati[i], COV_RADII):6.3f}, " + + f"{get_cov_radii(mol.ati[j], COV_RADII):6.3f}, " + + f"{(sum_radii * vdw_scaling):6.3f}" + ) + if distance <= sum_radii * vdw_scaling: graph.add_edge(i, j) # Detect connected components (fragments) @@ -196,10 +215,10 @@ def detect_fragments(mol: Molecule, verbosity: int = 1) -> list[Molecule]: def get_cov_radii(at: int, vdw_radii: str = "mlmgen") -> float: """ - D3 covalent radii. + Get the covalent radius of an atom in Angstrom, and scale it by a factor. """ if vdw_radii == "mlmgen": - rcov = [ + rcov = [ # CAUTION: array is given in units of Bohr! 0.80628308, 1.15903197, 3.02356173, @@ -295,11 +314,13 @@ def get_cov_radii(at: int, vdw_radii: str = "mlmgen") -> float: 3.88023730, 3.90543362, ] - return rcov[at] + # multiply the whole array with the BOHR2AA factor to get the radii in Angstrom + rcov = [rad * BOHR2AA for rad in rcov] elif vdw_radii == "pyykko": # Covalent radii (taken from Pyykko and Atsumi, Chem. Eur. J. 15, 2009, 188-197) # Values for metals decreased by 10% - covalent_rad_2009 = [ + # D3 covalent radii used to construct the coordination number + rcov = [ 0.32, 0.46, # H, He 1.20, @@ -419,8 +440,6 @@ def get_cov_radii(at: int, vdw_radii: str = "mlmgen") -> float: 1.48, 1.57, # Nh-Og ] - # D3 covalent radii used to construct the coordination number - covalent_rad_d3 = [4.0 / 3.0 * rad for rad in covalent_rad_2009] - return covalent_rad_d3[at] else: raise ValueError("Invalid vdw_radii argument.") + return rcov[at] diff --git a/src/mindlessgen/prog/config.py b/src/mindlessgen/prog/config.py index 54c51b7..6562729 100644 --- a/src/mindlessgen/prog/config.py +++ b/src/mindlessgen/prog/config.py @@ -179,6 +179,7 @@ def __init__(self: GenerateConfig) -> None: self._increase_scaling_factor: float = 1.3 self._element_composition: dict[int, tuple[int | None, int | None]] = {} self._forbidden_elements: list[int] | None = None + self._scale_vdw_radii: float = 4.0 / 3.0 def get_identifier(self) -> str: return "generate" @@ -364,6 +365,24 @@ def forbidden_elements(self: GenerateConfig, forbidden_str: str) -> None: self._forbidden_elements = sorted(list(forbidden_set)) + @property + def scale_vdw_radii(self): + """ + Get the scaling factor for van der Waals radii. + """ + return self._scale_vdw_radii + + @scale_vdw_radii.setter + def scale_vdw_radii(self, scale_vdw_radii: float): + """ + Set the scaling factor for van der Waals radii. + """ + if not isinstance(scale_vdw_radii, float): + raise TypeError("Scale van der Waals radii should be a float.") + if scale_vdw_radii <= 0: + raise ValueError("Scale van der Waals radii should be greater than 0.") + self._scale_vdw_radii = scale_vdw_radii + class RefineConfig(BaseConfig): """ diff --git a/test/test_molecules/test_refinement.py b/test/test_molecules/test_refinement.py index eff23a5..1471dfd 100644 --- a/test/test_molecules/test_refinement.py +++ b/test/test_molecules/test_refinement.py @@ -106,7 +106,9 @@ def test_detect_fragments_H6O2B2Ne2I1Os1Tl1( """ Test the detection of fragments in the molecule H2O2B2I1Os1. """ - fragmols = detect_fragments(mol_H6O2B2Ne2I1Os1Tl1, verbosity=0) + fragmols = detect_fragments( + mol=mol_H6O2B2Ne2I1Os1Tl1, vdw_scaling=1.3333, verbosity=0 + ) assert len(fragmols) == 7 # check that the first fragment is equal to the molecule H2O2B2I1Os1