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.contact;
022
023import org.biojava.nbio.structure.Atom;
024
025import java.io.Serializable;
026import java.util.*;
027
028
029/**
030 * A set of atom-atom contacts to hold the results of intra and inter-chain contact calculations
031 *
032 * @author duarte_j
033 *
034 */
035public class AtomContactSet implements Serializable, Iterable<AtomContact> {
036
037
038        private static final long serialVersionUID = 1L;
039
040        private HashMap<Pair<AtomIdentifier>, AtomContact> contacts;
041        private double cutoff;
042
043        public AtomContactSet(double cutoff) {
044                this.cutoff = cutoff;
045                this.contacts = new HashMap<>();
046        }
047
048        public void add(AtomContact contact) {
049                this.contacts.put(getAtomIdPairFromContact(contact), contact);
050        }
051
052        public void addAll(Collection<AtomContact> list) {
053                for (AtomContact contact:list) {
054                        this.contacts.put(getAtomIdPairFromContact(contact), contact);
055                }
056        }
057
058        public boolean hasContact(Atom atom1, Atom atom2) {
059                return hasContact(
060                                        new AtomIdentifier(atom1.getPDBserial(),atom1.getGroup().getChainId()),
061                                        new AtomIdentifier(atom2.getPDBserial(),atom2.getGroup().getChainId()) );
062        }
063
064        public boolean hasContact(AtomIdentifier atomId1, AtomIdentifier atomId2) {
065                return contacts.containsKey(new Pair<AtomIdentifier>(atomId1,atomId2));
066        }
067
068        /**
069         * Returns the corresponding AtomContact or null if no contact exists between the 2 given atoms
070         * @param atom1
071         * @param atom2
072         * @return
073         */
074        public AtomContact getContact(Atom atom1, Atom atom2) {
075                return contacts.get(new Pair<AtomIdentifier>(
076                                new AtomIdentifier(atom1.getPDBserial(),atom1.getGroup().getChainId()),
077                                new AtomIdentifier(atom2.getPDBserial(),atom2.getGroup().getChainId()) ));
078        }
079
080        public int size() {
081                return contacts.size();
082        }
083
084        @Override
085        public Iterator<AtomContact> iterator() {
086                return contacts.values().iterator();
087        }
088
089        private Pair<AtomIdentifier> getAtomIdPairFromContact(AtomContact contact) {
090                Pair<AtomIdentifier> pair = new Pair<>(
091                                new AtomIdentifier(contact.getPair().getFirst().getPDBserial(),contact.getPair().getFirst().getGroup().getChainId()),
092                                new AtomIdentifier(contact.getPair().getSecond().getPDBserial(),contact.getPair().getSecond().getGroup().getChainId()));
093
094                return pair;
095        }
096
097        /**
098         * Returns true if at least 1 contact from this set is within the given distance.
099         * Note that if the distance given is larger than the distance cutoff used to
100         * calculate the contacts then nothing will be found.
101         * @param distance
102         * @return
103         * @throws IllegalArgumentException if given distance is larger than distance cutoff
104         * used for calculation of contacts
105         */
106        public boolean hasContactsWithinDistance(double distance) {
107
108                if (distance>=cutoff)
109                        throw new IllegalArgumentException("Given distance "+
110                                        String.format("%.2f", distance)+" is larger than contacts' distance cutoff "+
111                                        String.format("%.2f", cutoff));
112
113                for (AtomContact contact:this.contacts.values()) {
114                        if (contact.getDistance()<distance) {
115                                return true;
116                        }
117                }
118                return false;
119        }
120
121        /**
122         * Returns the list of contacts from this set that are within the given distance.
123         * @param distance
124         * @return
125         * @throws IllegalArgumentException if given distance is larger than distance cutoff
126         * used for calculation of contacts
127         */
128        public List<AtomContact> getContactsWithinDistance(double distance) {
129
130                if (distance>=cutoff)
131                        throw new IllegalArgumentException("Given distance "+
132                                        String.format("%.2f", distance)+" is larger than contacts' distance cutoff "+
133                                        String.format("%.2f", cutoff));
134
135                List<AtomContact> list = new ArrayList<>();
136                for (AtomContact contact:this.contacts.values()) {
137                        if (contact.getDistance()<distance) {
138                                list.add(contact);
139                        }
140                }
141                return list;
142        }
143}