Source code for pymatgen.core.bonds

# coding: utf-8
# Copyright (c) Pymatgen Development Team.
# Distributed under the terms of the MIT License.

from __future__ import division, unicode_literals
import os
import json
import collections
import warnings

from pymatgen.core.periodic_table import get_el_sp

"""
This class implements definitions for various kinds of bonds. Typically used in
Molecule analysis.
"""


__author__ = "Shyue Ping Ong"
__copyright__ = "Copyright 2012, The Materials Project"
__version__ = "0.1"
__maintainer__ = "Shyue Ping Ong"
__email__ = "shyuep@gmail.com"
__date__ = "Jul 26, 2012"


def _load_bond_length_data():
    """Loads bond length data from json file"""
    with open(os.path.join(os.path.dirname(__file__),
                           "bond_lengths.json")) as f:
        data = collections.defaultdict(dict)
        for row in json.load(f):
            els = sorted(row['elements'])
            data[tuple(els)][row['bond_order']] = row['length']
        return data

bond_lengths = _load_bond_length_data()


[docs]class CovalentBond(object): """ Defines a covalent bond between two sites. """ def __init__(self, site1, site2): """ Initializes a covalent bond between two sites. Args: site1 (Site): First site. site2 (Site): Second site. """ self.site1 = site1 self.site2 = site2 @property def length(self): """ Length of the bond. """ return self.site1.distance(self.site2)
[docs] @staticmethod def is_bonded(site1, site2, tol=0.2, bond_order=None): """ Test if two sites are bonded, up to a certain limit. Args: site1 (Site): First site site2 (Site): Second site tol (float): Relative tolerance to test. Basically, the code checks if the distance between the sites is less than (1 + tol) * typical bond distances. Defaults to 0.2, i.e., 20% longer. bond_order: Bond order to test. If None, the code simply checks against all possible bond data. Defaults to None. Returns: Boolean indicating whether two sites are bonded. """ sp1 = list(site1.species_and_occu.keys())[0] sp2 = list(site2.species_and_occu.keys())[0] dist = site1.distance(site2) syms = tuple(sorted([sp1.symbol, sp2.symbol])) if syms in bond_lengths: all_lengths = bond_lengths[syms] if bond_order: return dist < (1 + tol) * all_lengths[bond_order] for v in all_lengths.values(): if dist < (1 + tol) * v: return True return False raise ValueError("No bond data for elements {} - {}".format(*syms))
def __repr__(self): return "Covalent bond between {} and {}".format(self.site1, self.site2) def __str__(self): return self.__repr__()
[docs]def get_bond_length(sp1, sp2, bond_order=1): """ Get the bond length between two species. Args: sp1 (Specie): First specie. sp2 (Specie): Second specie. bond_order: For species with different possible bond orders, this allows one to obtain the bond length for a particular bond order. For example, to get the C=C bond length instead of the C-C bond length, this should be set to 2. Defaults to 1. Returns: Bond length in Angstrom. If no data is available, the sum of the atomic radii is used. """ sp1 = get_el_sp(sp1) sp2 = get_el_sp(sp2) syms = tuple(sorted([sp1.symbol, sp2.symbol])) if syms in bond_lengths: all_lengths = bond_lengths[syms] if bond_order: return all_lengths.get(bond_order) else: return all_lengths.get(1) warnings.warn("No bond lengths for %s-%s found in database. Returning sum" "of atomic radius." % (sp1, sp2)) return sp1.atomic_radius + sp2.atomic_radius