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 * Created on 26.04.2004 021 * @author Andreas Prlic 022 * 023 */ 024package org.biojava.nbio.structure; 025 026import java.util.ArrayList; 027import java.util.List; 028import java.util.ListIterator; 029 030import org.biojava.nbio.structure.io.FileConvert; 031import org.slf4j.Logger; 032import org.slf4j.LoggerFactory; 033 034/** 035 * Implementation of a PDB Structure. This class 036 * provides the data contained in a PDB file. 037 * to get structure objects from different sources 038 * see io package. 039 * 040 * @author Andreas Prlic 041 * @author Jules Jacobsen 042 * @since 1.4 043 * @version %I% %G% 044 */ 045public class StructureImpl implements Structure { 046 047 private static final long serialVersionUID = -8344837138032851348L; 048 049 private static final Logger logger = LoggerFactory.getLogger(StructureImpl.class); 050 051 private PdbId pdbId ; 052 053 private List<Model> models; 054 055 private List<EntityInfo> entityInfos; 056 private List<DBRef> dbrefs; 057 private List<Bond> ssbonds; 058 private List<Site> sites; 059 private String name ; 060 private StructureIdentifier structureIdentifier; 061 062 private PDBHeader pdbHeader; 063 064 private boolean biologicalAssembly; 065 066 /** 067 * Constructs a StructureImpl object. 068 */ 069 public StructureImpl() { 070 super(); 071 072 models = new ArrayList<>(); 073 name = ""; 074 entityInfos = new ArrayList<>(); 075 dbrefs = new ArrayList<>(); 076 pdbHeader = new PDBHeader(); 077 ssbonds = new ArrayList<>(); 078 sites = new ArrayList<>(); 079 } 080 081 /** 082 * Construct a Structure object that only contains a single group 083 * 084 * @param g group object 085 */ 086 public StructureImpl(Group g){ 087 this(); 088 089 Chain c = new ChainImpl(); 090 c.addGroup(g); 091 092 addChain(c); 093 } 094 095 /** 096 * Construct a Structure object that contains a particular chain 097 * 098 * @param c chain 099 */ 100 public StructureImpl(Chain c){ 101 this(); 102 addChain(c); 103 } 104 105 /** 106 * Returns an identical copy of this structure . 107 * @return an identical Structure object 108 */ 109 @Override 110 public Structure clone() { 111 // Note: structures are also cloned in SubstructureIdentifier.reduce(). 112 // Changes might need to be made there as well 113 114 Structure n = new StructureImpl(); 115 // go through whole substructure and clone ... 116 117 // copy structure data 118 119 n.setPdbId(getPdbId()); 120 n.setName(getName()); 121 //TODO the header data is not being deep-copied, that's a minor issue since it is just some static metadata, but we should recheck this if needed - JD 2014-12-11 122 n.setPDBHeader(pdbHeader); 123 n.setDBRefs(this.getDBRefs()); 124 n.setSites(getSites()); 125 126 127 // go through each chain and clone chain 128 for (int i=0;i<nrModels();i++){ 129 List<Chain> cloned_model = new ArrayList<>(); 130 131 for (int j=0;j<size(i);j++){ 132 133 Chain cloned_chain = (Chain) getChainByIndex(i,j).clone(); 134 135 // setting the parent: can only be done from the parent 136 cloned_chain.setStructure(n); 137 138 cloned_model.add(cloned_chain); 139 140 } 141 n.addModel(cloned_model); 142 143 } 144 145 // deep-copying of entityInfofos is tricky: there's cross references also in the Chains 146 // beware: if we copy the entityInfos we would also need to reset the references to entityInfos in the individual chains 147 List<EntityInfo> newEntityInfoList = new ArrayList<>(); 148 for (EntityInfo entityInfo : this.entityInfos) { 149 EntityInfo newEntityInfo = new EntityInfo(entityInfo); // this sets everything but the chains 150 for (String asymId:entityInfo.getChainIds()) { 151 152 for (int modelNr=0;modelNr<n.nrModels();modelNr++) { 153 Chain newChain = n.getChain(asymId,modelNr); 154 if (newChain==null) { 155 // this actually happens for structure 1msh, which has no chain B for model 29 (clearly a deposition error) 156 logger.warn("Could not find chain asymId "+asymId+" of model "+modelNr+" while cloning entityInfo "+entityInfo.getMolId()+". Something is wrong!"); 157 continue; 158 } 159 newChain.setEntityInfo(newEntityInfo); 160 newEntityInfo.addChain(newChain); 161 } 162 } 163 newEntityInfoList.add(newEntityInfo); 164 } 165 n.setEntityInfos(newEntityInfoList); 166 // TODO ssbonds are complicated to clone: there are deep references inside Atom objects, how would we do it? - JD 2016-03-03 167 168 return n ; 169 } 170 171 /** {@inheritDoc} */ 172 @Override 173 public Group findGroup(String chainName, String pdbResnum, int modelnr) 174 throws StructureException { 175 176 177 // if structure is xray there will be only one "model". 178 if ( modelnr > models.size()) 179 throw new StructureException(" no model nr " + modelnr + 180 " in this structure. (contains "+models.size()+")"); 181 182 183 // first we need to gather all groups with the author id chainName: polymers, non-polymers and waters 184 Chain polyChain = getPolyChainByPDB(chainName, modelnr); 185 if(polyChain != null) { 186 List<Group> groups = new ArrayList<>(); 187 188 groups.addAll(polyChain.getAtomGroups()); 189 190 191 // there can be more than one non-poly chain for a given author id 192 for (Chain chain: getNonPolyChainsByPDB(chainName, modelnr)) { 193 groups.addAll(chain.getAtomGroups()); 194 } 195 196 Chain water = getWaterChainByPDB(chainName, modelnr); 197 198 if (water!=null) 199 groups.addAll(water.getAtomGroups()); 200 201 202 203 // now iterate over all groups 204 // in order to find the amino acid that has this pdbRenum. 205 206 for (Group g : groups) { 207 String rnum = g.getResidueNumber().toString(); 208 //System.out.println(g + " >" + rnum + "< >" + pdbResnum + "<"); 209 // we only mutate amino acids 210 // and ignore hetatoms and nucleotides in this case 211 if (rnum.equals(pdbResnum)) { 212 return g; 213 } 214 } 215 } 216 throw new StructureException("could not find group " + pdbResnum + 217 " in chain " + chainName); 218 } 219 220 221 /** {@inheritDoc} */ 222 @Override 223 public Group findGroup(String chainName, String pdbResnum) throws StructureException 224 { 225 return findGroup(chainName, pdbResnum, 0); 226 227 } 228 229 230 /** {@inheritDoc} */ 231 @Override 232 public void setName(String nam) { name = nam; } 233 234 /** {@inheritDoc} */ 235 @Override 236 public String getName() { return name; } 237 238 /** 239 * @return The StructureIdentifier used to create this structure 240 */ 241 @Override 242 public StructureIdentifier getStructureIdentifier() { 243 return structureIdentifier; 244 } 245 246 /** 247 * @param structureIdentifier the structureIdentifier corresponding to this structure 248 */ 249 @Override 250 public void setStructureIdentifier(StructureIdentifier structureIdentifier) { 251 this.structureIdentifier = structureIdentifier; 252 } 253 254 /** {@inheritDoc} */ 255 @Override 256 public void addChain(Chain chain) { 257 int modelnr = 0 ; 258 addChain(chain,modelnr); 259 } 260 261 /** {@inheritDoc} */ 262 @Override 263 public void addChain(Chain chain, int modelnr) { 264 // if model has not been initialized, init it! 265 chain.setStructure(this); 266 if (models.isEmpty()) { 267 Model model = new Model(); 268 List<Chain> modelChains = new ArrayList<>() ; 269 modelChains.add(chain); 270 model.setChains(modelChains); 271 models.add(model); 272 273 } else { 274 Model model = models.get(modelnr); 275 model.addChain(chain); 276 } 277 278 279 280 } 281 282 283 284 /** {@inheritDoc} */ 285 @Override 286 public Chain getChainByIndex(int number) { 287 288 int modelnr = 0 ; 289 290 return getChainByIndex(modelnr,number); 291 } 292 293 294 /** {@inheritDoc} */ 295 @Override 296 public Chain getChainByIndex(int modelnr,int number) { 297 298 Model model = models.get(modelnr); 299 300 return model.getChains().get(number); 301 } 302 303 304 305 /** {@inheritDoc} */ 306 @Override 307 public void addModel(List<Chain> modelChains){ 308 for (Chain c: modelChains){ 309 c.setStructure(this); 310 } 311 Model model = new Model(); 312 model.setChains(modelChains); 313 models.add(model); 314 } 315 316 317 /** {@inheritDoc} */ 318 @Override 319 public void setChains(List<Chain> chains){ 320 321 setModel(0,chains); 322 } 323 324 325 326 /** {@inheritDoc} */ 327 @Override 328 public void setModel(int position, List<Chain> modelChains){ 329 if (modelChains == null) 330 throw new IllegalArgumentException("trying to set model to null!"); 331 332 for (Chain c: modelChains) 333 c.setStructure(this); 334 335 //System.out.println("model size:" + models.size()); 336 337 338 Model model = new Model(); 339 model.setChains(modelChains); 340 341 if (models.isEmpty()){ 342 models.add(model); 343 } else { 344 models.set(position, model); 345 } 346 } 347 348 /** String representation. 349 * 350 */ 351 @Override 352 public String toString(){ 353 String newline = System.getProperty("line.separator"); 354 StringBuilder str = new StringBuilder(); 355 str.append("structure "); 356 str.append(name); 357 str.append(" "); 358 str.append(pdbId); 359 str.append(" "); 360 361 if ( nrModels()>1 ){ 362 str.append( " models: "); 363 str.append(nrModels()); 364 str.append(newline) ; 365 } 366 367 str.append(pdbHeader); 368 str.append(newline) ; 369 370 for (int i=0;i<nrModels();i++){ 371 if ( nrModels()>1 ) { 372 str.append(" model["); 373 str.append(i); 374 str.append("]:"); 375 str.append(newline); 376 } 377 str.append(" chains:"); 378 str.append(newline); 379 380 for (int j=0;j<size(i);j++){ 381 382 Chain cha = getChainByIndex(i,j); 383 List<Group> agr = cha.getAtomGroups(GroupType.AMINOACID); 384 List<Group> hgr = cha.getAtomGroups(GroupType.HETATM); 385 List<Group> ngr = cha.getAtomGroups(GroupType.NUCLEOTIDE); 386 387 388 389 390 str.append("chain ") 391 .append(j).append(": asymId:") 392 .append(cha.getId()) 393 .append(" authId:") 394 .append(cha.getName()).append(" "); 395 396 397 if ( cha.getEntityInfo() != null){ 398 EntityInfo comp = cha.getEntityInfo(); 399 String molName = comp.getDescription(); 400 if ( molName != null){ 401 str.append(molName); 402 } 403 String type = comp.getType().toString(); 404 str.append(" (") 405 .append(type) 406 .append(")"); 407 } 408 409 410 str.append(newline); 411 str.append(" length SEQRES: ").append(cha.getSeqResLength()); 412 str.append(" length ATOM: ").append(cha.getAtomLength()); 413 str.append(" aminos: ").append(agr.size()); 414 str.append(" hetatms: ").append(hgr.size()); 415 str.append(" nucleotides: ").append(ngr.size()).append(newline); 416 } 417 418 } 419 str.append("DBRefs: ").append(dbrefs.size()).append(newline); 420 for (DBRef dbref: dbrefs){ 421 str.append(dbref.toPDB()).append(newline); 422 } 423 str.append("Molecules: ").append(newline); 424 for (EntityInfo mol : entityInfos) { 425 str.append(mol).append(newline); 426 } 427 428 429 return str.toString() ; 430 } 431 432 @Override 433 public int size() { 434 int modelnr = 0 ; 435 436 if (!models.isEmpty()) { 437 return models.get(modelnr).getPolyChains().size(); 438 } 439 else { 440 return 0 ; 441 } 442 443 } 444 445 /** return number of chains of model. 446 * 447 */ 448 @Override 449 public int size(int modelnr) { return models.get(modelnr).size(); } 450 451 // some NMR stuff : 452 453 /** return number of models. */ 454 @Override 455 public int nrModels() { 456 return models.size() ; 457 } 458 459 /** 460 * Whether this Structure is a crystallographic structure or not. 461 * It will first check the experimental technique and if not present it will try 462 * to guess from the presence of a space group and sensible cell parameters 463 * 464 * @return true if crystallographic, false otherwise 465 */ 466 @Override 467 public boolean isCrystallographic() { 468 if (pdbHeader.getExperimentalTechniques()!=null) { 469 return ExperimentalTechnique.isCrystallographic(pdbHeader.getExperimentalTechniques()); 470 } else { 471 // no experimental technique known, we try to guess... 472 if (pdbHeader.getCrystallographicInfo().getSpaceGroup()!=null) { 473 // space group defined but no crystal cell: incomplete info, return false 474 return pdbHeader.getCrystallographicInfo().getCrystalCell() != null && 475 pdbHeader.getCrystallographicInfo().getCrystalCell().isCellReasonable(); 476 } 477 } 478 return false; 479 } 480 481 /** 482 * Whether this Structure is a NMR structure or not. 483 * It will first check the experimental technique and if not present it will try 484 * to guess from the presence of more than 1 model and from b-factors being 0 in first chain of first model 485 * @return true if NMR, false otherwise 486 */ 487 @Override 488 public boolean isNmr() { 489 490 // old implementation was: 491 //return nmrflag; 492 493 if (pdbHeader.getExperimentalTechniques()!=null) { 494 return ExperimentalTechnique.isNmr(pdbHeader.getExperimentalTechniques()); 495 } else { 496 // no experimental technique known, we try to guess... 497 if (nrModels()>1) { 498 if (pdbHeader.getCrystallographicInfo().getSpaceGroup()!=null) { 499 // multimodel, sg defined, but missing cell: must be NMR 500 if (pdbHeader.getCrystallographicInfo().getCrystalCell()==null) 501 return true; 502 // multi-model, sg defined and cell unreasonable: must be NMR 503 if (!pdbHeader.getCrystallographicInfo().getCrystalCell().isCellReasonable()) 504 return true; 505 } else { 506 // multi-model and missing space group: must be NMR 507 return true; 508 } 509 } 510 } 511 return false; 512 } 513 514 /** {@inheritDoc} */ 515 @Override 516 public List<Chain> getChains(int modelIdx){ 517 return getModel(modelIdx); 518 } 519 520 /** {@inheritDoc} */ 521 @Override 522 public List<Chain> getChains(){ 523 if (models.size()==0) { 524 return new ArrayList<>(0); 525 } 526 return getChains(0); 527 528 } 529 530 @Override 531 public List<Chain> getPolyChains() { 532 if (models.size()==0) { 533 return new ArrayList<>(0); 534 } 535 return getPolyChains(0); 536 } 537 538 @Override 539 public List<Chain> getPolyChains(int modelIdx) { 540 return models.get(modelIdx).getPolyChains(); 541 } 542 543 @Override 544 public List<Chain> getNonPolyChains() { 545 if (models.size()==0) { 546 return new ArrayList<>(0); 547 } 548 return getNonPolyChains(0); 549 } 550 551 @Override 552 public List<Chain> getNonPolyChains(int modelIdx) { 553 return models.get(modelIdx).getNonPolyChains(); 554 } 555 556 @Override 557 public List<Chain> getWaterChains() { 558 if (models.size()==0) { 559 return new ArrayList<>(0); 560 } 561 return getWaterChains(0); 562 } 563 564 @Override 565 public List<Chain> getWaterChains(int modelIdx) { 566 return models.get(modelIdx).getWaterChains(); 567 } 568 569 570 571 /** {@inheritDoc} */ 572 @Override 573 public void setChains(int modelnr, List<Chain> chains){ 574 for (Chain c: chains){ 575 c.setStructure(this); 576 } 577 if (models.size()>modelnr) { 578 models.remove(modelnr); 579 } 580 581 Model model = new Model(); 582 model.setChains(chains); 583 models.add(modelnr, model); 584 585 } 586 587 /** Retrieve all Chains belonging to a model . 588 * 589 * @param modelnr an int 590 * @return a List object 591 */ 592 @Override 593 public List<Chain> getModel(int modelnr) { 594 595 return models.get(modelnr).getChains(); 596 } 597 598 /** {@inheritDoc} */ 599 @Override 600 public Chain getChain(String asymId, int modelnr) { 601 602 List<Chain> chains = getChains(modelnr); 603 for (Chain c : chains) { 604 if (c.getId().equals(asymId)) { 605 return c; 606 } 607 } 608 return null; 609 610 } 611 612 /** {@inheritDoc} */ 613 @Override 614 public Chain getChain(String asymId) { 615 616 return getChain(asymId,0); 617 618 } 619 620 @Override 621 public Chain getPolyChain(String asymId) { 622 return getPolyChain(asymId, 0); 623 624 } 625 626 @Override 627 public Chain getPolyChain(String asymId, int modelIdx) { 628 Model model = models.get(modelIdx); 629 if (model==null) { 630 return null; 631 } 632 List<Chain> polyChains = model.getPolyChains(); 633 for (Chain c : polyChains){ 634 if (c.getId().equals(asymId)) 635 return c; 636 } 637 return null; 638 } 639 640 641 @Override 642 public Chain getNonPolyChain(String asymId) { 643 return getNonPolyChain(asymId, 0); 644 } 645 646 @Override 647 public Chain getNonPolyChain(String asymId, int modelIdx) { 648 Model model = models.get(modelIdx); 649 if (model==null) { 650 return null; 651 } 652 653 List<Chain> nonpolyChains = model.getNonPolyChains(); 654 for (Chain c : nonpolyChains){ 655 if (c.getId().equals(asymId)) 656 return c; 657 } 658 659 return null; 660 } 661 662 @Override 663 public Chain getPolyChainByPDB(String authId) { 664 return getPolyChainByPDB(authId, 0); 665 } 666 667 @Override 668 public Chain getPolyChainByPDB(String authId, int modelIdx) { 669 Model model = models.get(modelIdx); 670 if (model==null) { 671 return null; 672 } 673 674 List<Chain> polyChains = model.getPolyChains(); 675 for (Chain c : polyChains){ 676 if (c.getName().equals(authId)) 677 return c; 678 } 679 680 return null; 681 } 682 683 @Override 684 public List<Chain> getNonPolyChainsByPDB(String authId) { 685 return getNonPolyChainsByPDB(authId, 0); 686 } 687 688 @Override 689 public List<Chain> getNonPolyChainsByPDB(String authId, int modelIdx) { 690 List<Chain> chains = new ArrayList<>(); 691 Model model = models.get(modelIdx); 692 if (model==null) { 693 return chains; 694 } 695 696 697 List<Chain> nonpolyChains = model.getNonPolyChains(); 698 for (Chain c : nonpolyChains){ 699 if (c.getName().equals(authId)) 700 chains.add(c); 701 } 702 703 return chains; 704 } 705 706 @Override 707 public Chain getWaterChain(String asymId) { 708 return getWaterChain(asymId, 0); 709 } 710 711 712 @Override 713 public Chain getWaterChain(String asymId, int modelIdx) { 714 Model model = models.get(modelIdx); 715 if (model==null) { 716 return null; 717 } 718 List<Chain> waterChains = model.getWaterChains(); 719 for (Chain c : waterChains){ 720 if (c.getId().equals(asymId)) 721 return c; 722 } 723 return null; 724 } 725 726 727 @Override 728 public Chain getWaterChainByPDB(String authId) { 729 return getWaterChainByPDB(authId, 0); 730 } 731 732 733 @Override 734 public Chain getWaterChainByPDB(String authId, int modelIdx) { 735 Model model = models.get(modelIdx); 736 if (model==null) { 737 return null; 738 } 739 List<Chain> waterChains = model.getWaterChains(); 740 for (Chain c : waterChains){ 741 if (c.getName().equals(authId)) 742 return c; 743 } 744 745 return null; 746 } 747 748 749 750 /** {@inheritDoc} */ 751 @Override 752 public String toPDB() { 753 FileConvert f = new FileConvert(this) ; 754 return f.toPDB(); 755 } 756 757 /** {@inheritDoc} */ 758 @Override 759 public String toMMCIF() { 760 FileConvert f = new FileConvert(this); 761 return f.toMMCIF(); 762 } 763 764 /** {@inheritDoc} */ 765 @Override 766 public boolean hasChain(String authId) { 767 int modelnr = 0; 768 769 List<Chain> chains = getChains(modelnr); 770 for (Chain c : chains) { 771 // we check here with equals because we might want to distinguish between upper and lower case chains! 772 if (c.getId().equals(authId)) { 773 return true; 774 } 775 } 776 return false; 777 } 778 779 /** {@inheritDoc} */ 780 @Override 781 public boolean hasNonPolyChain(String asymId){ 782 int modelnr = 0; 783 784 List<Chain> chains = models.get(modelnr).getNonPolyChains(); 785 for (Chain c : chains) { 786 // we check here with equals because we might want to distinguish between upper and lower case chains! 787 if (c.getId().equals(asymId)) { 788 return true; 789 } 790 } 791 return false; 792 } 793 794 /** {@inheritDoc} */ 795 @Override 796 public boolean hasPdbChain(String authId) { 797 int modelnr = 0; 798 799 List<Chain> chains = getChains(modelnr); 800 for (Chain c : chains) { 801 // we check here with equals because we might want to distinguish between upper and lower case chains! 802 if (c.getName().equals(authId)) { 803 return true; 804 } 805 } 806 return false; 807 } 808 809 /** {@inheritDoc} */ 810 @Override 811 public void setEntityInfos(List<EntityInfo> molList){ 812 this.entityInfos = molList; 813 } 814 815 /** {@inheritDoc} */ 816 @Override 817 public void addEntityInfo(EntityInfo entityInfo) { 818 this.entityInfos.add(entityInfo); 819 } 820 821 /** {@inheritDoc} */ 822 @Override 823 public List<EntityInfo> getEntityInfos() { 824 return entityInfos; 825 } 826 827 /** {@inheritDoc} */ 828 @Override 829 public EntityInfo getEntityById(int entityId) { 830 for (EntityInfo mol : this.entityInfos){ 831 if (mol.getMolId()==entityId){ 832 return mol; 833 } 834 } 835 return null; 836 } 837 838 839 /** {@inheritDoc} */ 840 @Override 841 public List<DBRef> getDBRefs() { 842 return dbrefs; 843 } 844 845 846 /** {@inheritDoc} */ 847 @Override 848 public void setDBRefs(List<DBRef> dbrefs) { 849 if ( dbrefs == null) 850 throw new IllegalArgumentException("trying to set dbrefs to null!"); 851 852 for( DBRef ref : dbrefs){ 853 ref.setParent(this); 854 } 855 this.dbrefs = dbrefs; 856 } 857 858 859 /** {@inheritDoc} */ 860 @Override 861 public PDBHeader getPDBHeader() { 862 return pdbHeader; 863 } 864 865 /** {@inheritDoc} */ 866 @Override 867 public void setPDBHeader(PDBHeader pdbHeader){ 868 this.pdbHeader = pdbHeader; 869 } 870 871 /** {@inheritDoc} */ 872 @Override 873 public List<Bond> getSSBonds(){ 874 return ssbonds; 875 876 } 877 878 /** {@inheritDoc} */ 879 @Override 880 public void setSSBonds(List<Bond> ssbonds){ 881 this.ssbonds = ssbonds; 882 } 883 884 /** 885 * Adds a single disulfide Bond to this structure 886 * 887 * @param ssbond the SSBond. 888 */ 889 @Override 890 public void addSSBond(Bond ssbond){ 891 ssbonds.add(ssbond); 892 } 893 894 /** 895 * Return whether or not the entry has an associated journal article 896 * or publication. The JRNL section is not mandatory and thus may not be 897 * present. 898 * @return flag if a JournalArticle could be found. 899 */ 900 @Override 901 public boolean hasJournalArticle() { 902 return this.pdbHeader.hasJournalArticle(); 903 } 904 905 /** 906 * get the associated publication as defined by the JRNL records in a PDB 907 * file. 908 * @return a JournalArticle 909 */ 910 @Override 911 public JournalArticle getJournalArticle() { 912 return this.pdbHeader.getJournalArticle(); 913 } 914 915 /** 916 * set the associated publication as defined by the JRNL records in a PDB 917 * file. 918 * @param journalArticle the article 919 */ 920 @Override 921 public void setJournalArticle(JournalArticle journalArticle) { 922 this.pdbHeader.setJournalArticle(journalArticle); 923 } 924 925 /** 926 * @return the sites contained in this structure 927 */ 928 929 @Override 930 public List<Site> getSites() { 931 return sites; 932 } 933 934 /** 935 * @param sites the sites to set in the structure 936 */ 937 938 @Override 939 public void setSites(List<Site> sites) { 940 this.sites = sites; 941 } 942 943 /** Caution: we should probably remove this to avoid confusion. Currently this is always an empty list! 944 * 945 * @return a list of Groups listed in the HET records - this will not 946 * include any waters. 947 */ 948 949 /** 950 * Sets a flag to indicate if this structure is a biological assembly 951 * @param biologicalAssembly true if biological assembly, otherwise false 952 * @since 3.2 953 */ 954 @Override 955 public void setBiologicalAssembly(boolean biologicalAssembly) { 956 this.biologicalAssembly = biologicalAssembly; 957 } 958 959 /** 960 * Gets flag that indicates if this structure is a biological assembly 961 * @return the sites contained in this structure 962 * @since 3.2 963 */ 964 @Override 965 public boolean isBiologicalAssembly() { 966 return biologicalAssembly; 967 } 968 969 /** 970 * Sets crystallographic information for this structure 971 * @param crystallographicInfo crystallographic information 972 * @since 3.2 973 */ 974 975 @Override 976 public void setCrystallographicInfo(PDBCrystallographicInfo crystallographicInfo) { 977 this.pdbHeader.setCrystallographicInfo(crystallographicInfo); 978 } 979 980 /** 981 * Gets crystallographic information for this structure 982 * @return PDBCrystallographicInfo crystallographic information 983 * @since 3.2 984 */ 985 @Override 986 public PDBCrystallographicInfo getCrystallographicInfo() { 987 return pdbHeader.getCrystallographicInfo(); 988 } 989 990 /** {@inheritDoc} */ 991 @Override 992 public String getIdentifier() { 993 //1. StructureIdentifier 994 if(getStructureIdentifier() != null) { 995 return getStructureIdentifier().getIdentifier(); 996 } 997 //2. Name 998 if(getName() != null) { 999 return getName(); 1000 } 1001 //3. PDBCode + ranges 1002 return toCanonical().getIdentifier(); 1003 } 1004 1005 /** 1006 * {@inheritDoc} 1007 * @deprecated use {@link #getPdbId()} to get a {@link PdbId} object or getPdbId().getId() to get a {@link String} 1008 */ 1009 @Deprecated 1010 @Override 1011 public String getPDBCode () { 1012 if(pdbId == null) 1013 return null; 1014 return this.pdbId.getId() ; 1015 } 1016 1017 /** {@inheritDoc} 1018 * @deprecated use {@link #setPdbId(PdbId)} 1019 * */ 1020 @Deprecated 1021 @Override 1022 public void setPDBCode(String pdb_id){ 1023 if(pdb_id == null) { 1024 this.pdbId = null; 1025 }else { 1026 pdbId = new PdbId(pdb_id); 1027 } 1028 } 1029 1030 1031 1032 /** {@inheritDoc} 1033 * @since 6.0.0 1034 * */ 1035 public PdbId getPdbId() { 1036 return this.pdbId; 1037 } 1038 1039 /** {@inheritDoc} 1040 * @since 6.0.0 1041 * */ 1042 public void setPdbId(PdbId pdbId) { 1043 this.pdbId = pdbId; 1044 } 1045 1046 @Override 1047 public void resetModels() { 1048 models = new ArrayList<>(); 1049 } 1050 1051 /** 1052 * Creates a SubstructureIdentifier based on the residues in this Structure. 1053 * 1054 * Only the first and last residues of each chain are considered, so chains 1055 * with gaps 1056 * @return A {@link SubstructureIdentifier} with residue ranges constructed from each chain 1057 */ 1058 private SubstructureIdentifier toCanonical() { 1059 StructureIdentifier real = getStructureIdentifier(); 1060 if(real != null) { 1061 try { 1062 return real.toCanonical(); 1063 } catch (StructureException e) { 1064 // generate fake one if needed 1065 } 1066 } 1067 1068 // No identifier set, so generate based on residues present in the structure 1069 List<ResidueRange> range = new ArrayList<>(); 1070 for (Chain chain : getChains()) { 1071 List<Group> groups = chain.getAtomGroups(); 1072 ListIterator<Group> groupsIt = groups.listIterator(); 1073 if(!groupsIt.hasNext()) { 1074 continue; // no groups in chain 1075 } 1076 Group g = groupsIt.next(); 1077 ResidueNumber first = g.getResidueNumber(); 1078 1079 //TODO Detect missing intermediate residues -sbliven, 2015-01-28 1080 //Already better than previous whole-chain representation 1081 1082 // get last residue 1083 while(groupsIt.hasNext()) { 1084 g = groupsIt.next(); 1085 } 1086 ResidueNumber last = g.getResidueNumber(); 1087 1088 range.add(new ResidueRange(chain.getName(),first,last)); 1089 } 1090 return new SubstructureIdentifier(getPdbId(),range); 1091 } 1092 1093}