from __future__ import annotations
import logging
import scitbx.matrix
from cctbx import crystal, sgtbx
from cctbx.crystal_orientation import crystal_orientation
from cctbx.sgtbx import change_of_basis_op, subgroups
from cctbx.sgtbx.bravais_types import bravais_lattice
from dxtbx.model import Crystal
from rstbx.dps_core.lepage import iotbx_converter
from scitbx.array_family import flex
logger = logging.getLogger(__name__)
[docs]
def metric_supergroup(group):
return (
sgtbx.space_group_info(group=group)
.type()
.expand_addl_generators_of_euclidean_normalizer(True, True)
.build_derived_acentric_group()
)
[docs]
def groups_cache(fn):
class MultiClassCache:
"A set of caches for different bravais types"
instances = {}
def __new__(cls, classname):
if classname not in cls.instances:
cls.instances[classname] = {}
return cls.instances[classname]
def wrapped_calc(group_info: sgtbx.space_group_info, bravais_t: str):
cache = MultiClassCache(bravais_t)
info_str = group_info.type().lookup_symbol()
try:
result = cache[info_str]
except KeyError:
result = fn(group_info, bravais_t)
cache[info_str] = result
return result
return wrapped_calc
[docs]
@groups_cache
def calc_acentric_subgroups(
lattice_group_info: sgtbx.space_group_info, target_bravais_str: str
):
# Get list of sub-spacegroups
subgrs = subgroups.subgroups(lattice_group_info).groups_parent_setting()
# Order sub-groups
sort_values = flex.double()
for group in subgrs:
order_z = group.order_z()
space_group_number = sgtbx.space_group_type(group, False).number()
assert 1 <= space_group_number <= 230
sort_values.append(order_z * 1000 + space_group_number)
perm = flex.sort_permutation(sort_values, reverse=True)
acentric_subgroups = []
acentric_supergroups = []
cb_ops = []
for i_perm in perm:
acentric_subgroup = subgrs[i_perm]
cb_op = acentric_subgroup.info().type().cb_op()
ref_acentric_subgroup = acentric_subgroup.change_basis(cb_op)
# Ignore unwanted groups
if str(bravais_lattice(group=ref_acentric_subgroup)) != target_bravais_str:
continue
acentric_subgroups.append(acentric_subgroup)
cb_ops.append(cb_op)
acentric_supergroups.append(metric_supergroup(acentric_subgroup))
return acentric_subgroups, acentric_supergroups, cb_ops
[docs]
def find_matching_symmetry(
unit_cell,
target_space_group,
max_delta=5,
best_monoclinic_beta=True,
target_bravais_str=None,
):
cs = crystal.symmetry(unit_cell=unit_cell, space_group=sgtbx.space_group())
if target_bravais_str is None:
target_bravais_str = str(
bravais_lattice(group=target_space_group.info().reference_setting().group())
)
best_subgroup = None
best_angular_difference = 1e8
# code based on cctbx/sgtbx/lattice_symmetry.py but optimised to only
# look at subgroups with the correct bravais type
input_symmetry = cs
# Get cell reduction operator
cb_op_inp_minimum = input_symmetry.change_of_basis_op_to_minimum_cell()
# New symmetry object with changed basis
minimum_symmetry = input_symmetry.change_basis(cb_op_inp_minimum)
# Get highest symmetry compatible with lattice
lattice_group = sgtbx.lattice_symmetry_group(
minimum_symmetry.unit_cell(),
max_delta=max_delta,
enforce_max_delta_for_generated_two_folds=True,
)
lattice_group_info = lattice_group.info()
acentric_subgroups, acentric_supergroups, cb_ops = calc_acentric_subgroups(
lattice_group_info,
target_bravais_str,
)
for acentric_subgroup, acentric_supergroup, cb_op_minimum_ref in zip(
acentric_subgroups, acentric_supergroups, cb_ops
):
# Make symmetry object: unit-cell + space-group
# The unit cell is potentially modified to be exactly compatible
# with the space group symmetry.
subsym = crystal.symmetry(
unit_cell=minimum_symmetry.unit_cell(),
space_group=acentric_subgroup,
assert_is_compatible_unit_cell=False,
)
# Convert subgroup to reference setting
ref_subsym = subsym.change_basis(cb_op_minimum_ref)
# Choose best setting for monoclinic and orthorhombic systems
cb_op_best_cell = ref_subsym.change_of_basis_op_to_best_cell(
best_monoclinic_beta=best_monoclinic_beta
)
best_subsym = ref_subsym.change_basis(cb_op_best_cell)
# Total basis transformation
cb_op_best_cell = change_of_basis_op(
str(cb_op_best_cell), stop_chars="", r_den=144, t_den=144
)
cb_op_minimum_ref = change_of_basis_op(
str(cb_op_minimum_ref), stop_chars="", r_den=144, t_den=144
)
cb_op_inp_minimum = change_of_basis_op(
str(cb_op_inp_minimum), stop_chars="", r_den=144, t_den=144
)
cb_op_inp_best = cb_op_best_cell * cb_op_minimum_ref * cb_op_inp_minimum
# Use identity change-of-basis operator if possible
if best_subsym.unit_cell().is_similar_to(input_symmetry.unit_cell()):
cb_op_corr = cb_op_inp_best.inverse()
try:
best_subsym_corr = best_subsym.change_basis(cb_op_corr)
except RuntimeError as e:
if str(e).find("Unsuitable value for rational rotation matrix.") < 0:
raise
else:
if best_subsym_corr.space_group() == best_subsym.space_group():
cb_op_inp_best = cb_op_corr * cb_op_inp_best
max_angular_difference = sgtbx.lattice_symmetry_find_max_delta(
reduced_cell=minimum_symmetry.unit_cell(), space_group=acentric_supergroup
)
if max_angular_difference < best_angular_difference:
best_angular_difference = max_angular_difference
best_subgroup = {
"subsym": subsym,
"ref_subsym": ref_subsym,
"best_subsym": best_subsym,
"cb_op_inp_best": cb_op_inp_best,
"max_angular_difference": max_angular_difference,
}
if best_subgroup is not None:
return best_subgroup
[docs]
class SymmetryHandler:
[docs]
def __init__(self, unit_cell=None, space_group=None, max_delta=5):
self._max_delta = max_delta
self.target_symmetry_primitive = None
self.target_symmetry_reference_setting = None
self.cb_op_inp_ref = None
self.cb_op_inp_best = None
target_space_group = space_group
if target_space_group is not None:
target_space_group = target_space_group.build_derived_patterson_group()
if unit_cell is not None:
assert (
space_group
), "space_group must be provided in combination with unit_cell"
self.target_symmetry_inp = crystal.symmetry(
unit_cell=unit_cell, space_group=target_space_group
)
if str(bravais_lattice(group=target_space_group)) == "aP":
self.cb_op_inp_ref = (
self.target_symmetry_inp.change_of_basis_op_to_minimum_cell()
)
else:
self.cb_op_inp_ref = (
self.target_symmetry_inp.change_of_basis_op_to_reference_setting()
)
self.target_symmetry_reference_setting = (
self.target_symmetry_inp.change_basis(self.cb_op_inp_ref)
)
self.cb_op_inp_best = (
self.target_symmetry_reference_setting.change_of_basis_op_to_best_cell()
* self.cb_op_inp_ref
)
elif target_space_group is not None:
self.target_symmetry_inp = crystal.symmetry(space_group=target_space_group)
self.cb_op_inp_ref = (
target_space_group.info().change_of_basis_op_to_reference_setting()
)
self.target_symmetry_reference_setting = crystal.symmetry(
space_group=target_space_group.change_basis(self.cb_op_inp_ref)
)
cb_op_reference_to_primitive = self.target_symmetry_reference_setting.change_of_basis_op_to_primitive_setting()
if unit_cell:
self.target_symmetry_primitive = (
self.target_symmetry_reference_setting.change_basis(
cb_op_reference_to_primitive
)
)
else:
self.target_symmetry_primitive = crystal.symmetry(
space_group=self.target_symmetry_reference_setting.space_group().change_basis(
cb_op_reference_to_primitive
)
)
self.cb_op_ref_inp = self.cb_op_inp_ref.inverse()
self.cb_op_primitive_inp = (
self.cb_op_ref_inp * cb_op_reference_to_primitive.inverse()
)
if self.target_symmetry_reference_setting:
logger.debug(
"Target symmetry (reference setting):\n%s",
self.target_symmetry_reference_setting,
)
if self.target_symmetry_primitive:
logger.debug(
"Target symmetry (primitive cell):\n%s",
self.target_symmetry_primitive,
)
logger.debug("cb_op primitive->input: %s", self.cb_op_primitive_inp)
[docs]
def apply_symmetry(self, crystal_model):
"""Apply symmetry constraints to a crystal model.
Returns the crystal model (with symmetry constraints applied) in the same
setting as provided as input. The cb_op returned by the method is
that necessary to transform that model to the user-provided
target symmetry.
Args:
crystal_model (dxtbx.model.Crystal): The input crystal model to which to
apply symmetry constraints.
Returns: (dxtbx.model.Crystal, cctbx.sgtbx.change_of_basis_op):
The crystal model with symmetry constraints applied, and the change_of_basis_op
that transforms the returned model to the user-specified target symmetry.
"""
if not (
self.target_symmetry_primitive
and self.target_symmetry_primitive.space_group()
):
return crystal, sgtbx.change_of_basis_op()
target_space_group = self.target_symmetry_primitive.space_group()
A = crystal_model.get_A()
max_delta = self._max_delta
items = iotbx_converter(crystal_model.get_unit_cell(), max_delta=max_delta)
target_sg_ref = target_space_group.info().reference_setting().group()
best_angular_difference = 1e8
best_subgroup = None
for item in items:
if bravais_lattice(group=target_sg_ref) != item["bravais"]:
continue
if item["max_angular_difference"] < best_angular_difference:
best_angular_difference = item["max_angular_difference"]
best_subgroup = item
if best_subgroup is None:
return None, None
cb_op_inp_best = best_subgroup["cb_op_inp_best"]
best_subsym = best_subgroup["best_subsym"]
ref_subsym = best_subgroup["ref_subsym"]
cb_op_ref_best = ref_subsym.change_of_basis_op_to_best_cell()
cb_op_best_ref = cb_op_ref_best.inverse()
cb_op_inp_ref = cb_op_best_ref * cb_op_inp_best
cb_op_ref_inp = cb_op_inp_ref.inverse()
orient = crystal_orientation(A, True)
orient_ref = orient.change_basis(
scitbx.matrix.sqr((cb_op_inp_ref).c().as_double_array()[0:9]).transpose()
)
constrain_orient = orient_ref.constrain(best_subgroup["system"])
direct_matrix = constrain_orient.direct_matrix()
a = scitbx.matrix.col(direct_matrix[:3])
b = scitbx.matrix.col(direct_matrix[3:6])
c = scitbx.matrix.col(direct_matrix[6:9])
model = Crystal(a, b, c, space_group=target_sg_ref)
assert target_sg_ref.is_compatible_unit_cell(model.get_unit_cell())
model = model.change_basis(cb_op_ref_inp)
if self.cb_op_inp_best is not None:
# Then the unit cell has been provided: this is the cb_op to map to the
# user-provided input unit cell
return model, self.cb_op_inp_best.inverse() * cb_op_inp_best
if not self.cb_op_ref_inp.is_identity_op():
if self.target_symmetry_inp.space_group() == best_subsym.space_group():
# Handle where e.g. the user has requested I2 instead of the reference C2
return model, cb_op_inp_best
# The user has specified a setting that is not the reference setting
return model, self.cb_op_ref_inp * cb_op_inp_ref
# Default to reference setting
# This change of basis op will ensure that we get the best beta angle without
# changing the centring (e.g. from C2 to I2)
cb_op_ref_best = ref_subsym.change_of_basis_op_to_best_cell(
best_monoclinic_beta=False
)
return model, cb_op_ref_best * cb_op_inp_ref