Source code for stk._internal.functional_group_factories.smarts

"""
SMARTS Functional Group Factory
===============================

"""

from stk._internal.functional_group_factories.functional_group_factory import (
    FunctionalGroupFactory,
)
from stk._internal.functional_group_factories.utilities import get_atom_ids
from stk._internal.functional_groups.generic_functional_group import (
    GenericFunctionalGroup,
)


[docs] class SmartsFunctionalGroupFactory(FunctionalGroupFactory): """ Creates :class:`.GenericFunctionalGroup` instances. Examples -------- *Using SMARTS to Define Functional Groups* You want to create a building block which has :class:`.GenericFunctionalGroup` functional groups based on the SMARTS string: ``[Br][C]``. You want the ``C`` atom to be the *bonder* atom, and the ``Br`` atom to be the *deleter* atom. .. testcode:: using-smarts-to-define-functional-groups import stk building_block = stk.BuildingBlock( smiles='BrCCCBr', functional_groups=( stk.SmartsFunctionalGroupFactory( smarts='[Br][C]', bonders=(1, ), deleters=(0, ), ), ), ) .. testcode:: using-smarts-to-define-functional-groups :hide: fg1, fg2 = building_block.get_functional_groups() assert fg1.get_num_bonders() == 1 assert sum(1 for _ in fg1.get_deleters()) == 1 assert fg2.get_num_bonders() == 1 assert sum(1 for _ in fg2.get_deleters()) == 1 assert all( isinstance(atom, stk.C) for functional_group in building_block.get_functional_groups() for atom in functional_group.get_bonders() ) assert all( isinstance(atom, stk.Br) for functional_group in building_block.get_functional_groups() for atom in functional_group.get_deleters() ) See Also -------- :class:`.GenericFunctionalGroup` Defines *bonders* and *deleters*. """ def __init__(self, smarts, bonders, deleters, placers=None): """ Initialize a :class:`.SmartsFunctionalGroupFactory` instance. Parameters ---------- smarts : :class:`str` The SMARTS defining the functional group. bonders : :class:`tuple` of :class:`int` The indices of atoms in `smarts`, which are *bonder* atoms. deleters : :class:`tuple` of :class:`int` The indices of atoms in `smarts`, which are *deleter* atoms. placers : :class:`tuple` of :class:`int`, optional The indices of atoms in `smarts`, which are *placer* atoms. If ``None``, the *bonder* atoms will be used. """ self._smarts = smarts self._bonders = bonders self._deleters = deleters self._placers = bonders if placers is None else placers
[docs] def get_functional_groups(self, molecule): for atom_ids in get_atom_ids(self._smarts, molecule): atoms = tuple(molecule.get_atoms(atom_ids)) yield GenericFunctionalGroup( atoms=atoms, bonders=tuple(atoms[i] for i in self._bonders), deleters=tuple(atoms[i] for i in self._deleters), placers=tuple(atoms[i] for i in self._placers), )