001/*
002 *                    BioJava development code
003 *
004 * This code may be freely distributed and modified under the
005 * terms of the GNU Lesser General Public Licence.  This should
006 * be distributed with the code.  If you do not have a copy,
007 * see:
008 *
009 *      http://www.gnu.org/copyleft/lesser.html
010 *
011 * Copyright for this code is held jointly by the individual
012 * authors.  These should be listed in @author doc comments.
013 *
014 * For more information on the BioJava project and its aims,
015 * or to join the biojava-l mailing list, visit the home page
016 * at:
017 *
018 *      http://www.biojava.org/
019 *
020 */
021package org.biojava.nbio.structure.xtal;
022
023
024import org.biojava.nbio.structure.*;
025import org.biojava.nbio.structure.contact.AtomContactSet;
026import org.biojava.nbio.structure.contact.StructureInterface;
027import org.biojava.nbio.structure.contact.StructureInterfaceList;
028import org.slf4j.Logger;
029import org.slf4j.LoggerFactory;
030
031import javax.vecmath.Matrix4d;
032import javax.vecmath.Point3i;
033import javax.vecmath.Vector3d;
034import java.util.*;
035
036
037/**
038 * A class containing methods to find interfaces in a given crystallographic Structure by
039 * reconstructing the crystal lattice through application of symmetry operators
040 *
041 * @author Jose Duarte
042 *
043 */
044
045public class CrystalBuilder {
046
047        public static final String NCS_CHAINID_SUFFIX_CHAR = "n";
048
049        // Default number of cell neighbors to try in interface search (in 3 directions of space).
050        // In the search, only bounding box overlaps are tried, thus there's not so much overhead in adding
051        // more cells. We actually tested it and using numCells from 1 to 10 didn't change runtimes at all.
052        // Examples with interfaces in distant neighbor cells:
053        //   2nd neighbors: 3hz3, 1wqj, 2de3, 1jcd
054        //   3rd neighbors: 3bd3, 1men, 2gkp, 1wui
055        //   5th neighbors: 2ahf, 2h2z
056        //   6th neighbors: 1was (in fact interfaces appear only at 5th neighbors for it)
057        // Maybe this could be avoided by previously translating the given molecule to the first cell,
058        // BUT! some bona fide cases exist, e.g. 2d3e: it is properly placed at the origin but the molecule
059        // is enormously long in comparison with the dimensions of the unit cell, some interfaces come at the 7th neighbor.
060        // After a scan of the whole PDB (Oct 2013) using numCells=50, the highest one was 4jgc with
061        // interfaces up to the 11th neighbor. Other high ones (9th neighbors) are 4jbm and 4k3t.
062        // We set the default value to 20 to be on the safe side. Runtime does not seem to be affected at all - JD 2020-01-12
063        // Some good examples in this posting in CCP4: https://www.jiscmail.ac.uk/cgi-bin/webadmin?A2=CCP4BB;45b2755d.2001
064        // in any case the 5m3h example in the posting seems to have contacts only up to the 11th neighbor.
065        public static final int DEF_NUM_CELLS = 20;
066
067        /**
068         * Default maximum distance between two chains to be considered an interface.
069         * @see #getUniqueInterfaces(double)
070         */
071        public static final double DEFAULT_INTERFACE_DISTANCE_CUTOFF = 5.5;
072
073        public static final Matrix4d IDENTITY = new Matrix4d(1,0,0,0, 0,1,0,0, 0,0,1,0, 0,0,0,1);
074
075
076        /**
077         * Whether to consider HETATOMs in contact calculations
078         */
079        private static final boolean INCLUDE_HETATOMS = true;
080
081        private Structure structure;
082        private PDBCrystallographicInfo crystallographicInfo;
083        private int numPolyChainsAu;
084        private int numOperatorsSg;
085        private Map<String,Matrix4d> chainNcsOps = null;
086        private Map<String,String> chainOrigNames = null;
087
088        private static final Logger logger = LoggerFactory.getLogger(CrystalBuilder.class);
089
090        private int numCells;
091
092        private ArrayList<CrystalTransform> visitedCrystalTransforms;
093        private Map<String,Map<Matrix4d,StructureInterface>> visitedNcsChainPairs = null;
094
095        private boolean searchBeyondAU;
096        private Matrix4d[] ops;
097
098        /**
099         * Special constructor for NCS-aware CrystalBuilder.
100         * The output list of interfaces will be pre-clustered by NCS-equivalence.
101         * Run {@link CrystalBuilder#expandNcsOps(Structure, Map, Map)} first to extend the AU
102         * and get the equivalence information.
103         * @param structure
104         *          NCS-extended structure
105         * @param chainOrigNames
106         *          chain names mapped to the original chain names (pre-NCS extension)
107         * @param chainNcsOps
108         *          chain names mapped to the ncs operators that was used to generate them
109         * @since 5.0.0
110         */
111        public CrystalBuilder(Structure structure, Map<String,String> chainOrigNames, Map<String,Matrix4d> chainNcsOps) {
112                this(structure);
113                this.chainOrigNames = chainOrigNames;
114                this.chainNcsOps = chainNcsOps;
115        }
116
117        public CrystalBuilder(Structure structure) {
118                this.structure = structure;
119                this.crystallographicInfo = structure.getCrystallographicInfo();
120                this.numPolyChainsAu = structure.getPolyChains().size();
121
122                this.searchBeyondAU = false;
123                if (structure.isCrystallographic()) {
124
125                        this.searchBeyondAU = true;
126
127                        // we need to check space group not null for the cases where the entry is crystallographic but
128                        // the space group is not a standard one recognized by biojava, e.g. 1mnk (SG: 'I 21')
129                        if (this.crystallographicInfo.isNonStandardSg()) {
130                                logger.warn("Space group is non-standard, will only calculate asymmetric unit interfaces.");
131                                this.searchBeyondAU = false;
132                        }
133
134                        // just in case we still check for space group null (a user pdb file could potentially be crystallographic and have no space group)
135                        if (this.crystallographicInfo.getSpaceGroup() == null) {
136                                logger.warn("Space group is null, will only calculate asymmetric unit interfaces.");
137                                this.searchBeyondAU = false;
138                        }
139
140                        // we need to check crystal cell not null for the rare cases where the entry is crystallographic but
141                        // the crystal cell is not given, e.g. 2i68, 2xkm, 4bpq
142                        if (this.crystallographicInfo.getCrystalCell() == null) {
143                                logger.warn("Could not find a crystal cell definition, will only calculate asymmetric unit interfaces.");
144                                this.searchBeyondAU = false;
145                        }
146
147                        // check for cases like 4hhb that are in a non-standard coordinate frame convention, see https://github.com/eppic-team/owl/issues/4
148                        if (this.crystallographicInfo.isNonStandardCoordFrameConvention()) {
149                                logger.warn("Non-standard coordinate frame convention, will only calculate asymmetric unit interfaces.");
150                                this.searchBeyondAU = false;
151                        }
152                }
153
154                if (this.searchBeyondAU) {
155                        // explore the crystal
156                        this.numOperatorsSg = this.crystallographicInfo.getSpaceGroup().getMultiplicity();
157                        this.ops = this.crystallographicInfo.getTransformationsOrthonormal();
158                } else {
159                        // look for contacts within structure as given
160                        this.numOperatorsSg = 1;
161                        this.ops = new Matrix4d[1];
162                        this.ops[0] = new Matrix4d(IDENTITY);
163                }
164
165                this.numCells = DEF_NUM_CELLS;
166
167        }
168
169
170        /**
171         * @return true if this CrystalBuilder is NCS-aware.
172         * @since 5.0.0
173         */
174        public boolean hasNcsOps() {
175                return chainNcsOps != null;
176        }
177
178        /**
179         * Set the number of neighboring crystal cells that will be used in the search for contacts
180         * @param numCells
181         */
182        public void setNumCells(int numCells) {
183                this.numCells = numCells;
184        }
185
186        private void initialiseVisited() {
187                visitedCrystalTransforms = new ArrayList<>();
188                if(this.hasNcsOps()) {
189                        visitedNcsChainPairs = new HashMap<>();
190                }
191        }
192
193        /**
194         * Returns the list of unique interfaces that the given Structure has upon
195         * generation of all crystal symmetry mates. An interface is defined as any pair of chains
196         * that contact, i.e. for which there is at least a pair of atoms (one from each chain) within
197         * the default cutoff distance.
198         * @return
199         * @see #DEFAULT_INTERFACE_DISTANCE_CUTOFF
200         */
201        public StructureInterfaceList getUniqueInterfaces() {
202                return getUniqueInterfaces(DEFAULT_INTERFACE_DISTANCE_CUTOFF);
203        }
204
205        /**
206         * Returns the list of unique interfaces that the given Structure has upon
207         * generation of all crystal symmetry mates. An interface is defined as any pair of chains
208         * that contact, i.e. for which there is at least a pair of atoms (one from each chain) within
209         * the given cutoff distance.
210         * @param cutoff the distance cutoff for 2 chains to be considered in contact
211         * @return
212         */
213        public StructureInterfaceList getUniqueInterfaces(double cutoff) {
214
215
216                StructureInterfaceList set = new StructureInterfaceList();
217
218                // certain structures in the PDB are not macromolecules (contain no polymeric chains at all), e.g. 1ao2
219                // with the current mmCIF parsing, those will be empty since purely non-polymeric chains are removed
220                // see commit e9562781f23da0ebf3547146a307d7edd5741090
221                if (numPolyChainsAu==0) {
222                        logger.warn("No chains present in the structure! No interfaces will be calculated");
223                        return set;
224                }
225
226                // pass the chainOrigNames map in NCS case so that StructureInterfaceList can deal with original to NCS chain names conversion
227                if (chainOrigNames!=null) {
228                        set.setChainOrigNamesMap(chainOrigNames);
229                }
230
231                // initialising the visited ArrayList for keeping track of symmetry redundancy
232                initialiseVisited();
233
234
235
236                // the isCrystallographic() condition covers 3 cases:
237                // a) entries with expMethod X-RAY/other diffraction and defined crystalCell (most usual case)
238                // b) entries with expMethod null but defined crystalCell (e.g. PDB file with CRYST1 record but no expMethod annotation)
239                // c) entries with expMethod not X-RAY (e.g. NMR) and defined crystalCell (NMR entries do have a dummy CRYST1 record "1 1 1 90 90 90 P1")
240                // d) isCrystallographic will be false if the structure is crystallographic but the space group was not recognized
241
242
243                calcInterfacesCrystal(set, cutoff);
244
245                return set;
246        }
247
248        /**
249         * Calculate interfaces between original asymmetric unit and neighboring
250         * whole unit cells, including the original full unit cell i.e. i=0,j=0,k=0
251         * @param set
252         * @param cutoff
253         */
254        private void calcInterfacesCrystal(StructureInterfaceList set, double cutoff) {
255
256
257                // initialising debugging vars
258                long start = -1;
259                long end = -1;
260                int trialCount = 0;
261                int skippedRedundant = 0;
262                int skippedAUsNoOverlap = 0;
263                int skippedChainsNoOverlap = 0;
264                int skippedSelfEquivalent = 0;
265
266                // The bounding boxes of all AUs of the unit cell
267                UnitCellBoundingBox bbGrid = new UnitCellBoundingBox(numOperatorsSg, numPolyChainsAu);;
268                // we calculate all the bounds of each of the asym units, those will then be reused and translated
269                bbGrid.setBbs(structure, ops, INCLUDE_HETATOMS);
270
271
272                // if not crystallographic there's no search to do in other cells, only chains within "AU" will be checked
273                if (!searchBeyondAU) numCells = 0;
274
275                boolean verbose = logger.isDebugEnabled();
276
277                if (verbose) {
278                        trialCount = 0;
279                        start= System.currentTimeMillis();
280                        int neighbors = (2*numCells+1)*(2*numCells+1)*(2*numCells+1)-1;
281                        int auTrials = (numPolyChainsAu*(numPolyChainsAu-1))/2;
282                        int trials = numPolyChainsAu*numOperatorsSg*numPolyChainsAu*neighbors;
283                        logger.debug("Chain clash trials within original AU: {}", auTrials);
284                        logger.debug(
285                                        "Chain clash trials between the original AU and the neighbouring "+neighbors+
286                                        " whole unit cells ("+numCells+" neighbours)" +
287                                        "(2x"+numPolyChainsAu+"chains x "+numOperatorsSg+"AUs x "+neighbors+"cells) : "+trials);
288                        logger.debug("Total trials: {}", (auTrials+trials));
289                }
290
291                List<Chain> polyChains = structure.getPolyChains();
292
293                for (int a=-numCells;a<=numCells;a++) {
294                        for (int b=-numCells;b<=numCells;b++) {
295                                for (int c=-numCells;c<=numCells;c++) {
296
297                                        Point3i trans = new Point3i(a,b,c);
298                                        Vector3d transOrth = new Vector3d(a,b,c);
299                                        if (a!=0 || b!=0 || c!=0) {
300                                                // we avoid doing the transformation for 0,0,0 (in case it's not crystallographic)
301                                                this.crystallographicInfo.getCrystalCell().transfToOrthonormal(transOrth);
302                                        }
303
304                                        UnitCellBoundingBox bbGridTrans = bbGrid.getTranslatedBbs(transOrth);
305
306                                        for (int n=0;n<numOperatorsSg;n++) {
307
308                                                // short-cut strategies
309                                                // 1) we skip first of all if the bounding boxes of the AUs don't overlap
310                                                if (!bbGrid.getAuBoundingBox(0).overlaps(bbGridTrans.getAuBoundingBox(n), cutoff)) {
311                                                        skippedAUsNoOverlap++;
312                                                        continue;
313                                                }
314
315                                                // 2) we check if we didn't already see its equivalent symmetry operator partner
316                                                CrystalTransform tt = new CrystalTransform(this.crystallographicInfo.getSpaceGroup(), n);
317                                                tt.translate(trans);
318                                                if (isRedundantTransform(tt)) {
319                                                        skippedRedundant++;
320                                                        continue;
321                                                }
322                                                addVisitedTransform(tt);
323
324
325                                                boolean selfEquivalent = false;
326
327                                                // 3) an operator can be "self redundant" if it is the inverse of itself (involutory, e.g. all pure 2-folds with no translation)
328                                                if (tt.isEquivalent(tt)) {
329                                                        logger.debug("Transform {} is equivalent to itself, will skip half of i-chains to j-chains comparisons", tt.toString());
330                                                        // in this case we can't skip the operator, but we can skip half of the matrix comparisons e.g. j>i
331                                                        // we set a flag and do that within the loop below
332                                                        selfEquivalent = true;
333                                                }
334
335                                                StringBuilder builder = null;
336                                                if (verbose) builder = new StringBuilder(String.valueOf(tt)).append(" ");
337
338                                                // Now that we know that boxes overlap and operator is not redundant, we have to go to the details
339                                                int contactsFound = 0;
340
341                                                for (int j=0;j<numPolyChainsAu;j++) {
342
343                                                        for (int i=0;i<numPolyChainsAu;i++) { // we only have to compare the original asymmetric unit to every full cell around
344
345                                                                if(selfEquivalent && (j>i)) {
346                                                                        // in case of self equivalency of the operator we can safely skip half of the matrix
347                                                                        skippedSelfEquivalent++;
348                                                                        continue;
349                                                                }
350                                                                // special case of original AU, we don't compare a chain to itself
351                                                                if (n==0 && a==0 && b==0 && c==0 && i==j) continue;
352
353                                                                // before calculating the AtomContactSet we check for overlap, then we save putting atoms into the grid
354                                                                if (!bbGrid.getChainBoundingBox(0,i).overlaps(bbGridTrans.getChainBoundingBox(n,j),cutoff)) {
355                                                                        skippedChainsNoOverlap++;
356                                                                        if (verbose) {
357                                                                                builder.append(".");
358                                                                        }
359                                                                        continue;
360                                                                }
361
362                                                                trialCount++;
363
364                                                                // finally we've gone through all short-cuts and the 2 chains seem to be close enough:
365                                                                // we do the calculation of contacts
366                                                                Chain chaini = polyChains.get(i);
367                                                                Chain chainj = polyChains.get(j);
368
369                                                                if (n!=0 || a!=0 || b!=0 || c!=0) {
370                                                                        Matrix4d mJCryst = new Matrix4d(ops[n]);
371                                                                        translate(mJCryst, transOrth);
372                                                                        chainj = (Chain)chainj.clone();
373                                                                        Calc.transform(chainj,mJCryst);
374                                                                }
375
376                                                                StructureInterface interf = calcContacts(chaini, chainj, cutoff, tt, builder);
377                                                                if (interf == null) {
378                                                                        continue;
379                                                                }
380
381                                                                contactsFound++;
382                                                                if(this.hasNcsOps()) {
383                                                                        StructureInterface interfNcsRef = findNcsRef(interf);
384                                                                        set.addNcsEquivalent(interf,interfNcsRef);
385                                                                } else {
386                                                                        set.add(interf);
387                                                                }
388                                                        }
389                                                }
390
391                                                if( verbose ) {
392                                                        if (a==0 && b==0 && c==0 && n==0)
393                                                                builder.append(" "+contactsFound+"("+(numPolyChainsAu*(numPolyChainsAu-1))/2+")");
394                                                        else if (selfEquivalent)
395                                                                builder.append(" "+contactsFound+"("+(numPolyChainsAu*(numPolyChainsAu+1))/2+")");
396                                                        else
397                                                                builder.append(" "+contactsFound+"("+numPolyChainsAu*numPolyChainsAu+")");
398
399                                                        logger.debug(builder.toString());
400                                                }
401                                        }
402                                }
403                        }
404                }
405
406                end = System.currentTimeMillis();
407                logger.debug("\n{} chain-chain clash trials done. Time {}{}s", trialCount, (end-start), 1000);
408                logger.debug("  skipped (not overlapping AUs)       : {}", skippedAUsNoOverlap);
409                logger.debug("  skipped (not overlapping chains)    : {}", skippedChainsNoOverlap);
410                logger.debug("  skipped (sym redundant op pairs)    : {}", skippedRedundant);
411                logger.debug("  skipped (sym redundant self op)     : {}", skippedSelfEquivalent);
412                logger.debug("Found {} interfaces.", set.size());
413        }
414
415
416        /**
417         * Checks whether given interface is NCS-redundant, i.e., an identical interface between NCS copies of
418         * these molecules has already been seen, and returns this (reference) interface.
419         *
420         * @param interf
421         *          StructureInterface
422         * @return  already seen interface that is NCS-equivalent to interf,
423         *          null if such interface is not found.
424         */
425        private StructureInterface findNcsRef(StructureInterface interf) {
426                if (!this.hasNcsOps()) {
427                        return null;
428                }
429                String chainIName = interf.getMoleculeIds().getFirst();
430                String iOrigName = chainOrigNames.get(chainIName);
431
432                String chainJName = interf.getMoleculeIds().getSecond();
433                String jOrigName = chainOrigNames.get(chainJName);
434
435                Matrix4d mJCryst;
436                if(this.searchBeyondAU) {
437                        mJCryst = interf.getTransforms().getSecond().getMatTransform();
438                        mJCryst = crystallographicInfo.getCrystalCell().transfToOrthonormal(mJCryst);
439                } else {
440                        mJCryst = IDENTITY;
441                }
442
443                // Let X1,...Xn be the original coords, before NCS transforms (M1...Mk)
444                // current chain i: M_i * X_i
445                // current chain j: Cn * M_j * X_j
446
447                // transformation to bring chain j near X_i: M_i^(-1) * Cn * M_j
448                // transformation to bring chain i near X_j: (Cn * M_j)^(-1) * M_i = (M_i^(-1) * Cn * M_j)^(-1)
449
450                Matrix4d mChainIInv = new Matrix4d(chainNcsOps.get(chainIName));
451                mChainIInv.invert();
452
453                Matrix4d mJNcs = new Matrix4d(chainNcsOps.get(chainJName));
454
455                Matrix4d j2iNcsOrigin = new Matrix4d(mChainIInv);
456                j2iNcsOrigin.mul(mJCryst);
457                //overall transformation to bring current chainj from its NCS origin to i's
458                j2iNcsOrigin.mul(mJNcs);
459
460                //overall transformation to bring current chaini from its NCS origin to j's
461                Matrix4d i2jNcsOrigin = new Matrix4d(j2iNcsOrigin);
462                i2jNcsOrigin.invert();
463
464                String matchChainIdsIJ = iOrigName + jOrigName;
465                String matchChainIdsJI = jOrigName + iOrigName;
466
467                // same original chain names
468                Optional<Matrix4d> matchDirect =
469                                visitedNcsChainPairs.computeIfAbsent(matchChainIdsIJ, k-> new HashMap<>()).entrySet().stream().
470                                        map(r->r.getKey()).
471                                        filter(r->r.epsilonEquals(j2iNcsOrigin,0.01)).
472                                        findFirst();
473
474                Matrix4d matchMatrix = matchDirect.orElse(null);
475                String matchChainIds = matchChainIdsIJ;
476
477                if(matchMatrix == null) {
478                        // reversed original chain names with inverted transform
479                        Optional<Matrix4d> matchInverse =
480                                        visitedNcsChainPairs.computeIfAbsent(matchChainIdsJI, k-> new HashMap<>()).entrySet().stream().
481                                        map(r->r.getKey()).
482                                        filter(r->r.epsilonEquals(i2jNcsOrigin,0.01)).
483                                        findFirst();
484                        matchMatrix = matchInverse.orElse(null);
485                        matchChainIds = matchChainIdsJI;
486                }
487
488                StructureInterface matchInterface = null;
489
490                if (matchMatrix == null) {
491                        visitedNcsChainPairs.get(matchChainIdsIJ).put(j2iNcsOrigin,interf);
492                } else {
493                        matchInterface = visitedNcsChainPairs.get(matchChainIds).get(matchMatrix);
494                }
495
496                return matchInterface;
497        }
498
499        private StructureInterface calcContacts(Chain chaini, Chain chainj, double cutoff, CrystalTransform tt, StringBuilder builder) {
500                // note that we don't consider hydrogens when calculating contacts
501                AtomContactSet graph = StructureTools.getAtomsInContact(chaini, chainj, cutoff, INCLUDE_HETATOMS);
502
503                if (graph.size()>0) {
504                        if (builder != null) builder.append("x");
505
506                        CrystalTransform transf = new CrystalTransform(this.crystallographicInfo.getSpaceGroup());
507                        StructureInterface interf = new StructureInterface(
508                                        StructureTools.getAllAtomArray(chaini), StructureTools.getAllAtomArray(chainj),
509                                        chaini.getName(), chainj.getName(),
510                                        graph,
511                                        transf, tt);
512
513                        return interf;
514
515                } else {
516                        if (builder != null) builder.append("o");
517                        return null;
518                }
519        }
520
521        private void addVisitedTransform(CrystalTransform tt) {
522                visitedCrystalTransforms.add(tt);
523        }
524
525        /**
526         * Checks whether given transformId/translation is symmetry redundant
527         * Two transformations are symmetry redundant if their matrices (4d) multiplication gives the identity, i.e.
528         * if one is the inverse of the other.
529         * @param tt
530         * @return
531         */
532        private boolean isRedundantTransform(CrystalTransform tt) {
533
534                Iterator<CrystalTransform> it = visitedCrystalTransforms.iterator();
535                while (it.hasNext()) {
536                        CrystalTransform v = it.next();
537
538                        if (tt.isEquivalent(v)) {
539
540                                logger.debug("Skipping redundant transformation: "+tt+", equivalent to "+v);
541
542                                // there's only 1 possible equivalent partner for each visited matrix
543                                // (since the equivalent is its inverse matrix and the inverse matrix is unique)
544                                // thus once the partner has been seen, we don't need to check it ever again
545                                it.remove();
546
547                                return true;
548                        }
549                }
550
551                return false;
552        }
553
554        public void translate(Matrix4d m, Vector3d translation) {
555                m.m03 = m.m03+translation.x;
556                m.m13 = m.m13+translation.y;
557                m.m23 = m.m23+translation.z;
558        }
559
560        /**
561         * Apply the NCS operators in the given Structure adding new chains as needed.
562         * All chains are (re)assigned ids of the form: original_chain_id+ncs_operator_index+{@value #NCS_CHAINID_SUFFIX_CHAR}.
563         * @param structure
564         *          the structure to expand
565         * @param chainOrigNames
566         *          new chain names mapped to the original chain names
567         * @param chainNcsOps
568         *          new chain names mapped to the ncs operators that was used to generate them
569         * @since 5.0.0
570         */
571        public static void expandNcsOps(Structure structure, Map<String,String> chainOrigNames, Map<String,Matrix4d> chainNcsOps) {
572                PDBCrystallographicInfo xtalInfo = structure.getCrystallographicInfo();
573                if (xtalInfo ==null) return;
574
575                if (xtalInfo.getNcsOperators()==null || xtalInfo.getNcsOperators().length==0)
576                        return;
577
578                List<Chain> chainsToAdd = new ArrayList<>();
579
580                Matrix4d identity = new Matrix4d();
581                identity.setIdentity();
582
583                Matrix4d[] ncsOps = xtalInfo.getNcsOperators();
584
585                for (Chain c:structure.getChains()) {
586                        String cOrigId = c.getId();
587                        String cOrigName = c.getName();
588
589                        for (int iOperator = 0; iOperator < ncsOps.length; iOperator++) {
590                                Matrix4d m = ncsOps[iOperator];
591
592                                Chain clonedChain = (Chain)c.clone();
593                                String newChainId = cOrigId+(iOperator+1)+NCS_CHAINID_SUFFIX_CHAR;
594                                String newChainName = cOrigName+(iOperator+1)+NCS_CHAINID_SUFFIX_CHAR;
595                                clonedChain.setId(newChainId);
596                                clonedChain.setName(newChainName);
597
598                                setChainIdsInResidueNumbers(clonedChain, newChainName);
599                                Calc.transform(clonedChain, m);
600
601                                chainsToAdd.add(clonedChain);
602                                c.getEntityInfo().addChain(clonedChain);
603
604                                chainOrigNames.put(newChainName,cOrigName);
605                                chainNcsOps.put(newChainName,m);
606                        }
607
608                        chainNcsOps.put(cOrigName,identity);
609                        chainOrigNames.put(cOrigName,cOrigName);
610                }
611
612                chainsToAdd.forEach(structure::addChain);
613        }
614
615        /**
616         * Auxiliary method to reset chain ids of residue numbers in a chain.
617         * Used when cloning chains and resetting their ids: one needs to take care of
618         * resetting the ids within residue numbers too.
619         * @param c
620         * @param newChainName
621         */
622        private static void setChainIdsInResidueNumbers(Chain c, String newChainName) {
623                for (Group g:c.getAtomGroups()) {
624                        g.setResidueNumber(newChainName, g.getResidueNumber().getSeqNum(), g.getResidueNumber().getInsCode());
625                }
626                for (Group g:c.getSeqResGroups()) {
627                        if (g.getResidueNumber()==null) continue;
628                        g.setResidueNumber(newChainName, g.getResidueNumber().getSeqNum(), g.getResidueNumber().getInsCode());
629                }
630        }
631
632}