Gemmi C++ API
Loading...
Searching...
No Matches
smcif.hpp
Go to the documentation of this file.
1// Copyright 2018 Global Phasing Ltd.
2//
3// Read small molecule CIF file into SmallStructure (from small.hpp).
4
5#ifndef GEMMI_SMCIF_HPP_
6#define GEMMI_SMCIF_HPP_
7
8#include "small.hpp" // for SmallStructure
9#include "cifdoc.hpp"
10#include "numb.hpp" // for as_number
11#include "sprintf.hpp" // for to_str
12
13namespace gemmi {
14
15inline
17 using cif::as_number;
18 using cif::as_string;
19 cif::Block& block = const_cast<cif::Block&>(block_);
21 st.name = block.name;
22
23 // unit cell and symmetry
24 cif::Table cell = block.find("_cell_",
25 {"length_a", "length_b", "length_c",
26 "angle_alpha", "angle_beta", "angle_gamma"});
27 if (cell.ok()) {
28 auto c = cell.one();
29 if (!cif::is_null(c[0]) && !cif::is_null(c[1]) && !cif::is_null(c[2]))
30 st.cell.set(as_number(c[0]), as_number(c[1]), as_number(c[2]),
31 as_number(c[3]), as_number(c[4]), as_number(c[5]));
32 }
33 for (const char* tag : {"_space_group_name_H-M_alt",
34 "_symmetry_space_group_name_H-M"})
35 if (const std::string* val = block.find_value(tag)) {
36 st.spacegroup_hm = as_string(*val);
37 break;
38 }
39
41 cif::Table atom_table = block.find("_atom_site_",
42 {"label",
43 "?type_symbol",
44 "?fract_x",
45 "?fract_y",
46 "?fract_z",
47 "?U_iso_or_equiv",
48 "?B_iso_or_equiv",
49 "?occupancy",
50 "?disorder_group"});
51 for (auto row : atom_table) {
53 site.label = as_string(row[kLabel]);
54 if (row.has(kSymbol))
55 site.type_symbol = as_string(row[kSymbol]);
56 else
57 site.type_symbol = site.label;
58 if (row.has(kX))
59 site.fract.x = as_number(row[kX]);
60 if (row.has(kY))
61 site.fract.y = as_number(row[kY]);
62 if (row.has(kZ))
63 site.fract.z = as_number(row[kZ]);
64 if (row.has(kUiso))
65 site.u_iso = as_number(row[kUiso], 0.0);
66 else if (row.has(kBiso))
67 site.u_iso = as_number(row[kBiso], 0.0) / u_to_b();
68 if (row.has(kOcc))
69 site.occ = as_number(row[kOcc], 1.0);
70 if (row.has2(kDisorderGroup))
71 // ignore non-integer _atom_site_disorder_group
72 site.disorder_group = string_to_int(row[kDisorderGroup], false);
73 split_element_and_charge(site.type_symbol, &site);
74 st.sites.push_back(site);
75 }
76
77 std::vector<SmallStructure::Site>::iterator aniso_site = st.sites.begin();
78 for (auto row : block.find("_atom_site_aniso_",
79 {"label", "U_11", "U_22", "U_33",
80 "U_12", "U_13", "U_23"})) {
81 std::string aniso_label = row.str(0);
82 if (aniso_site == st.sites.end() || aniso_site->label != aniso_label) {
83 aniso_site = std::find_if(st.sites.begin(), st.sites.end(),
84 [&](SmallStructure::Site& x) { return x.label == aniso_label; });
85 if (aniso_site == st.sites.end())
86 continue;
87 }
88 aniso_site->aniso.u11 = as_number(row[1], 0.0);
89 aniso_site->aniso.u22 = as_number(row[2], 0.0);
90 aniso_site->aniso.u33 = as_number(row[3], 0.0);
91 aniso_site->aniso.u12 = as_number(row[4], 0.0);
92 aniso_site->aniso.u13 = as_number(row[5], 0.0);
93 aniso_site->aniso.u23 = as_number(row[6], 0.0);
94 ++aniso_site;
95 }
96
97 for (auto row : block.find("_atom_type_",
98 {"symbol",
99 "scat_dispersion_real",
100 "scat_dispersion_imag"})) {
101 SmallStructure::AtomType atom_type;
102 atom_type.symbol = row.str(0);
103 atom_type.dispersion_real = cif::as_number(row[1]);
104 atom_type.dispersion_imag = cif::as_number(row[2]);
105 split_element_and_charge(atom_type.symbol, &atom_type);
106 st.atom_types.push_back(atom_type);
107 }
108 if (cif::Column w_col = block.find_values("_diffrn_radiation_wavelength"))
109 st.wavelength = cif::as_number(w_col.at(0));
110 st.setup_cell_images();
111
112 return st;
113}
114
116 cif::Block block;
117 block.name = st.name;
118 if (st.cell.is_crystal()) {
119 block.set_pair("_cell_length_a", to_str(st.cell.a));
120 block.set_pair("_cell_length_b", to_str(st.cell.b));
121 block.set_pair("_cell_length_c", to_str(st.cell.c));
122 block.set_pair("_cell_angle_alpha", to_str(st.cell.alpha));
123 block.set_pair("_cell_angle_beta", to_str(st.cell.beta));
124 block.set_pair("_cell_angle_gamma", to_str(st.cell.gamma));
125 }
126 if (!st.spacegroup_hm.empty())
127 block.set_pair("_symmetry_space_group_name_H-M", cif::quote(st.spacegroup_hm));
128 if (st.wavelength != 0)
129 block.set_pair("_diffrn_radiation_wavelength", to_str(st.wavelength));
130
131 cif::Loop& atom_loop = block.init_loop("_atom_site_", {"label",
132 "type_symbol",
133 "fract_x",
134 "fract_y",
135 "fract_z",
136 "U_iso_or_equiv",
137 "occupancy",
138 "disorder_group"});
139 for (const SmallStructure::Site& site: st.sites) {
140 atom_loop.add_row({
141 cif::quote(site.label),
142 site.type_symbol.empty() ? site.element_and_charge_symbol()
143 : cif::quote(site.type_symbol),
144 to_str(site.fract.x),
145 to_str(site.fract.y),
146 to_str(site.fract.z),
147 to_str(site.u_iso),
148 to_str(site.occ),
149 site.disorder_group == 0 ? "." : std::to_string(site.disorder_group)
150 });
151 }
152
153 bool has_aniso = false;
154 for (const SmallStructure::Site& site: st.sites)
155 if (site.aniso.nonzero()) {
156 has_aniso = true;
157 break;
158 }
159 if (has_aniso) {
160 cif::Loop& aniso_loop = block.init_loop("_atom_site_aniso_", {
161 "label", "U_11", "U_22", "U_33", "U_12", "U_13", "U_23"});
162 for (const SmallStructure::Site& site: st.sites)
163 if (site.aniso.nonzero())
164 aniso_loop.add_row({
165 cif::quote(site.label),
166 to_str(site.aniso.u11), to_str(site.aniso.u22), to_str(site.aniso.u33),
167 to_str(site.aniso.u12), to_str(site.aniso.u13), to_str(site.aniso.u23)
168 });
169 }
170
171 return block;
172}
173
174} // namespace gemmi
175#endif
double as_number(const std::string &s, double nan=NAN)
Definition numb.hpp:19
std::string as_string(const std::string &value)
Definition cifdoc.hpp:81
bool is_null(const std::string &value)
Definition cifdoc.hpp:77
std::string quote(std::string v)
Definition cifdoc.hpp:1162
int string_to_int(const char *p, bool checked, size_t length=0)
Definition atox.hpp:73
void split_element_and_charge(const std::string &label, T *dest)
Definition small.hpp:104
SmallStructure make_small_structure_from_block(const cif::Block &block_)
Definition smcif.hpp:16
cif::Block make_cif_block_from_small_structure(const SmallStructure &st)
Definition smcif.hpp:115
GEMMI_DLL int GEMMI_DLL int std::string to_str(double d)
Definition sprintf.hpp:36
constexpr double u_to_b()
Definition math.hpp:29
Column find_values(const std::string &tag)
Definition cifdoc.hpp:848
Table find(const std::string &prefix, const std::vector< std::string > &tags)
Definition cifdoc.hpp:961
void set_pair(const std::string &tag, const std::string &value)
Definition cifdoc.hpp:890
Loop & init_loop(const std::string &prefix, std::vector< std::string > tags)
Definition cifdoc.hpp:488
std::string name
Definition cifdoc.hpp:444
const std::string * find_value(const std::string &tag) const
Definition cifdoc.hpp:834
bool ok() const
Definition cifdoc.hpp:350