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.asa; 022 023import org.biojava.nbio.structure.*; 024import org.biojava.nbio.structure.contact.Contact; 025import org.biojava.nbio.structure.contact.Grid; 026import org.slf4j.Logger; 027import org.slf4j.LoggerFactory; 028 029import javax.vecmath.Point3d; 030import javax.vecmath.Vector3d; 031import java.util.*; 032import java.util.concurrent.ExecutorService; 033import java.util.concurrent.Executors; 034 035 036/** 037 * Class to calculate Accessible Surface Areas based on 038 * the rolling ball algorithm by Shrake and Rupley. 039 * <p> 040 * The code is adapted from a python implementation at http://boscoh.com/protein/asapy 041 * (now source is available at https://github.com/boscoh/asa). 042 * Thanks to Bosco K. Ho for a great piece of code and for his fantastic blog. 043 * <p> 044 * A few optimizations come from Eisenhaber et al, J Comp Chemistry 1994 045 * (https://onlinelibrary.wiley.com/doi/epdf/10.1002/jcc.540160303) 046 * <p> 047 * See 048 * Shrake, A., and J. A. Rupley. "Environment and Exposure to Solvent of Protein Atoms. 049 * Lysozyme and Insulin." JMB (1973) 79:351-371. 050 * Lee, B., and Richards, F.M. "The interpretation of Protein Structures: Estimation of 051 * Static Accessibility" JMB (1971) 55:379-400 052 * 053 * @author Jose Duarte 054 */ 055public class AsaCalculator { 056 057 private static final Logger logger = LoggerFactory.getLogger(AsaCalculator.class); 058 059 /** 060 * The default value for number of sphere points to sample. 061 * See this paper for a nice study on the effect of this parameter: https://f1000research.com/articles/5-189/v1 062 */ 063 public static final int DEFAULT_N_SPHERE_POINTS = 1000; 064 public static final double DEFAULT_PROBE_SIZE = 1.4; 065 public static final int DEFAULT_NTHREADS = 1; 066 067 private static final boolean DEFAULT_USE_SPATIAL_HASHING = true; 068 069 070 071 // Chothia's amino acid atoms vdw radii 072 public static final double TRIGONAL_CARBON_VDW = 1.76; 073 public static final double TETRAHEDRAL_CARBON_VDW = 1.87; 074 public static final double TRIGONAL_NITROGEN_VDW = 1.65; 075 public static final double TETRAHEDRAL_NITROGEN_VDW = 1.50; 076 public static final double SULFUR_VDW = 1.85; 077 public static final double OXIGEN_VDW = 1.40; 078 079 // Chothia's nucleotide atoms vdw radii 080 public static final double NUC_CARBON_VDW = 1.80; 081 public static final double NUC_NITROGEN_VDW = 1.60; 082 public static final double PHOSPHOROUS_VDW = 1.90; 083 084 085 086 087 088 private class AsaCalcWorker implements Runnable { 089 090 private final int i; 091 private final double[] asas; 092 093 private AsaCalcWorker(int i, double[] asas) { 094 this.i = i; 095 this.asas = asas; 096 } 097 098 @Override 099 public void run() { 100 asas[i] = calcSingleAsa(i); 101 } 102 } 103 104 static class IndexAndDistance { 105 final int index; 106 final double dist; 107 IndexAndDistance(int index, double dist) { 108 this.index = index; 109 this.dist = dist; 110 } 111 } 112 113 114 private final Point3d[] atomCoords; 115 private final Atom[] atoms; 116 private final double[] radii; 117 private final double probe; 118 private final int nThreads; 119 private Vector3d[] spherePoints; 120 private double cons; 121 private IndexAndDistance[][] neighborIndices; 122 123 private boolean useSpatialHashingForNeighbors; 124 125 /** 126 * Constructs a new AsaCalculator. Subsequently call {@link #calculateAsas()} 127 * or {@link #getGroupAsas()} to calculate the ASAs 128 * Only non-Hydrogen atoms are considered in the calculation. 129 * @param structure the structure, all non-H atoms will be used 130 * @param probe the probe size 131 * @param nSpherePoints the number of points to be used in generating the spherical 132 * dot-density, the more points the more accurate (and slower) calculation 133 * @param nThreads the number of parallel threads to use for the calculation 134 * @param hetAtoms if true HET residues are considered, if false they aren't, equivalent to 135 * NACCESS' -h option 136 */ 137 public AsaCalculator(Structure structure, double probe, int nSpherePoints, int nThreads, boolean hetAtoms) { 138 this.atoms = StructureTools.getAllNonHAtomArray(structure, hetAtoms); 139 this.atomCoords = Calc.atomsToPoints(atoms); 140 this.probe = probe; 141 this.nThreads = nThreads; 142 143 this.useSpatialHashingForNeighbors = DEFAULT_USE_SPATIAL_HASHING; 144 145 // initialising the radii by looking them up through AtomRadii 146 radii = new double[atomCoords.length]; 147 for (int i=0;i<atomCoords.length;i++) { 148 radii[i] = getRadius(atoms[i]); 149 } 150 151 initSpherePoints(nSpherePoints); 152 } 153 154 /** 155 * Constructs a new AsaCalculator. Subsequently call {@link #calculateAsas()} 156 * or {@link #getGroupAsas()} to calculate the ASAs. 157 * @param atoms an array of atoms not containing Hydrogen atoms 158 * @param probe the probe size 159 * @param nSpherePoints the number of points to be used in generating the spherical 160 * dot-density, the more points the more accurate (and slower) calculation 161 * @param nThreads the number of parallel threads to use for the calculation 162 * @throws IllegalArgumentException if any atom in the array is a Hydrogen atom 163 */ 164 public AsaCalculator(Atom[] atoms, double probe, int nSpherePoints, int nThreads) { 165 this.atoms = atoms; 166 this.atomCoords = Calc.atomsToPoints(atoms); 167 this.probe = probe; 168 this.nThreads = nThreads; 169 170 this.useSpatialHashingForNeighbors = DEFAULT_USE_SPATIAL_HASHING; 171 172 for (Atom atom:atoms) { 173 if (atom.getElement()==Element.H) 174 throw new IllegalArgumentException("Can't calculate ASA for an array that contains Hydrogen atoms "); 175 } 176 177 // initialising the radii by looking them up through AtomRadii 178 radii = new double[atoms.length]; 179 for (int i=0;i<atoms.length;i++) { 180 radii[i] = getRadius(atoms[i]); 181 } 182 183 initSpherePoints(nSpherePoints); 184 } 185 186 /** 187 * Constructs a new AsaCalculator. Subsequently call {@link #calcSingleAsa(int)} 188 * to calculate the atom ASAs. The given radius parameter will be taken as the radius for 189 * all points given. No ASA calculation per group will be possible with this constructor, so 190 * usage of {@link #getGroupAsas()} will result in a NullPointerException. 191 * @param atomCoords 192 * the coordinates representing the center of atoms 193 * @param probe 194 * the probe size 195 * @param nSpherePoints 196 * the number of points to be used in generating the spherical 197 * dot-density, the more points the more accurate (and slower) calculation 198 * @param nThreads 199 * the number of parallel threads to use for the calculation 200 * @param radius 201 * the radius that will be assign to all given coordinates 202 */ 203 public AsaCalculator(Point3d[] atomCoords, double probe, int nSpherePoints, int nThreads, double radius) { 204 this.atoms = null; 205 this.atomCoords = atomCoords; 206 this.probe = probe; 207 this.nThreads = nThreads; 208 209 this.useSpatialHashingForNeighbors = DEFAULT_USE_SPATIAL_HASHING; 210 211 // initialising the radii to the given radius for all atoms 212 radii = new double[atomCoords.length]; 213 for (int i=0;i<atomCoords.length;i++) { 214 radii[i] = radius; 215 } 216 217 initSpherePoints(nSpherePoints); 218 } 219 220 private void initSpherePoints(int nSpherePoints) { 221 222 logger.debug("Will use {} sphere points", nSpherePoints); 223 224 // initialising the sphere points to sample 225 spherePoints = generateSpherePoints(nSpherePoints); 226 227 cons = 4.0 * Math.PI / nSpherePoints; 228 } 229 230 /** 231 * Calculates ASA for all atoms and return them as a GroupAsa 232 * array (one element per residue in structure) containing ASAs per residue 233 * and per atom. 234 * The sorting of Groups in returned array is as specified by {@link org.biojava.nbio.structure.ResidueNumber} 235 * @return 236 */ 237 public GroupAsa[] getGroupAsas() { 238 239 TreeMap<ResidueNumber, GroupAsa> asas = new TreeMap<>(); 240 241 double[] asasPerAtom = calculateAsas(); 242 243 for (int i=0;i<atomCoords.length;i++) { 244 Group g = atoms[i].getGroup(); 245 if (!asas.containsKey(g.getResidueNumber())) { 246 GroupAsa groupAsa = new GroupAsa(g); 247 groupAsa.addAtomAsaU(asasPerAtom[i]); 248 asas.put(g.getResidueNumber(), groupAsa); 249 } else { 250 GroupAsa groupAsa = asas.get(g.getResidueNumber()); 251 groupAsa.addAtomAsaU(asasPerAtom[i]); 252 } 253 } 254 255 return asas.values().toArray(new GroupAsa[0]); 256 } 257 258 /** 259 * Calculates the Accessible Surface Areas for the atoms given in constructor and with parameters given. 260 * Beware that the parallel implementation is quite memory hungry. It scales well as long as there is 261 * enough memory available. 262 * @return an array with asa values corresponding to each atom of the input array 263 */ 264 public double[] calculateAsas() { 265 266 double[] asas = new double[atomCoords.length]; 267 268 long start = System.currentTimeMillis(); 269 if (useSpatialHashingForNeighbors) { 270 logger.debug("Will use spatial hashing to find neighbors"); 271 neighborIndices = findNeighborIndicesSpatialHashing(); 272 } else { 273 logger.debug("Will not use spatial hashing to find neighbors"); 274 neighborIndices = findNeighborIndices(); 275 } 276 long end = System.currentTimeMillis(); 277 logger.debug("Took {} s to find neighbors", (end-start)/1000.0); 278 279 start = System.currentTimeMillis(); 280 if (nThreads<=1) { // (i.e. it will also be 1 thread if 0 or negative number specified) 281 logger.debug("Will use 1 thread for ASA calculation"); 282 for (int i=0;i<atomCoords.length;i++) { 283 asas[i] = calcSingleAsa(i); 284 } 285 286 } else { 287 logger.debug("Will use {} threads for ASA calculation", nThreads); 288 289 ExecutorService threadPool = Executors.newFixedThreadPool(nThreads); 290 291 for (int i=0;i<atomCoords.length;i++) { 292 threadPool.submit(new AsaCalcWorker(i,asas)); 293 } 294 295 threadPool.shutdown(); 296 297 while (!threadPool.isTerminated()); 298 299 } 300 end = System.currentTimeMillis(); 301 logger.debug("Took {} s to calculate all {} atoms ASAs (excluding neighbors calculation)", (end-start)/1000.0, atomCoords.length); 302 303 return asas; 304 } 305 306 /** 307 * Set the useSpatialHashingForNeighbors flag to use spatial hashing to calculate neighbors (true) or all-to-all 308 * distance calculation (false). Default is {@value DEFAULT_USE_SPATIAL_HASHING}. 309 * Use for testing performance only. 310 * @param useSpatialHashingForNeighbors the flag 311 */ 312 void setUseSpatialHashingForNeighbors(boolean useSpatialHashingForNeighbors) { 313 this.useSpatialHashingForNeighbors = useSpatialHashingForNeighbors; 314 } 315 316 /** 317 * Returns list of 3d coordinates of points on a unit sphere using the 318 * Golden Section Spiral algorithm. 319 * @param nSpherePoints the number of points to be used in generating the spherical dot-density 320 * @return the array of points as Vector3d objects 321 */ 322 private Vector3d[] generateSpherePoints(int nSpherePoints) { 323 Vector3d[] points = new Vector3d[nSpherePoints]; 324 double inc = Math.PI * (3.0 - Math.sqrt(5.0)); 325 double offset = 2.0 / nSpherePoints; 326 for (int k=0;k<nSpherePoints;k++) { 327 double y = k * offset - 1.0 + (offset / 2.0); 328 double r = Math.sqrt(1.0 - y*y); 329 double phi = k * inc; 330 points[k] = new Vector3d(Math.cos(phi)*r, y, Math.sin(phi)*r); 331 } 332 return points; 333 } 334 335 /** 336 * Returns the 2-dimensional array with neighbor indices for every atom. 337 * @return 2-dimensional array of size: n_atoms x n_neighbors_per_atom 338 */ 339 IndexAndDistance[][] findNeighborIndices() { 340 341 // looking at a typical protein case, number of neighbours are from ~10 to ~50, with an average of ~30 342 int initialCapacity = 60; 343 344 IndexAndDistance[][] nbsIndices = new IndexAndDistance[atomCoords.length][]; 345 346 for (int k=0; k<atomCoords.length; k++) { 347 double radius = radii[k] + probe + probe; 348 349 List<IndexAndDistance> thisNbIndices = new ArrayList<>(initialCapacity); 350 351 for (int i = 0; i < atomCoords.length; i++) { 352 if (i == k) continue; 353 354 double dist = atomCoords[i].distance(atomCoords[k]); 355 356 if (dist < radius + radii[i]) { 357 thisNbIndices.add(new IndexAndDistance(i, dist)); 358 } 359 } 360 361 IndexAndDistance[] indicesArray = thisNbIndices.toArray(new IndexAndDistance[0]); 362 nbsIndices[k] = indicesArray; 363 } 364 return nbsIndices; 365 } 366 367 /** 368 * Returns the 2-dimensional array with neighbor indices for every atom, 369 * using spatial hashing to avoid all to all distance calculation. 370 * @return 2-dimensional array of size: n_atoms x n_neighbors_per_atom 371 */ 372 IndexAndDistance[][] findNeighborIndicesSpatialHashing() { 373 374 // looking at a typical protein case, number of neighbours are from ~10 to ~50, with an average of ~30 375 int initialCapacity = 60; 376 377 List<Contact> contactList = calcContacts(); 378 Map<Integer, List<IndexAndDistance>> indices = new HashMap<>(atomCoords.length); 379 for (Contact contact : contactList) { 380 // note contacts are stored 1-way only, with j>i 381 int i = contact.getI(); 382 int j = contact.getJ(); 383 384 List<IndexAndDistance> iIndices; 385 List<IndexAndDistance> jIndices; 386 if (!indices.containsKey(i)) { 387 iIndices = new ArrayList<>(initialCapacity); 388 indices.put(i, iIndices); 389 } else { 390 iIndices = indices.get(i); 391 } 392 if (!indices.containsKey(j)) { 393 jIndices = new ArrayList<>(initialCapacity); 394 indices.put(j, jIndices); 395 } else { 396 jIndices = indices.get(j); 397 } 398 399 double radius = radii[i] + probe + probe; 400 double dist = contact.getDistance(); 401 if (dist < radius + radii[j]) { 402 iIndices.add(new IndexAndDistance(j, dist)); 403 jIndices.add(new IndexAndDistance(i, dist)); 404 } 405 } 406 407 // convert map to array for fast access 408 IndexAndDistance[][] nbsIndices = new IndexAndDistance[atomCoords.length][]; 409 for (Map.Entry<Integer, List<IndexAndDistance>> entry : indices.entrySet()) { 410 List<IndexAndDistance> list = entry.getValue(); 411 IndexAndDistance[] indexAndDistances = list.toArray(new IndexAndDistance[0]); 412 nbsIndices[entry.getKey()] = indexAndDistances; 413 } 414 415 // important: some atoms might have no neighbors at all: we need to initialise to empty arrays 416 for (int i=0; i<nbsIndices.length; i++) { 417 if (nbsIndices[i] == null) { 418 nbsIndices[i] = new IndexAndDistance[0]; 419 } 420 } 421 422 return nbsIndices; 423 } 424 425 Point3d[] getAtomCoords() { 426 return atomCoords; 427 } 428 429 private List<Contact> calcContacts() { 430 if (atomCoords.length == 0) 431 return new ArrayList<>(); 432 double maxRadius = 0; 433 OptionalDouble optionalDouble = Arrays.stream(radii).max(); 434 if (optionalDouble.isPresent()) 435 maxRadius = optionalDouble.getAsDouble(); 436 double cutoff = maxRadius + maxRadius + probe + probe; 437 logger.debug("Max radius is {}, cutoff is {}", maxRadius, cutoff); 438 Grid grid = new Grid(cutoff); 439 grid.addCoords(atomCoords); 440 return grid.getIndicesContacts(); 441 } 442 443 private double calcSingleAsa(int i) { 444 Point3d atom_i = atomCoords[i]; 445 446 int n_neighbor = neighborIndices[i].length; 447 IndexAndDistance[] neighbor_indices = neighborIndices[i]; 448 // Sorting by closest to farthest away neighbors achieves faster runtimes when checking for occluded 449 // sphere sample points below. This follows the ideas exposed in 450 // Eisenhaber et al, J Comp Chemistry 1994 (https://onlinelibrary.wiley.com/doi/epdf/10.1002/jcc.540160303) 451 // This is essential for performance. In my tests this brings down the number of occlusion checks in loop below to 452 // an average of n_sphere_points/10 per atom i, producing ~ x4 performance gain overall 453 Arrays.sort(neighbor_indices, Comparator.comparingDouble(o -> o.dist)); 454 455 double radius_i = probe + radii[i]; 456 457 int n_accessible_point = 0; 458 // purely for debugging 459 int[] numDistsCalced = null; 460 if (logger.isDebugEnabled()) numDistsCalced = new int[n_neighbor]; 461 462 // now we precalculate anything depending only on i,j in equation 3 in Eisenhaber 1994 463 double[] sqRadii = new double[n_neighbor]; 464 Vector3d[] aj_minus_ais = new Vector3d[n_neighbor]; 465 for (int nbArrayInd =0; nbArrayInd<n_neighbor; nbArrayInd++) { 466 int j = neighbor_indices[nbArrayInd].index; 467 double dist = neighbor_indices[nbArrayInd].dist; 468 double radius_j = radii[j] + probe; 469 // see equation 3 in Eisenhaber 1994 470 sqRadii[nbArrayInd] = (dist*dist + radius_i*radius_i - radius_j*radius_j)/(2*radius_i); 471 Vector3d aj_minus_ai = new Vector3d(atomCoords[j]); 472 aj_minus_ai.sub(atom_i); 473 aj_minus_ais[nbArrayInd] = aj_minus_ai; 474 } 475 476 for (Vector3d point: spherePoints){ 477 boolean is_accessible = true; 478 479 // note that the neighbors are sorted by distance, achieving optimal performance in this inner loop 480 // See Eisenhaber et al, J Comp Chemistry 1994 481 482 for (int nbArrayInd =0; nbArrayInd<n_neighbor; nbArrayInd++) { 483 484 // see equation 3 in Eisenhaber 1994. This is slightly more efficient than 485 // calculating distances to the actual sphere points on atom_i (which would be obtained with: 486 // Point3d test_point = new Point3d(point.x*radius + atom_i.x,point.y*radius + atom_i.y,point.z*radius + atom_i.z)) 487 double dotProd = aj_minus_ais[nbArrayInd].dot(point); 488 489 if (numDistsCalced!=null) numDistsCalced[nbArrayInd]++; 490 491 if (dotProd > sqRadii[nbArrayInd]) { 492 is_accessible = false; 493 break; 494 } 495 } 496 if (is_accessible) { 497 n_accessible_point++; 498 } 499 } 500 501 // purely for debugging 502 if (numDistsCalced!=null) { 503 int sum = 0; 504 for (int numDistCalcedForJ : numDistsCalced) sum += numDistCalcedForJ; 505 logger.debug("Number of sample points distances calculated for neighbors of i={} : average {}, all {}", i, (double) sum / (double) n_neighbor, numDistsCalced); 506 } 507 508 return cons*n_accessible_point*radius_i*radius_i; 509 } 510 511 /** 512 * Gets the radius for given amino acid and atom 513 * @param amino 514 * @param atom 515 * @return 516 */ 517 private static double getRadiusForAmino(AminoAcid amino, Atom atom) { 518 519 if (atom.getElement().equals(Element.H)) return Element.H.getVDWRadius(); 520 // some unusual entries (e.g. 1tes) contain Deuterium atoms in standard aminoacids 521 if (atom.getElement().equals(Element.D)) return Element.D.getVDWRadius(); 522 523 String atomCode = atom.getName(); 524 char aa = amino.getAminoType(); 525 526 // here we use the values that Chothia gives in his paper (as NACCESS does) 527 if (atom.getElement()==Element.O) { 528 return OXIGEN_VDW; 529 } 530 else if (atom.getElement()==Element.S) { 531 return SULFUR_VDW; 532 } 533 else if (atom.getElement()==Element.N) { 534 if ("NZ".equals(atomCode)) return TETRAHEDRAL_NITROGEN_VDW; // tetrahedral Nitrogen 535 return TRIGONAL_NITROGEN_VDW; // trigonal Nitrogen 536 } 537 else if (atom.getElement()==Element.C) { // it must be a carbon 538 if ("C".equals(atomCode) || 539 "CE1".equals(atomCode) || "CE2".equals(atomCode) || "CE3".equals(atomCode) || 540 "CH2".equals(atomCode) || 541 "CZ".equals(atomCode) || "CZ2".equals(atomCode) || "CZ3".equals(atomCode)) { 542 return TRIGONAL_CARBON_VDW; // trigonal Carbon 543 } 544 else if ("CA".equals(atomCode) || "CB".equals(atomCode) || 545 "CE".equals(atomCode) || 546 "CG1".equals(atomCode) || "CG2".equals(atomCode)) { 547 return TETRAHEDRAL_CARBON_VDW; // tetrahedral Carbon 548 } 549 // the rest of the cases (CD, CD1, CD2, CG) depend on amino acid 550 else { 551 switch (aa) { 552 case 'F': 553 case 'W': 554 case 'Y': 555 case 'H': 556 case 'D': 557 case 'N': 558 return TRIGONAL_CARBON_VDW; 559 560 case 'P': 561 case 'K': 562 case 'R': 563 case 'M': 564 case 'I': 565 case 'L': 566 return TETRAHEDRAL_CARBON_VDW; 567 568 case 'Q': 569 case 'E': 570 if ("CD".equals(atomCode)) return TRIGONAL_CARBON_VDW; 571 else if ("CG".equals(atomCode)) return TETRAHEDRAL_CARBON_VDW; 572 573 default: 574 logger.info("Unexpected carbon atom {} for aminoacid {}, assigning its standard vdw radius", atomCode, aa); 575 return Element.C.getVDWRadius(); 576 } 577 } 578 579 // not any of the expected atoms 580 } else { 581 // non standard aas, (e.g. MSE, LLP) will always have this problem, 582 logger.debug("Unexpected atom {} for aminoacid {} ({}), assigning its standard vdw radius", atomCode, aa, amino.getPDBName()); 583 584 return atom.getElement().getVDWRadius(); 585 } 586 } 587 588 589 /** 590 * Gets the radius for given nucleotide atom 591 * @param atom 592 * @return 593 */ 594 private static double getRadiusForNucl(NucleotideImpl nuc, Atom atom) { 595 596 if (atom.getElement().equals(Element.H)) return Element.H.getVDWRadius(); 597 if (atom.getElement().equals(Element.D)) return Element.D.getVDWRadius(); 598 599 if (atom.getElement()==Element.C) return NUC_CARBON_VDW; 600 601 if (atom.getElement()==Element.N) return NUC_NITROGEN_VDW; 602 603 if (atom.getElement()==Element.P) return PHOSPHOROUS_VDW; 604 605 if (atom.getElement()==Element.O) return OXIGEN_VDW; 606 607 logger.info("Unexpected atom "+atom.getName()+" for nucleotide "+nuc.getPDBName()+", assigning its standard vdw radius"); 608 return atom.getElement().getVDWRadius(); 609 } 610 611 612 /** 613 * Gets the van der Waals radius of the given atom following the values defined by 614 * Chothia (1976) J.Mol.Biol.105,1-14 615 * NOTE: the vdw values defined by the paper assume no Hydrogens and thus "inflates" 616 * slightly the heavy atoms to account for Hydrogens. Thus this method cannot be used 617 * in a structure that contains Hydrogens! 618 * 619 * If atom is neither part of a nucleotide nor of a standard aminoacid, 620 * the default vdw radius for the element is returned. If atom is of 621 * unknown type (element) the vdw radius of {@link Element#N} is returned 622 * 623 * @param atom 624 * @return 625 */ 626 public static double getRadius(Atom atom) { 627 628 if (atom.getElement()==null) { 629 logger.warn("Unrecognised atom "+atom.getName()+" with serial "+atom.getPDBserial()+ 630 ", assigning the default vdw radius (Nitrogen vdw radius)."); 631 return Element.N.getVDWRadius(); 632 } 633 634 Group res = atom.getGroup(); 635 636 if (res==null) { 637 logger.warn("Unknown parent residue for atom "+atom.getName()+" with serial "+ 638 atom.getPDBserial()+", assigning its default vdw radius"); 639 return atom.getElement().getVDWRadius(); 640 } 641 642 GroupType type = res.getType(); 643 644 if (type == GroupType.AMINOACID) return getRadiusForAmino(((AminoAcid)res), atom); 645 646 if (type == GroupType.NUCLEOTIDE) return getRadiusForNucl((NucleotideImpl)res,atom); 647 648 649 return atom.getElement().getVDWRadius(); 650 } 651 652}