@datagrok-libraries/bio 5.3.0 → 5.4.1
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/package.json +1 -1
- package/src/utils/atomic-works.d.ts +2 -0
- package/src/utils/atomic-works.d.ts.map +1 -0
- package/src/utils/atomic-works.js +354 -0
- package/src/utils/const.d.ts +48 -0
- package/src/utils/const.d.ts.map +1 -0
- package/src/utils/const.js +29 -0
- package/src/utils/macromolecule.js +2 -2
- package/src/utils/monomer-library.d.ts +43 -0
- package/src/utils/monomer-library.d.ts.map +1 -0
- package/src/utils/monomer-library.js +154 -0
- package/src/utils/monomer-utils.d.ts +10 -0
- package/src/utils/monomer-utils.d.ts.map +1 -0
- package/src/utils/monomer-utils.js +125 -0
- package/src/utils/notation-converter.d.ts.map +1 -1
- package/src/utils/notation-converter.js +5 -1
- package/src/utils/to-atomic-level.d.ts +4 -0
- package/src/utils/to-atomic-level.d.ts.map +1 -0
- package/src/utils/to-atomic-level.js +1095 -0
- package/src/utils/units-handler.d.ts +4 -0
- package/src/utils/units-handler.d.ts.map +1 -1
- package/src/utils/units-handler.js +6 -8
- package/tsconfig.json +1 -1
package/package.json
CHANGED
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"atomic-works.d.ts","sourceRoot":"","sources":["atomic-works.ts"],"names":[],"mappings":"AAuYA,wBAAsB,WAAW,CAAC,QAAQ,EAAE,GAAG,EAAE,EAAE,GAAG,OAAO,CAAC,MAAM,EAAE,CAAC,CAqBtE"}
|
|
@@ -0,0 +1,354 @@
|
|
|
1
|
+
var __awaiter = (this && this.__awaiter) || function (thisArg, _arguments, P, generator) {
|
|
2
|
+
function adopt(value) { return value instanceof P ? value : new P(function (resolve) { resolve(value); }); }
|
|
3
|
+
return new (P || (P = Promise))(function (resolve, reject) {
|
|
4
|
+
function fulfilled(value) { try { step(generator.next(value)); } catch (e) { reject(e); } }
|
|
5
|
+
function rejected(value) { try { step(generator["throw"](value)); } catch (e) { reject(e); } }
|
|
6
|
+
function step(result) { result.done ? resolve(result.value) : adopt(result.value).then(fulfilled, rejected); }
|
|
7
|
+
step((generator = generator.apply(thisArg, _arguments || [])).next());
|
|
8
|
+
});
|
|
9
|
+
};
|
|
10
|
+
import * as grok from 'datagrok-api/grok';
|
|
11
|
+
function extractAtomDataV3K(v3KMolblock) {
|
|
12
|
+
const numbers = extractAtomAndBondCountsV3K(v3KMolblock);
|
|
13
|
+
let begin = v3KMolblock.indexOf('M V30 BEGIN ATOM'); // V3000 block for atom coordinates
|
|
14
|
+
begin = v3KMolblock.indexOf('\n', begin);
|
|
15
|
+
let end = begin;
|
|
16
|
+
const atomIndices = Array(numbers.atomCount);
|
|
17
|
+
const atomTypes = Array(numbers.atomCount);
|
|
18
|
+
const x = Array(numbers.atomCount);
|
|
19
|
+
const y = Array(numbers.atomCount);
|
|
20
|
+
for (let i = 0; i < numbers.atomCount; i++) {
|
|
21
|
+
begin = v3KMolblock.indexOf('V30', begin) + 4;
|
|
22
|
+
end = v3KMolblock.indexOf(' ', begin);
|
|
23
|
+
atomIndices[i] = parseInt(v3KMolblock.substring(begin, end));
|
|
24
|
+
begin = end + 1;
|
|
25
|
+
end = v3KMolblock.indexOf(' ', begin);
|
|
26
|
+
atomTypes[i] = v3KMolblock.substring(begin, end);
|
|
27
|
+
begin = end + 1;
|
|
28
|
+
end = v3KMolblock.indexOf(' ', begin);
|
|
29
|
+
x[i] = parseFloat(v3KMolblock.substring(begin, end));
|
|
30
|
+
begin = end + 1;
|
|
31
|
+
end = v3KMolblock.indexOf(' ', begin);
|
|
32
|
+
y[i] = parseFloat(v3KMolblock.substring(begin, end));
|
|
33
|
+
begin = v3KMolblock.indexOf('\n', begin) + 1;
|
|
34
|
+
}
|
|
35
|
+
return { atomIndices: atomIndices, atomTypes: atomTypes, x: x, y: y };
|
|
36
|
+
}
|
|
37
|
+
// todo: extractRgroupData
|
|
38
|
+
function extractAtomAndBondCountsV3K(v3KMolblock) {
|
|
39
|
+
v3KMolblock = v3KMolblock.replaceAll('\r', ''); // equalize old and new sdf standards
|
|
40
|
+
// parse atom count
|
|
41
|
+
let idxBegin = v3KMolblock.indexOf('COUNTS') + 7;
|
|
42
|
+
let idxEnd = v3KMolblock.indexOf(' ', idxBegin);
|
|
43
|
+
const numOfAtoms = parseInt(v3KMolblock.substring(idxBegin, idxEnd));
|
|
44
|
+
// parse bond count
|
|
45
|
+
idxBegin = idxEnd + 1;
|
|
46
|
+
idxEnd = v3KMolblock.indexOf(' ', idxBegin);
|
|
47
|
+
const numOfBonds = parseInt(v3KMolblock.substring(idxBegin, idxEnd));
|
|
48
|
+
return { atomCount: numOfAtoms, bondCount: numOfBonds };
|
|
49
|
+
}
|
|
50
|
+
function getIndices(v2KMolblock, v3KMolblock) {
|
|
51
|
+
// one should also take into account that there can be multiple M RGP lines
|
|
52
|
+
// with R-Groups specified
|
|
53
|
+
let begin = v2KMolblock.indexOf('M RGP', 0) + 8;
|
|
54
|
+
let end = v2KMolblock.indexOf('\n', begin);
|
|
55
|
+
// todo: maybe this part deserves a separate function
|
|
56
|
+
// there may be situation when the rgp information is distributed among
|
|
57
|
+
// multiple lines, this must be taken into account
|
|
58
|
+
const rgpStringParsed = v2KMolblock.substring(begin, end).replaceAll(' ', ' ').replaceAll(' ', ' ').split(' ');
|
|
59
|
+
const rgpData = rgpStringParsed.map((el) => parseInt(el));
|
|
60
|
+
// rgpData[0] is the number of R-groups
|
|
61
|
+
// the following code sets remFirst to node# to which RGP#1 is substituted
|
|
62
|
+
// todo: handle the exceptional case when there is not enough rgroups
|
|
63
|
+
// todo: handle the exceptional case when the order is different
|
|
64
|
+
const remFirst = rgpData[1]; // leftmost Rgroup node
|
|
65
|
+
const remLast = rgpData[rgpData.length - 2]; // rightmost Rgroup node
|
|
66
|
+
// if (remFirst === -1 || remLast === -1)
|
|
67
|
+
// throw new Error('RGP parsing: first and last groups have wrong format');
|
|
68
|
+
// const remFirst = rgpData[2] === '1' ? parseInt(rgpData[1]) : parseInt(rgpData[3]);
|
|
69
|
+
// const remLast = rgpData[2] === '2' ? parseInt(rgpData[1]) : parseInt(rgpData[3]);
|
|
70
|
+
// todo: rename 'numbers'
|
|
71
|
+
const numbers = extractAtomAndBondCountsV3K(v3KMolblock);
|
|
72
|
+
let indexBonds = v3KMolblock.indexOf('M V30 BEGIN BOND'); // V3000 bond block
|
|
73
|
+
indexBonds = v3KMolblock.indexOf('\n', indexBonds);
|
|
74
|
+
begin = indexBonds;
|
|
75
|
+
end = indexBonds;
|
|
76
|
+
let first = 0; // todo: improve notation
|
|
77
|
+
let last = 0;
|
|
78
|
+
let remBondFirst = 0;
|
|
79
|
+
let remBondLast = 0;
|
|
80
|
+
// iterate over edges of the graph and find those
|
|
81
|
+
for (let j = 0; j < numbers.bondCount; j++) {
|
|
82
|
+
if (first === 0 || last === 0) {
|
|
83
|
+
begin = v3KMolblock.indexOf('V30', begin) + 4;
|
|
84
|
+
end = v3KMolblock.indexOf('\n', begin);
|
|
85
|
+
const bondStringParsed = v3KMolblock.substring(begin, end).replaceAll(' ', ' ').replaceAll(' ', ' ').split(' ');
|
|
86
|
+
const bondData = bondStringParsed.map((el) => parseInt(el));
|
|
87
|
+
if (bondData[2] === remFirst) { // bondData[2] is the 1st node/atom of the bond
|
|
88
|
+
first = bondData[3]; // bondData[3] is the 2nd node/atom of the bond
|
|
89
|
+
remBondFirst = bondData[0]; // bondData[0] is the idx of the associated bond/edge
|
|
90
|
+
}
|
|
91
|
+
else if (bondData[3] === remFirst) {
|
|
92
|
+
first = bondData[2];
|
|
93
|
+
remBondFirst = bondData[0];
|
|
94
|
+
}
|
|
95
|
+
else if (bondData[2] === remLast) {
|
|
96
|
+
last = bondData[3];
|
|
97
|
+
remBondLast = bondData[0];
|
|
98
|
+
}
|
|
99
|
+
else if (bondData[3] === remLast) {
|
|
100
|
+
last = bondData[2];
|
|
101
|
+
remBondLast = bondData[0];
|
|
102
|
+
}
|
|
103
|
+
}
|
|
104
|
+
}
|
|
105
|
+
return { first, last, remFirst, remLast, remBondFirst, remBondLast };
|
|
106
|
+
}
|
|
107
|
+
/* provide description */
|
|
108
|
+
function rotateBackboneV3K(v3KMolblock, indices) {
|
|
109
|
+
return __awaiter(this, void 0, void 0, function* () {
|
|
110
|
+
const coordinates = extractAtomDataV3K(v3KMolblock); // naming?
|
|
111
|
+
const atomCount = coordinates.atomIndices.length;
|
|
112
|
+
const first = indices['first'];
|
|
113
|
+
const last = indices['last'];
|
|
114
|
+
const xCenter = (coordinates.x[last] + coordinates.x[first]) / 2;
|
|
115
|
+
const yCenter = (coordinates.y[last] + coordinates.y[first]) / 2;
|
|
116
|
+
//place to center
|
|
117
|
+
for (let i = 0; i < atomCount; i++) {
|
|
118
|
+
coordinates.x[i] -= xCenter;
|
|
119
|
+
coordinates.y[i] -= yCenter;
|
|
120
|
+
}
|
|
121
|
+
let angle = 0;
|
|
122
|
+
if (coordinates.x[first] === 0) { // both vertices are on OY
|
|
123
|
+
angle = coordinates.y[first] > coordinates.y[last] ? Math.PI / 2 : -Math.PI / 2;
|
|
124
|
+
}
|
|
125
|
+
else if (coordinates.y[first] === 0) { // both vertices are on OX
|
|
126
|
+
angle = coordinates.x[first] > coordinates.x[last] ? Math.PI : 0; // v
|
|
127
|
+
}
|
|
128
|
+
else {
|
|
129
|
+
const tangent = coordinates.y[first] / coordinates.x[first];
|
|
130
|
+
if (coordinates.x[first] < coordinates.x[last])
|
|
131
|
+
angle = tangent > 0 ? -Math.atan(tangent) : Math.atan(tangent);
|
|
132
|
+
else
|
|
133
|
+
angle = tangent > 0 ? Math.PI - Math.atan(tangent) : Math.atan(tangent) - Math.PI;
|
|
134
|
+
}
|
|
135
|
+
const cos = Math.cos(angle);
|
|
136
|
+
const sin = Math.sin(angle);
|
|
137
|
+
for (let i = 0; i < atomCount; i++) {
|
|
138
|
+
const xAdd = coordinates.x[i];
|
|
139
|
+
coordinates.x[i] = xAdd * cos - coordinates.y[i] * sin;
|
|
140
|
+
coordinates.y[i] = xAdd * sin + coordinates.y[i] * cos;
|
|
141
|
+
}
|
|
142
|
+
//place to right
|
|
143
|
+
const xShift = coordinates.x[first];
|
|
144
|
+
for (let i = 0; i < atomCount; i++)
|
|
145
|
+
coordinates.x[i] -= xShift;
|
|
146
|
+
//rewrite v3KMolblock
|
|
147
|
+
let index = v3KMolblock.indexOf('M V30 BEGIN ATOM'); // V3000 index for atoms coordinates
|
|
148
|
+
index = v3KMolblock.indexOf('\n', index);
|
|
149
|
+
let indexEnd = index;
|
|
150
|
+
for (let i = 0; i < atomCount; i++) {
|
|
151
|
+
index = v3KMolblock.indexOf('V30', index) + 4;
|
|
152
|
+
index = v3KMolblock.indexOf(' ', index) + 1;
|
|
153
|
+
index = v3KMolblock.indexOf(' ', index) + 1;
|
|
154
|
+
indexEnd = v3KMolblock.indexOf(' ', index) + 1;
|
|
155
|
+
indexEnd = v3KMolblock.indexOf(' ', indexEnd);
|
|
156
|
+
v3KMolblock = v3KMolblock.slice(0, index) +
|
|
157
|
+
coordinates.x[i] + ' ' + coordinates.y[i] +
|
|
158
|
+
v3KMolblock.slice(indexEnd);
|
|
159
|
+
index = v3KMolblock.indexOf('\n', index) + 1;
|
|
160
|
+
}
|
|
161
|
+
return v3KMolblock;
|
|
162
|
+
});
|
|
163
|
+
}
|
|
164
|
+
/* provide description */
|
|
165
|
+
function linkV3K(monomers) {
|
|
166
|
+
let macroMolBlock = '\nDatagrok macromolecule handler\n\n';
|
|
167
|
+
macroMolBlock += ' 0 0 0 0 0 0 999 V3000\n';
|
|
168
|
+
macroMolBlock += 'M V30 BEGIN CTAB\n';
|
|
169
|
+
let atomBlock = '';
|
|
170
|
+
let bondBlock = '';
|
|
171
|
+
// const collectionBlock = '';
|
|
172
|
+
// const collection: number [] = [];
|
|
173
|
+
let atomCount = 0;
|
|
174
|
+
let bondCount = 0;
|
|
175
|
+
let xShift = 0; // ?
|
|
176
|
+
for (let i = 0; i < monomers.length; i++) {
|
|
177
|
+
let v3KMolfile = monomers[i]['molfile'];
|
|
178
|
+
const first = monomers[i]['indices']['first'];
|
|
179
|
+
const last = monomers[i]['indices']['last'];
|
|
180
|
+
const remFirst = monomers[i]['indices']['remFirst'];
|
|
181
|
+
const remLast = monomers[i]['indices']['remLast'];
|
|
182
|
+
const remBondFirst = monomers[i]['indices']['remBondFirst'];
|
|
183
|
+
const remBondLast = monomers[i]['indices']['remBondLast'];
|
|
184
|
+
// todo: handle r-groups
|
|
185
|
+
v3KMolfile = v3KMolfile.replaceAll('(-\nM V30 ', '(')
|
|
186
|
+
.replaceAll('-\nM V30 ', '').replaceAll(' )', ')');
|
|
187
|
+
// todo: improve naming
|
|
188
|
+
const numbers = extractAtomAndBondCountsV3K(v3KMolfile);
|
|
189
|
+
const coordinates = extractAtomDataV3K(v3KMolfile);
|
|
190
|
+
let indexAtoms = v3KMolfile.indexOf('M V30 BEGIN ATOM'); // V3000 atom block
|
|
191
|
+
indexAtoms = v3KMolfile.indexOf('\n', indexAtoms);
|
|
192
|
+
let index = indexAtoms;
|
|
193
|
+
let indexEnd = indexAtoms;
|
|
194
|
+
const totalShift = xShift - coordinates.x[first - 1];
|
|
195
|
+
for (let j = 0; j < numbers.atomCount; j++) {
|
|
196
|
+
if (coordinates.atomIndices[j] != remFirst && coordinates.atomIndices[j] != remLast) { //|| i === 0) {
|
|
197
|
+
//rewrite atom number
|
|
198
|
+
index = v3KMolfile.indexOf('V30', index) + 4;
|
|
199
|
+
indexEnd = v3KMolfile.indexOf(' ', index);
|
|
200
|
+
let atomNumber = parseInt(v3KMolfile.substring(index, indexEnd));
|
|
201
|
+
atomNumber = (atomNumber > remFirst && atomNumber > remLast) ? atomNumber - 2 :
|
|
202
|
+
(atomNumber > remFirst || atomNumber > remLast) ? atomNumber - 1 : atomNumber;
|
|
203
|
+
atomNumber += atomCount;
|
|
204
|
+
v3KMolfile = v3KMolfile.slice(0, index) + atomNumber + v3KMolfile.slice(indexEnd);
|
|
205
|
+
//rewrite coordinates
|
|
206
|
+
index = v3KMolfile.indexOf(' ', index) + 1;
|
|
207
|
+
index = v3KMolfile.indexOf(' ', index) + 1;
|
|
208
|
+
indexEnd = v3KMolfile.indexOf(' ', index);
|
|
209
|
+
let coordinate = Math.round(10000 * (parseFloat(v3KMolfile.substring(index, indexEnd)) + totalShift)) / 10000;
|
|
210
|
+
v3KMolfile = v3KMolfile.slice(0, index) + coordinate + v3KMolfile.slice(indexEnd);
|
|
211
|
+
index = v3KMolfile.indexOf(' ', index) + 1;
|
|
212
|
+
indexEnd = v3KMolfile.indexOf(' ', index);
|
|
213
|
+
coordinate = Math.round(10000 * (parseFloat(v3KMolfile.substring(index, indexEnd)))) / 10000;
|
|
214
|
+
v3KMolfile = v3KMolfile.slice(0, index) + coordinate + v3KMolfile.slice(indexEnd);
|
|
215
|
+
index = v3KMolfile.indexOf('\n', index) + 1;
|
|
216
|
+
}
|
|
217
|
+
else {
|
|
218
|
+
index = v3KMolfile.indexOf('M V30', index) - 1;
|
|
219
|
+
indexEnd = v3KMolfile.indexOf('\n', index + 1);
|
|
220
|
+
v3KMolfile = v3KMolfile.slice(0, index) + v3KMolfile.slice(indexEnd);
|
|
221
|
+
}
|
|
222
|
+
}
|
|
223
|
+
const indexAtomsEnd = v3KMolfile.indexOf('M V30 END ATOM');
|
|
224
|
+
atomBlock += v3KMolfile.substring(indexAtoms + 1, indexAtomsEnd);
|
|
225
|
+
let indexBonds = v3KMolfile.indexOf('M V30 BEGIN BOND'); // V3000 index for bonds
|
|
226
|
+
indexBonds = v3KMolfile.indexOf('\n', indexBonds);
|
|
227
|
+
index = indexBonds;
|
|
228
|
+
indexEnd = indexBonds;
|
|
229
|
+
let bondNumber = 0;
|
|
230
|
+
for (let j = 0; j < numbers.bondCount; j++) {
|
|
231
|
+
//rewrite bond number
|
|
232
|
+
index = v3KMolfile.indexOf('V30', index) + 4;
|
|
233
|
+
indexEnd = v3KMolfile.indexOf(' ', index);
|
|
234
|
+
bondNumber = parseInt(v3KMolfile.substring(index, indexEnd));
|
|
235
|
+
if (bondNumber === remBondFirst || bondNumber === remBondLast) {
|
|
236
|
+
indexEnd = v3KMolfile.indexOf('\n', index) + 1;
|
|
237
|
+
index -= 7;
|
|
238
|
+
v3KMolfile = v3KMolfile.slice(0, index) + v3KMolfile.slice(indexEnd);
|
|
239
|
+
continue;
|
|
240
|
+
}
|
|
241
|
+
bondNumber = (bondNumber > remBondFirst && bondNumber > remBondLast) ? bondNumber - 2 :
|
|
242
|
+
(bondNumber > remBondFirst || bondNumber > remBondLast) ? bondNumber - 1 : bondNumber;
|
|
243
|
+
bondNumber += bondCount;
|
|
244
|
+
v3KMolfile = v3KMolfile.slice(0, index) + bondNumber + v3KMolfile.slice(indexEnd);
|
|
245
|
+
//rewrite atom pair in bond
|
|
246
|
+
index = v3KMolfile.indexOf(' ', index) + 1;
|
|
247
|
+
index = v3KMolfile.indexOf(' ', index) + 1;
|
|
248
|
+
indexEnd = v3KMolfile.indexOf(' ', index);
|
|
249
|
+
let atomNumber = parseInt(v3KMolfile.substring(index, indexEnd));
|
|
250
|
+
atomNumber = (atomNumber > remFirst && atomNumber > remLast) ? atomNumber - 2 :
|
|
251
|
+
(atomNumber > remFirst || atomNumber > remLast) ? atomNumber - 1 : atomNumber;
|
|
252
|
+
atomNumber += atomCount;
|
|
253
|
+
v3KMolfile = v3KMolfile.slice(0, index) + atomNumber + v3KMolfile.slice(indexEnd);
|
|
254
|
+
index = v3KMolfile.indexOf(' ', index) + 1;
|
|
255
|
+
indexEnd = Math.min(v3KMolfile.indexOf('\n', index), v3KMolfile.indexOf(' ', index));
|
|
256
|
+
atomNumber = parseInt(v3KMolfile.substring(index, indexEnd));
|
|
257
|
+
atomNumber = (atomNumber > remFirst && atomNumber > remLast) ? atomNumber - 2 :
|
|
258
|
+
(atomNumber > remFirst || atomNumber > remLast) ? atomNumber - 1 : atomNumber;
|
|
259
|
+
atomNumber += atomCount;
|
|
260
|
+
v3KMolfile = v3KMolfile.slice(0, index) + atomNumber + v3KMolfile.slice(indexEnd);
|
|
261
|
+
index = v3KMolfile.indexOf('\n', index) + 1;
|
|
262
|
+
}
|
|
263
|
+
const indexBondEnd = v3KMolfile.indexOf('M V30 END BOND');
|
|
264
|
+
bondBlock += v3KMolfile.substring(indexBonds + 1, indexBondEnd);
|
|
265
|
+
//let indexCollection = v3KMolfile.indexOf('M V30 MDLV30/STEABS ATOMS=('); // V3000 index for collections
|
|
266
|
+
// while (indexCollection != -1) {
|
|
267
|
+
// indexCollection += 28;
|
|
268
|
+
// const collectionEnd = v3KMolfile.indexOf(')', indexCollection);
|
|
269
|
+
// const collectionEntries = v3KMolfile.substring(indexCollection, collectionEnd).split(' ').slice(1);
|
|
270
|
+
// collectionEntries.forEach((e: string) => {
|
|
271
|
+
// collection.push(parseInt(e) + atomCount);
|
|
272
|
+
// });
|
|
273
|
+
// indexCollection = collectionEnd;
|
|
274
|
+
// indexCollection = v3KMolfile.indexOf('M V30 MDLV30/STEABS ATOMS=(', indexCollection);
|
|
275
|
+
// }
|
|
276
|
+
atomCount += numbers.atomCount - 2;
|
|
277
|
+
bondCount += numbers.bondCount - 2;
|
|
278
|
+
xShift += coordinates.x[last] - coordinates.x[first] + 1;
|
|
279
|
+
if (i === monomers.length - 1) {
|
|
280
|
+
atomCount++;
|
|
281
|
+
const shift = xShift + 0.2;
|
|
282
|
+
atomBlock += 'M V30 ' + atomCount + ' O ' + shift + ' 0 0.000000 0\n';
|
|
283
|
+
}
|
|
284
|
+
bondCount++;
|
|
285
|
+
if (i === monomers.length - 1) {
|
|
286
|
+
const rightTerminal = (last > remFirst && last > remLast) ? last + atomCount - (numbers.atomCount - 2) - 3 :
|
|
287
|
+
(last > remFirst || last > remLast) ? last + atomCount - (numbers.atomCount - 2) - 2 :
|
|
288
|
+
last + atomCount - (numbers.atomCount - 2) - 1;
|
|
289
|
+
bondBlock += 'M V30 ' + bondCount + ' 1 ' + rightTerminal + ' ' + atomCount + '\n';
|
|
290
|
+
}
|
|
291
|
+
else {
|
|
292
|
+
const rightTerminal = (last > remFirst && last > remLast) ? last + atomCount - (numbers.atomCount - 2) - 2 :
|
|
293
|
+
(last > remFirst || last > remLast) ? last + atomCount - (numbers.atomCount - 2) - 1 :
|
|
294
|
+
last + atomCount - (numbers.atomCount - 2);
|
|
295
|
+
const next = monomers[i + 1]['indices'];
|
|
296
|
+
const nextFirst = next['first'];
|
|
297
|
+
const nextRemFirst = next['remFirst'];
|
|
298
|
+
const nextRemLast = next['remLast'];
|
|
299
|
+
const leftTerminal = (nextFirst > nextRemFirst && nextFirst > nextRemLast) ? nextFirst + atomCount - 2 :
|
|
300
|
+
(nextFirst > nextRemFirst || nextFirst > nextRemLast) ? nextFirst + atomCount - 1 :
|
|
301
|
+
nextFirst + atomCount;
|
|
302
|
+
bondBlock += 'M V30 ' + bondCount + ' 1 ' + rightTerminal + ' ' + leftTerminal + '\n';
|
|
303
|
+
}
|
|
304
|
+
}
|
|
305
|
+
// const entries = 4;
|
|
306
|
+
// const collNumber = Math.ceil(collection.length / entries);
|
|
307
|
+
// collectionBlock += 'M V30 MDLV30/STEABS ATOMS=(' + collection.length + ' -\n';
|
|
308
|
+
// for (let i = 0; i < collNumber; i++) {
|
|
309
|
+
// collectionBlock += 'M V30 ';
|
|
310
|
+
// const entriesCurrent = i + 1 === collNumber ? collection.length - (collNumber - 1)*entries : entries;
|
|
311
|
+
// for (let j = 0; j < entriesCurrent; j++) {
|
|
312
|
+
// collectionBlock += (j + 1 === entriesCurrent) ?
|
|
313
|
+
// (i === collNumber - 1 ? collection[entries*i + j] + ')\n' : collection[entries*i + j] + ' -\n') :
|
|
314
|
+
// collection[entries*i + j] + ' ';
|
|
315
|
+
// }
|
|
316
|
+
// }
|
|
317
|
+
//generate file
|
|
318
|
+
macroMolBlock += 'M V30 COUNTS ' + atomCount + ' ' + bondCount + ' 0 0 0\n';
|
|
319
|
+
macroMolBlock += 'M V30 BEGIN ATOM\n';
|
|
320
|
+
macroMolBlock += atomBlock;
|
|
321
|
+
macroMolBlock += 'M V30 END ATOM\n';
|
|
322
|
+
macroMolBlock += 'M V30 BEGIN BOND\n';
|
|
323
|
+
macroMolBlock += bondBlock;
|
|
324
|
+
macroMolBlock += 'M V30 END BOND\n';
|
|
325
|
+
//macroMolBlock += 'M V30 BEGIN COLLECTION\n';
|
|
326
|
+
//macroMolBlock += collectionBlock;
|
|
327
|
+
//macroMolBlock += 'M V30 END COLLECTION\n';
|
|
328
|
+
macroMolBlock += 'M V30 END CTAB\n';
|
|
329
|
+
macroMolBlock += 'M END\n';
|
|
330
|
+
return macroMolBlock;
|
|
331
|
+
}
|
|
332
|
+
export function getMacroMol(monomers) {
|
|
333
|
+
return __awaiter(this, void 0, void 0, function* () {
|
|
334
|
+
const result = [];
|
|
335
|
+
const moduleRdkit = yield grok.functions.call('Chem:getRdKitModule');
|
|
336
|
+
for (let i = 0; i < monomers.length; i++) {
|
|
337
|
+
for (let j = 0; j < monomers[i].length; j++) {
|
|
338
|
+
const molObj = moduleRdkit.get_mol(monomers[i][j]['molfile']);
|
|
339
|
+
const v3KMolblock = molObj.get_v3Kmolblock();
|
|
340
|
+
const indices = getIndices(monomers[i][j]['molfile'], v3KMolblock);
|
|
341
|
+
monomers[i][j]['indices'] = indices;
|
|
342
|
+
// a new molfile for 'rotated' is obtained in v3k format
|
|
343
|
+
monomers[i][j]['molfile'] = yield rotateBackboneV3K(v3KMolblock, indices);
|
|
344
|
+
// monomers[i][j]['molfile'] = v3KMolblock;
|
|
345
|
+
molObj === null || molObj === void 0 ? void 0 : molObj.delete();
|
|
346
|
+
}
|
|
347
|
+
// seemingly, at this stage the bond is reconstructed
|
|
348
|
+
result.push(linkV3K(monomers[i]));
|
|
349
|
+
// result.push(monomers[i][0]['molfile']);
|
|
350
|
+
}
|
|
351
|
+
return result;
|
|
352
|
+
});
|
|
353
|
+
}
|
|
354
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"atomic-works.js","sourceRoot":"","sources":["atomic-works.ts"],"names":[],"mappings":"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","sourcesContent":["import * as grok from 'datagrok-api/grok';\n\ntype Indices = {\n  first: number, // node to which remFirst is attached\n  last: number, // node to which remLast is attached\n  remFirst: number, // \"leftmost\" r-group node of the monomer\n  remLast: number, // \"rightmost\" r-group node of the monomer\n  remBondFirst: number, // idx of the bond between first and remFirst\n  remBondLast: number, // idx of the bond between last and remLast\n}\n\n// todo: improve\ntype AtomData = {\n  atomIndices: number[],\n  atomTypes: string[],\n  x: number[],\n  y: number[],\n}\n\nfunction extractAtomDataV3K(v3KMolblock: string): AtomData {\n  const numbers = extractAtomAndBondCountsV3K(v3KMolblock);\n  let begin = v3KMolblock.indexOf('M  V30 BEGIN ATOM'); // V3000 block for atom coordinates\n  begin = v3KMolblock.indexOf('\\n', begin);\n  let end = begin;\n\n  const atomIndices: number[] = Array(numbers.atomCount);\n  const atomTypes: string[] = Array(numbers.atomCount);\n  const x: number[] = Array(numbers.atomCount);\n  const y: number[] = Array(numbers.atomCount);\n\n  for (let i = 0; i < numbers.atomCount; i++) {\n    begin = v3KMolblock.indexOf('V30', begin) + 4;\n    end = v3KMolblock.indexOf(' ', begin);\n    atomIndices[i] = parseInt(v3KMolblock.substring(begin, end));\n\n    begin = end + 1;\n    end = v3KMolblock.indexOf(' ', begin);\n    atomTypes[i] = v3KMolblock.substring(begin, end);\n\n    begin = end + 1;\n    end = v3KMolblock.indexOf(' ', begin);\n    x[i] = parseFloat(v3KMolblock.substring(begin, end));\n\n    begin = end + 1;\n    end = v3KMolblock.indexOf(' ', begin);\n    y[i] = parseFloat(v3KMolblock.substring(begin, end));\n\n    begin = v3KMolblock.indexOf('\\n', begin) + 1;\n  }\n\n  return {atomIndices: atomIndices, atomTypes: atomTypes, x: x, y: y};\n}\n\n// todo: extractRgroupData\n\nfunction extractAtomAndBondCountsV3K(v3KMolblock: string): {atomCount: number, bondCount: number} {\n  v3KMolblock = v3KMolblock.replaceAll('\\r', ''); // equalize old and new sdf standards\n\n  // parse atom count\n  let idxBegin = v3KMolblock.indexOf('COUNTS') + 7;\n  let idxEnd = v3KMolblock.indexOf(' ', idxBegin);\n  const numOfAtoms = parseInt(v3KMolblock.substring(idxBegin, idxEnd));\n\n  // parse bond count\n  idxBegin = idxEnd + 1;\n  idxEnd = v3KMolblock.indexOf(' ', idxBegin);\n  const numOfBonds = parseInt(v3KMolblock.substring(idxBegin, idxEnd));\n\n  return {atomCount: numOfAtoms, bondCount: numOfBonds};\n}\n\nfunction getIndices(v2KMolblock: string, v3KMolblock: string): Indices {\n  // one should also take into account that there can be multiple M RGP lines\n  // with R-Groups specified\n  let begin = v2KMolblock.indexOf('M  RGP', 0) + 8;\n  let end = v2KMolblock.indexOf('\\n', begin);\n\n  // todo: maybe this part deserves a separate function\n  // there may be situation when the rgp information is distributed among\n  // multiple lines, this must be taken into account\n  const rgpStringParsed = v2KMolblock.substring(begin, end).replaceAll('  ', ' ').replaceAll('  ', ' ').split(' ');\n  const rgpData = rgpStringParsed.map((el) => parseInt(el));\n  // rgpData[0] is the number of R-groups\n  // the following code sets remFirst to node# to which RGP#1 is substituted\n\n  // todo: handle the exceptional case when there is not enough rgroups\n  // todo: handle the exceptional case when the order is different\n  const remFirst = rgpData[1]; // leftmost Rgroup node\n  const remLast = rgpData[rgpData.length - 2]; // rightmost Rgroup node\n  // if (remFirst === -1 || remLast === -1)\n  //   throw new Error('RGP parsing: first and last groups have wrong format');\n\n  // const remFirst = rgpData[2] === '1' ? parseInt(rgpData[1]) : parseInt(rgpData[3]);\n  // const remLast = rgpData[2] === '2' ? parseInt(rgpData[1]) : parseInt(rgpData[3]);\n\n  // todo: rename 'numbers'\n  const numbers = extractAtomAndBondCountsV3K(v3KMolblock);\n  let indexBonds = v3KMolblock.indexOf('M  V30 BEGIN BOND'); // V3000 bond block\n  indexBonds = v3KMolblock.indexOf('\\n', indexBonds);\n  begin = indexBonds;\n  end = indexBonds;\n\n  let first = 0; // todo: improve notation\n  let last = 0;\n  let remBondFirst = 0;\n  let remBondLast = 0;\n\n  // iterate over edges of the graph and find those\n  for (let j = 0; j < numbers.bondCount; j++) {\n    if (first === 0 || last === 0) {\n      begin = v3KMolblock.indexOf('V30', begin) + 4;\n      end = v3KMolblock.indexOf('\\n', begin);\n      const bondStringParsed = v3KMolblock.substring(begin, end).replaceAll('  ', ' ').replaceAll('  ', ' ').split(' ');\n      const bondData = bondStringParsed.map((el) => parseInt(el));\n\n      if (bondData[2] === remFirst) { // bondData[2] is the 1st node/atom of the bond\n        first = bondData[3]; // bondData[3] is the 2nd node/atom of the bond\n        remBondFirst = bondData[0]; // bondData[0] is the idx of the associated bond/edge\n      } else if (bondData[3] === remFirst) {\n        first = bondData[2];\n        remBondFirst = bondData[0];\n      } else if (bondData[2] === remLast) {\n        last = bondData[3];\n        remBondLast = bondData[0];\n      } else if (bondData[3] === remLast) {\n        last = bondData[2];\n        remBondLast = bondData[0];\n      }\n    }\n  }\n\n  return {first, last, remFirst, remLast, remBondFirst, remBondLast};\n}\n\n/* provide description */\nasync function rotateBackboneV3K(v3KMolblock: string, indices:any): Promise<string> {\n  const coordinates = extractAtomDataV3K(v3KMolblock); // naming?\n  const atomCount = coordinates.atomIndices.length;\n  const first = indices['first'];\n  const last = indices['last'];\n\n  const xCenter = (coordinates.x[last] + coordinates.x[first])/2;\n  const yCenter = (coordinates.y[last] + coordinates.y[first])/2;\n\n  //place to center\n  for (let i = 0; i < atomCount; i++) {\n    coordinates.x[i] -= xCenter;\n    coordinates.y[i] -= yCenter;\n  }\n\n  let angle = 0;\n  if (coordinates.x[first] === 0) { // both vertices are on OY\n    angle = coordinates.y[first] > coordinates.y[last] ? Math.PI/2 : -Math.PI/2;\n  } else if (coordinates.y[first] === 0) { // both vertices are on OX\n    angle = coordinates.x[first] > coordinates.x[last] ? Math.PI : 0; // v\n  } else {\n    const tangent = coordinates.y[first]/coordinates.x[first];\n    if (coordinates.x[first] < coordinates.x[last])\n      angle = tangent > 0 ? -Math.atan(tangent) : Math.atan(tangent);\n    else\n      angle = tangent > 0 ? Math.PI - Math.atan(tangent) : Math.atan(tangent) - Math.PI;\n  }\n\n  const cos = Math.cos(angle);\n  const sin = Math.sin(angle);\n\n  for (let i = 0; i < atomCount; i++) {\n    const xAdd = coordinates.x[i];\n    coordinates.x[i] = xAdd*cos - coordinates.y[i]*sin;\n    coordinates.y[i] = xAdd*sin + coordinates.y[i]*cos;\n  }\n\n  //place to right\n  const xShift = coordinates.x[first];\n  for (let i = 0; i < atomCount; i++)\n    coordinates.x[i] -= xShift;\n\n  //rewrite v3KMolblock\n  let index = v3KMolblock.indexOf('M  V30 BEGIN ATOM'); // V3000 index for atoms coordinates\n  index = v3KMolblock.indexOf('\\n', index);\n  let indexEnd = index;\n  for (let i = 0; i < atomCount; i++) {\n    index = v3KMolblock.indexOf('V30', index) + 4;\n    index = v3KMolblock.indexOf(' ', index) + 1;\n    index = v3KMolblock.indexOf(' ', index) + 1;\n    indexEnd = v3KMolblock.indexOf(' ', index) + 1;\n    indexEnd = v3KMolblock.indexOf(' ', indexEnd);\n\n    v3KMolblock = v3KMolblock.slice(0, index) +\n      coordinates.x[i] + ' ' + coordinates.y[i] +\n      v3KMolblock.slice(indexEnd);\n\n    index = v3KMolblock.indexOf('\\n', index) + 1;\n  }\n\n  return v3KMolblock;\n}\n\n/* provide description */\nfunction linkV3K(monomers: any[]): string {\n  let macroMolBlock = '\\nDatagrok macromolecule handler\\n\\n';\n  macroMolBlock += '  0  0  0  0  0  0            999 V3000\\n';\n  macroMolBlock += 'M  V30 BEGIN CTAB\\n';\n  let atomBlock = '';\n  let bondBlock = '';\n  // const collectionBlock = '';\n  // const collection: number [] = [];\n  let atomCount = 0;\n  let bondCount = 0;\n  let xShift = 0; // ?\n\n  for (let i = 0; i < monomers.length; i++) {\n    let v3KMolfile = monomers[i]['molfile'];\n    const first = monomers[i]['indices']['first'];\n    const last = monomers[i]['indices']['last'];\n    const remFirst = monomers[i]['indices']['remFirst'];\n    const remLast = monomers[i]['indices']['remLast'];\n    const remBondFirst = monomers[i]['indices']['remBondFirst'];\n    const remBondLast = monomers[i]['indices']['remBondLast'];\n    // todo: handle r-groups\n    v3KMolfile = v3KMolfile.replaceAll('(-\\nM  V30 ', '(')\n      .replaceAll('-\\nM  V30 ', '').replaceAll(' )', ')');\n    // todo: improve naming\n    const numbers = extractAtomAndBondCountsV3K(v3KMolfile);\n    const coordinates = extractAtomDataV3K(v3KMolfile);\n\n    let indexAtoms = v3KMolfile.indexOf('M  V30 BEGIN ATOM'); // V3000 atom block\n    indexAtoms = v3KMolfile.indexOf('\\n', indexAtoms);\n    let index = indexAtoms;\n    let indexEnd = indexAtoms;\n    const totalShift = xShift - coordinates.x[first - 1];\n\n    for (let j = 0; j < numbers.atomCount; j++) {\n      if (coordinates.atomIndices[j] != remFirst && coordinates.atomIndices[j] != remLast) { //|| i === 0) {\n        //rewrite atom number\n        index = v3KMolfile.indexOf('V30', index) + 4;\n        indexEnd = v3KMolfile.indexOf(' ', index);\n\n        let atomNumber = parseInt(v3KMolfile.substring(index, indexEnd));\n        atomNumber = (atomNumber > remFirst && atomNumber > remLast) ? atomNumber - 2 :\n          (atomNumber > remFirst || atomNumber > remLast) ? atomNumber - 1 : atomNumber;\n        atomNumber += atomCount;\n        v3KMolfile = v3KMolfile.slice(0, index) + atomNumber + v3KMolfile.slice(indexEnd);\n\n        //rewrite coordinates\n        index = v3KMolfile.indexOf(' ', index) + 1;\n        index = v3KMolfile.indexOf(' ', index) + 1;\n        indexEnd = v3KMolfile.indexOf(' ', index);\n\n        let coordinate = Math.round(10000*(parseFloat(v3KMolfile.substring(index, indexEnd)) + totalShift))/10000;\n        v3KMolfile = v3KMolfile.slice(0, index) + coordinate + v3KMolfile.slice(indexEnd);\n\n        index = v3KMolfile.indexOf(' ', index) + 1;\n        indexEnd = v3KMolfile.indexOf(' ', index);\n        coordinate = Math.round(10000*(parseFloat(v3KMolfile.substring(index, indexEnd))))/10000;\n        v3KMolfile = v3KMolfile.slice(0, index) + coordinate + v3KMolfile.slice(indexEnd);\n\n        index = v3KMolfile.indexOf('\\n', index) + 1;\n      } else {\n        index = v3KMolfile.indexOf('M  V30', index) - 1;\n        indexEnd = v3KMolfile.indexOf('\\n', index + 1);\n        v3KMolfile = v3KMolfile.slice(0, index) + v3KMolfile.slice(indexEnd);\n      }\n    }\n\n    const indexAtomsEnd = v3KMolfile.indexOf('M  V30 END ATOM');\n    atomBlock += v3KMolfile.substring(indexAtoms + 1, indexAtomsEnd);\n\n    let indexBonds = v3KMolfile.indexOf('M  V30 BEGIN BOND'); // V3000 index for bonds\n    indexBonds = v3KMolfile.indexOf('\\n', indexBonds);\n    index = indexBonds;\n    indexEnd = indexBonds;\n    let bondNumber = 0;\n\n    for (let j = 0; j < numbers.bondCount; j++) {\n      //rewrite bond number\n      index = v3KMolfile.indexOf('V30', index) + 4;\n      indexEnd = v3KMolfile.indexOf(' ', index);\n      bondNumber = parseInt(v3KMolfile.substring(index, indexEnd));\n\n      if (bondNumber === remBondFirst || bondNumber === remBondLast) {\n        indexEnd = v3KMolfile.indexOf('\\n', index) + 1;\n        index -=7;\n        v3KMolfile = v3KMolfile.slice(0, index) + v3KMolfile.slice(indexEnd);\n        continue;\n      }\n\n      bondNumber = (bondNumber > remBondFirst && bondNumber > remBondLast) ? bondNumber - 2 :\n        (bondNumber > remBondFirst || bondNumber > remBondLast) ? bondNumber - 1 : bondNumber;\n      bondNumber += bondCount;\n\n      v3KMolfile = v3KMolfile.slice(0, index) + bondNumber + v3KMolfile.slice(indexEnd);\n\n      //rewrite atom pair in bond\n      index = v3KMolfile.indexOf(' ', index) + 1;\n      index = v3KMolfile.indexOf(' ', index) + 1;\n      indexEnd = v3KMolfile.indexOf(' ', index);\n      let atomNumber = parseInt(v3KMolfile.substring(index, indexEnd));\n      atomNumber = (atomNumber > remFirst && atomNumber > remLast) ? atomNumber - 2 :\n        (atomNumber > remFirst || atomNumber > remLast) ? atomNumber - 1 : atomNumber;\n      atomNumber += atomCount;\n      v3KMolfile = v3KMolfile.slice(0, index) + atomNumber + v3KMolfile.slice(indexEnd);\n      index = v3KMolfile.indexOf(' ', index) + 1;\n      indexEnd = Math.min(v3KMolfile.indexOf('\\n', index), v3KMolfile.indexOf(' ', index));\n      atomNumber = parseInt(v3KMolfile.substring(index, indexEnd));\n      atomNumber = (atomNumber > remFirst && atomNumber > remLast) ? atomNumber - 2 :\n        (atomNumber > remFirst || atomNumber > remLast) ? atomNumber - 1 : atomNumber;\n      atomNumber += atomCount;\n      v3KMolfile = v3KMolfile.slice(0, index) + atomNumber + v3KMolfile.slice(indexEnd);\n\n      index = v3KMolfile.indexOf('\\n', index) + 1;\n    }\n\n    const indexBondEnd = v3KMolfile.indexOf('M  V30 END BOND');\n    bondBlock += v3KMolfile.substring(indexBonds + 1, indexBondEnd);\n    //let indexCollection = v3KMolfile.indexOf('M  V30 MDLV30/STEABS ATOMS=('); // V3000 index for collections\n\n    // while (indexCollection != -1) {\n    //   indexCollection += 28;\n    //   const collectionEnd = v3KMolfile.indexOf(')', indexCollection);\n    //   const collectionEntries = v3KMolfile.substring(indexCollection, collectionEnd).split(' ').slice(1);\n    //   collectionEntries.forEach((e: string) => {\n    //     collection.push(parseInt(e) + atomCount);\n    //   });\n    //   indexCollection = collectionEnd;\n    //   indexCollection = v3KMolfile.indexOf('M  V30 MDLV30/STEABS ATOMS=(', indexCollection);\n    // }\n\n    atomCount += numbers.atomCount - 2;\n    bondCount += numbers.bondCount - 2;\n    xShift += coordinates.x[last] - coordinates.x[first] + 1;\n\n    if (i === monomers.length -1) {\n      atomCount++;\n      const shift = xShift + 0.2;\n      atomBlock += 'M  V30 ' + atomCount + ' O ' + shift + ' 0 0.000000 0\\n';\n    }\n    bondCount++;\n    if (i === monomers.length -1) {\n      const rightTerminal = (last > remFirst && last > remLast) ? last + atomCount - (numbers.atomCount - 2) - 3:\n        (last > remFirst || last > remLast) ? last + atomCount - (numbers.atomCount - 2) - 2 :\n          last + atomCount - (numbers.atomCount - 2) - 1;\n      bondBlock += 'M  V30 ' + bondCount + ' 1 ' + rightTerminal + ' ' + atomCount + '\\n';\n    } else {\n      const rightTerminal = (last > remFirst && last > remLast) ? last + atomCount - (numbers.atomCount - 2) - 2:\n        (last > remFirst || last > remLast) ? last + atomCount - (numbers.atomCount - 2) - 1 :\n          last + atomCount - (numbers.atomCount - 2);\n\n      const next = monomers[i + 1]['indices'];\n      const nextFirst = next['first'];\n      const nextRemFirst = next['remFirst'];\n      const nextRemLast = next['remLast'];\n\n      const leftTerminal = (nextFirst > nextRemFirst && nextFirst > nextRemLast) ? nextFirst + atomCount - 2 :\n        (nextFirst > nextRemFirst || nextFirst > nextRemLast) ? nextFirst + atomCount - 1 :\n          nextFirst + atomCount;\n\n      bondBlock += 'M  V30 ' + bondCount + ' 1 ' + rightTerminal + ' ' + leftTerminal + '\\n';\n    }\n  }\n\n  // const entries = 4;\n  // const collNumber = Math.ceil(collection.length / entries);\n  // collectionBlock += 'M  V30 MDLV30/STEABS ATOMS=(' + collection.length + ' -\\n';\n  // for (let i = 0; i < collNumber; i++) {\n  //   collectionBlock += 'M  V30 ';\n  //   const entriesCurrent = i + 1 === collNumber ? collection.length - (collNumber - 1)*entries : entries;\n  //   for (let j = 0; j < entriesCurrent; j++) {\n  //     collectionBlock += (j + 1 === entriesCurrent) ?\n  //       (i === collNumber - 1 ? collection[entries*i + j] + ')\\n' : collection[entries*i + j] + ' -\\n') :\n  //       collection[entries*i + j] + ' ';\n  //   }\n  // }\n\n  //generate file\n  macroMolBlock += 'M  V30 COUNTS ' + atomCount + ' ' + bondCount + ' 0 0 0\\n';\n  macroMolBlock += 'M  V30 BEGIN ATOM\\n';\n  macroMolBlock += atomBlock;\n  macroMolBlock += 'M  V30 END ATOM\\n';\n  macroMolBlock += 'M  V30 BEGIN BOND\\n';\n  macroMolBlock += bondBlock;\n  macroMolBlock += 'M  V30 END BOND\\n';\n  //macroMolBlock += 'M  V30 BEGIN COLLECTION\\n';\n  //macroMolBlock += collectionBlock;\n  //macroMolBlock += 'M  V30 END COLLECTION\\n';\n  macroMolBlock += 'M  V30 END CTAB\\n';\n  macroMolBlock += 'M  END\\n';\n\n  return macroMolBlock;\n}\n\nexport async function getMacroMol(monomers: any[][]): Promise<string[]> {\n  const result: string[] = [];\n  const moduleRdkit = await grok.functions.call('Chem:getRdKitModule');\n  for (let i = 0; i < monomers.length; i++) {\n    for (let j = 0; j < monomers[i].length; j++) {\n      const molObj = moduleRdkit.get_mol(monomers[i][j]['molfile']);\n      const v3KMolblock = molObj.get_v3Kmolblock();\n      const indices = getIndices(monomers[i][j]['molfile'], v3KMolblock);\n      monomers[i][j]['indices'] = indices;\n\n      // a new molfile for 'rotated' is obtained in v3k format\n      monomers[i][j]['molfile'] = await rotateBackboneV3K(v3KMolblock, indices);\n      // monomers[i][j]['molfile'] = v3KMolblock;\n      molObj?.delete();\n    }\n    // seemingly, at this stage the bond is reconstructed\n    result.push(linkV3K(monomers[i]));\n    // result.push(monomers[i][0]['molfile']);\n  }\n\n  return result;\n}\n"]}
|
|
@@ -0,0 +1,48 @@
|
|
|
1
|
+
export declare const HELM_CORE_LIB_FILENAME = "/data/HELMCoreLibrary.json";
|
|
2
|
+
export declare const enum HELM_FIELDS {
|
|
3
|
+
MONOMER_TYPE = "monomerType",
|
|
4
|
+
SMILES = "smiles",
|
|
5
|
+
NAME = "name",
|
|
6
|
+
AUTHOR = "author",
|
|
7
|
+
MOLFILE = "molfile",
|
|
8
|
+
NATURAL_ANALOG = "naturalAnalog",
|
|
9
|
+
RGROUPS = "rgroups",
|
|
10
|
+
CREATE_DATE = "createDate",
|
|
11
|
+
ID = "id",
|
|
12
|
+
POLYMER_TYPE = "polymerType",
|
|
13
|
+
SYMBOL = "symbol"
|
|
14
|
+
}
|
|
15
|
+
export declare const enum RGROUP_FIELDS {
|
|
16
|
+
CAP_GROUP_SMILES = "capGroupSmiles",
|
|
17
|
+
CAP_GROUP_SMILES_UPPERCASE = "capGroupSMILES",
|
|
18
|
+
ALTER_ID = "alternateId",
|
|
19
|
+
CAP_GROUP_NAME = "capGroupName",
|
|
20
|
+
LABEL = "label"
|
|
21
|
+
}
|
|
22
|
+
export declare const enum HELM_POLYMER_TYPE {
|
|
23
|
+
PEPTIDE = "PEPTIDE",
|
|
24
|
+
RNA = "RNA"
|
|
25
|
+
}
|
|
26
|
+
export declare const enum HELM_MONOMER_TYPE {
|
|
27
|
+
BACKBONE = "Backbone",
|
|
28
|
+
TERMINAL = "Terminal",
|
|
29
|
+
BRANCH = "Branch"
|
|
30
|
+
}
|
|
31
|
+
export declare const HELM_CORE_FIELDS: HELM_FIELDS[];
|
|
32
|
+
export declare const SDF_MONOMER_NAME = "MonomerName";
|
|
33
|
+
export declare const jsonSdfMonomerLibDict: {
|
|
34
|
+
monomerType: null;
|
|
35
|
+
smiles: null;
|
|
36
|
+
name: string;
|
|
37
|
+
author: null;
|
|
38
|
+
molfile: string;
|
|
39
|
+
naturalAnalog: string;
|
|
40
|
+
rgroups: string;
|
|
41
|
+
createDate: null;
|
|
42
|
+
id: null;
|
|
43
|
+
polymerType: string;
|
|
44
|
+
symbol: string;
|
|
45
|
+
};
|
|
46
|
+
export declare const MONOMER_ENCODE_MIN = 256;
|
|
47
|
+
export declare const MONOMER_ENCODE_MAX = 1034;
|
|
48
|
+
//# sourceMappingURL=const.d.ts.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"const.d.ts","sourceRoot":"","sources":["const.ts"],"names":[],"mappings":"AAIA,eAAO,MAAM,sBAAsB,+BAA+B,CAAC;AAEnE,0BAAkB,WAAW;IAC3B,YAAY,gBAAgB;IAC5B,MAAM,WAAW;IACjB,IAAI,SAAS;IACb,MAAM,WAAW;IACjB,OAAO,YAAY;IACnB,cAAc,kBAAkB;IAChC,OAAO,YAAY;IACnB,WAAW,eAAe;IAC1B,EAAE,OAAO;IACT,YAAY,gBAAgB;IAC5B,MAAM,WAAW;CAClB;AAGD,0BAAkB,aAAa;IAC7B,gBAAgB,mBAAmB;IACnC,0BAA0B,mBAAmB;IAC7C,QAAQ,gBAAgB;IACxB,cAAc,iBAAiB;IAC/B,KAAK,UAAU;CAChB;AAGD,0BAAkB,iBAAiB;IACjC,OAAO,YAAY;IACnB,GAAG,QAAQ;CACZ;AAED,0BAAkB,iBAAiB;IACjC,QAAQ,aAAa;IACrB,QAAQ,aAAa;IACrB,MAAM,WAAW;CAClB;AAGD,eAAO,MAAM,gBAAgB,eAO5B,CAAC;AAEF,eAAO,MAAM,gBAAgB,gBAAgB,CAAC;AAG9C,eAAO,MAAM,qBAAqB;;;;;;;;;;;;CAYjC,CAAC;AAGF,eAAO,MAAM,kBAAkB,MAAQ,CAAC;AACxC,eAAO,MAAM,kBAAkB,OAAQ,CAAC"}
|
|
@@ -0,0 +1,29 @@
|
|
|
1
|
+
export const HELM_CORE_LIB_FILENAME = '/data/HELMCoreLibrary.json';
|
|
2
|
+
// core fields of HELM library object used in toAtomicLevel function
|
|
3
|
+
export const HELM_CORE_FIELDS = [
|
|
4
|
+
"symbol" /* HELM_FIELDS.SYMBOL */,
|
|
5
|
+
"molfile" /* HELM_FIELDS.MOLFILE */,
|
|
6
|
+
"rgroups" /* HELM_FIELDS.RGROUPS */,
|
|
7
|
+
"name" /* HELM_FIELDS.NAME */,
|
|
8
|
+
// HELM_FIELDS.MONOMER_TYPE, // add if terminal monomers for PEPTIDEs to be
|
|
9
|
+
// supported
|
|
10
|
+
];
|
|
11
|
+
export const SDF_MONOMER_NAME = 'MonomerName';
|
|
12
|
+
// todo: ideally, keys should be expressed via constants
|
|
13
|
+
export const jsonSdfMonomerLibDict = {
|
|
14
|
+
'monomerType': null,
|
|
15
|
+
'smiles': null,
|
|
16
|
+
'name': 'MonomerName',
|
|
17
|
+
'author': null,
|
|
18
|
+
'molfile': 'molecule',
|
|
19
|
+
'naturalAnalog': 'MonomerNaturalAnalogCode',
|
|
20
|
+
'rgroups': 'MonomerCaps',
|
|
21
|
+
'createDate': null,
|
|
22
|
+
'id': null,
|
|
23
|
+
'polymerType': 'MonomerType',
|
|
24
|
+
'symbol': 'MonomerCode'
|
|
25
|
+
};
|
|
26
|
+
// range of hex nubers used in PepSea library to endode monomers
|
|
27
|
+
export const MONOMER_ENCODE_MIN = 0x100;
|
|
28
|
+
export const MONOMER_ENCODE_MAX = 0x40A;
|
|
29
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiY29uc3QuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyJjb25zdC50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFJQSxNQUFNLENBQUMsTUFBTSxzQkFBc0IsR0FBRyw0QkFBNEIsQ0FBQztBQXFDbkUsb0VBQW9FO0FBQ3BFLE1BQU0sQ0FBQyxNQUFNLGdCQUFnQixHQUFHOzs7OztJQUs5QiwyRUFBMkU7SUFDM0UsWUFBWTtDQUNiLENBQUM7QUFFRixNQUFNLENBQUMsTUFBTSxnQkFBZ0IsR0FBRyxhQUFhLENBQUM7QUFFOUMsd0RBQXdEO0FBQ3hELE1BQU0sQ0FBQyxNQUFNLHFCQUFxQixHQUFHO0lBQ25DLGFBQWEsRUFBRSxJQUFJO0lBQ25CLFFBQVEsRUFBRSxJQUFJO0lBQ2QsTUFBTSxFQUFFLGFBQWE7SUFDckIsUUFBUSxFQUFFLElBQUk7SUFDZCxTQUFTLEVBQUUsVUFBVTtJQUNyQixlQUFlLEVBQUUsMEJBQTBCO0lBQzNDLFNBQVMsRUFBRSxhQUFhO0lBQ3hCLFlBQVksRUFBRSxJQUFJO0lBQ2xCLElBQUksRUFBRSxJQUFJO0lBQ1YsYUFBYSxFQUFFLGFBQWE7SUFDNUIsUUFBUSxFQUFFLGFBQWE7Q0FDeEIsQ0FBQztBQUVGLGdFQUFnRTtBQUNoRSxNQUFNLENBQUMsTUFBTSxrQkFBa0IsR0FBRyxLQUFLLENBQUM7QUFDeEMsTUFBTSxDQUFDLE1BQU0sa0JBQWtCLEdBQUcsS0FBSyxDQUFDIiwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0ICogYXMgdWkgZnJvbSAnZGF0YWdyb2stYXBpL3VpJztcbmltcG9ydCAqIGFzIGdyb2sgZnJvbSAnZGF0YWdyb2stYXBpL2dyb2snO1xuaW1wb3J0ICogYXMgREcgZnJvbSAnZGF0YWdyb2stYXBpL2RnJztcblxuZXhwb3J0IGNvbnN0IEhFTE1fQ09SRV9MSUJfRklMRU5BTUUgPSAnL2RhdGEvSEVMTUNvcmVMaWJyYXJ5Lmpzb24nO1xuXG5leHBvcnQgY29uc3QgZW51bSBIRUxNX0ZJRUxEUyB7XG4gIE1PTk9NRVJfVFlQRSA9ICdtb25vbWVyVHlwZScsXG4gIFNNSUxFUyA9ICdzbWlsZXMnLFxuICBOQU1FID0gJ25hbWUnLFxuICBBVVRIT1IgPSAnYXV0aG9yJyxcbiAgTU9MRklMRSA9ICdtb2xmaWxlJyxcbiAgTkFUVVJBTF9BTkFMT0cgPSAnbmF0dXJhbEFuYWxvZycsXG4gIFJHUk9VUFMgPSAncmdyb3VwcycsXG4gIENSRUFURV9EQVRFID0gJ2NyZWF0ZURhdGUnLFxuICBJRCA9ICdpZCcsXG4gIFBPTFlNRVJfVFlQRSA9ICdwb2x5bWVyVHlwZScsXG4gIFNZTUJPTCA9ICdzeW1ib2wnXG59XG5cbi8vIGZpZWxkcyBvZiBcInJncm91cHNcIiBzdWItb2JqZWN0IGluIEhFTE0gbGlicmFyeVxuZXhwb3J0IGNvbnN0IGVudW0gUkdST1VQX0ZJRUxEUyB7XG4gIENBUF9HUk9VUF9TTUlMRVMgPSAnY2FwR3JvdXBTbWlsZXMnLFxuICBDQVBfR1JPVVBfU01JTEVTX1VQUEVSQ0FTRSA9ICdjYXBHcm91cFNNSUxFUycsIC8vIGFsYXMsIGJvdGggdmFyaWFudHMgY29leGlzdFxuICBBTFRFUl9JRCA9ICdhbHRlcm5hdGVJZCcsXG4gIENBUF9HUk9VUF9OQU1FID0gJ2NhcEdyb3VwTmFtZScsXG4gIExBQkVMID0gJ2xhYmVsJyxcbn1cblxuLy8gcG9zc2libGUgdmFsdWVzIG9mIHBvbHltZXJzXG5leHBvcnQgY29uc3QgZW51bSBIRUxNX1BPTFlNRVJfVFlQRSB7XG4gIFBFUFRJREUgPSAnUEVQVElERScsXG4gIFJOQSA9ICdSTkEnLFxufVxuXG5leHBvcnQgY29uc3QgZW51bSBIRUxNX01PTk9NRVJfVFlQRSB7XG4gIEJBQ0tCT05FID0gJ0JhY2tib25lJyxcbiAgVEVSTUlOQUwgPSAnVGVybWluYWwnLFxuICBCUkFOQ0ggPSAnQnJhbmNoJyxcbn1cblxuLy8gY29yZSBmaWVsZHMgb2YgSEVMTSBsaWJyYXJ5IG9iamVjdCB1c2VkIGluIHRvQXRvbWljTGV2ZWwgZnVuY3Rpb25cbmV4cG9ydCBjb25zdCBIRUxNX0NPUkVfRklFTERTID0gW1xuICBIRUxNX0ZJRUxEUy5TWU1CT0wsXG4gIEhFTE1fRklFTERTLk1PTEZJTEUsXG4gIEhFTE1fRklFTERTLlJHUk9VUFMsXG4gIEhFTE1fRklFTERTLk5BTUUsXG4gIC8vIEhFTE1fRklFTERTLk1PTk9NRVJfVFlQRSwgLy8gYWRkIGlmIHRlcm1pbmFsIG1vbm9tZXJzIGZvciBQRVBUSURFcyB0byBiZVxuICAvLyBzdXBwb3J0ZWRcbl07XG5cbmV4cG9ydCBjb25zdCBTREZfTU9OT01FUl9OQU1FID0gJ01vbm9tZXJOYW1lJztcblxuLy8gdG9kbzogaWRlYWxseSwga2V5cyBzaG91bGQgYmUgZXhwcmVzc2VkIHZpYSBjb25zdGFudHNcbmV4cG9ydCBjb25zdCBqc29uU2RmTW9ub21lckxpYkRpY3QgPSB7XG4gICdtb25vbWVyVHlwZSc6IG51bGwsXG4gICdzbWlsZXMnOiBudWxsLFxuICAnbmFtZSc6ICdNb25vbWVyTmFtZScsXG4gICdhdXRob3InOiBudWxsLFxuICAnbW9sZmlsZSc6ICdtb2xlY3VsZScsXG4gICduYXR1cmFsQW5hbG9nJzogJ01vbm9tZXJOYXR1cmFsQW5hbG9nQ29kZScsXG4gICdyZ3JvdXBzJzogJ01vbm9tZXJDYXBzJyxcbiAgJ2NyZWF0ZURhdGUnOiBudWxsLFxuICAnaWQnOiBudWxsLFxuICAncG9seW1lclR5cGUnOiAnTW9ub21lclR5cGUnLFxuICAnc3ltYm9sJzogJ01vbm9tZXJDb2RlJ1xufTtcblxuLy8gcmFuZ2Ugb2YgaGV4IG51YmVycyB1c2VkIGluIFBlcFNlYSBsaWJyYXJ5IHRvIGVuZG9kZSBtb25vbWVyc1xuZXhwb3J0IGNvbnN0IE1PTk9NRVJfRU5DT0RFX01JTiA9IDB4MTAwO1xuZXhwb3J0IGNvbnN0IE1PTk9NRVJfRU5DT0RFX01BWCA9IDB4NDBBO1xuIl19
|