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.io;
022
023import java.text.DecimalFormat;
024import java.util.ArrayList;
025import java.util.List;
026import org.biojava.nbio.structure.Atom;
027import org.biojava.nbio.structure.Bond;
028import org.biojava.nbio.structure.Group;
029
030
031public class GroupToSDF {
032
033
034        public String getText(Group thisGroup) {
035                // Fuction to convert a Group to a strign of  the MDL molnlock
036                StringBuilder sb = new StringBuilder();
037                sb.append(getHeader(thisGroup));
038                sb.append(getCtab(thisGroup));
039                return sb.toString();
040        }
041
042        private String getCtab(Group thisGroup){
043                DecimalFormat df = new DecimalFormat("0.0000");
044                // The thre string builders for the three parts of the MDL block
045                StringBuilder header = new StringBuilder();
046                StringBuilder atomList = new StringBuilder();
047                StringBuilder bondOrders = new StringBuilder();
048                int numBonds = 0;
049                List<Atom> atoms = thisGroup.getAtoms();
050                for(Atom a: thisGroup.getAtoms()){
051                        /// ALL SHOULD BE TO FOUR DECIMAL PLACES WITH A CERTAIN NUMBER OF SPACES BEFOREa
052                        String spaceX = getSpace(10, df.format(a.getX()));
053                        String spaceY = getSpace(10, df.format(a.getY()));
054                        String spaceZ = getSpace(10, df.format(a.getZ()));
055                        String spaceEle = getSpace(4, a.getElement().toString());
056                        atomList.append(spaceX+df.format(a.getX())+spaceY+df.format(a.getY())+spaceZ+df.format(a.getZ())+" "+a.getElement().toString()+spaceEle+"0  0  0  0  0  0  0  0  0  0  0  0\n");
057                        for(Bond b: a.getBonds()){
058                                Atom otherAtom = b.getOther(a);
059                                if(atoms.indexOf(otherAtom)>=atoms.indexOf(a)){
060                                        continue;
061                                }
062                                if(atoms.indexOf(otherAtom)<0){
063                                        continue;
064                                }
065                                if(atoms.indexOf(a)<0){
066                                        continue;
067                                }
068                                numBonds++;
069                                // Deal with the index infromation
070                                String indexOther = Integer.toString(atoms.indexOf(otherAtom)+1);
071                                String index = Integer.toString(atoms.indexOf(a)+1);
072                                String order = Integer.toString(b.getBondOrder());
073                                String spaceIndOne = getSpace(3, index);
074                                String spaceIndTwo = getSpace(3, indexOther);
075                                String spaceOrder = getSpace(3, order);
076                                bondOrders.append(spaceIndOne).append(index).append(spaceIndTwo).append(indexOther).append(spaceOrder).append(order).append("  0\n");
077                        }
078                }
079                StringBuilder outString = new  StringBuilder();
080                // Add the first line now
081                String spaceNumAtoms = getSpace(3, Integer.toString(thisGroup.getAtoms().size()));
082                String spaceNumBonds = getSpace(3, Integer.toString(numBonds));
083                header.append("\n");
084                header.append(spaceNumAtoms).append(thisGroup.getAtoms().size()).append(spaceNumBonds).append(numBonds).append("  0  0  0  0  0  0  0  0999 V2000\n");
085                // Now add the header, atom, bond and charge information togeyher
086                outString.append(header.toString());
087                outString.append(atomList.toString());
088                outString.append(bondOrders.toString());
089                outString.append(getCharges(thisGroup));
090                // Add the final line and the $$$$ delimiter
091                outString.append("M  END\n$$$$");
092                // Get the string and return it
093                return outString.toString();
094        }
095
096        private Object getCharges(Group thisGroup) {
097                // Get the array of charges
098                List<Number> chargeList = getAtomCharges(thisGroup);
099                // Build the string
100                StringBuilder outS = new StringBuilder();
101                int chargeCount=0;
102                // Loop through the charges -> maximum 8 charges per line
103                for(int i =0; i<chargeList.size();i++){
104                        short charge = (Short) chargeList.get(i);
105                        if(charge!=0){
106                                if(chargeCount==0){
107                                        outS.append("M  CHG   N");
108                                }
109                                outS.append(getSpace(4, Integer.toString(i))+(i+1));
110                                outS.append(getSpace(4, Short.toString(charge))+charge);
111                                chargeCount++;
112                        }
113                        //
114                        if(chargeCount==8){
115                                outS.append("\n");
116                                outS.replace(0, 10, "M  CHG   "+chargeCount);
117                                chargeCount=0;
118                        }
119
120                }
121                if(chargeCount==0){
122                        return "";
123                }
124                // Now add the charge count
125                outS.replace(0, 10, "M  CHG  "+chargeCount);
126                outS.append("\n");
127                // Now return the string
128                return outS.toString();
129        }
130
131        private static List<Number> getAtomCharges(Group group) {
132                // The list to store the answer
133                List<Number> outArr = new ArrayList<>();
134                // Get the atom charge Information
135                for(Atom a: group.getAtoms()){
136                        outArr.add(a.getCharge());
137                }
138                return outArr;
139        }
140
141        private String getSpace(int inputNum, String format) {
142                // Function to get the spaces between numbers
143                StringBuilder sb = new StringBuilder();
144                for(int i=0; i<(inputNum-format.length());i++){
145                        sb.append(" ");
146                }
147                return sb.toString();
148        }
149
150        private String getHeader(Group thisGroup) {
151                // Make the header info for the start of the block
152                StringBuilder sb = new StringBuilder();
153                sb.append(thisGroup.getPDBName()).append("\n");
154                sb.append("Made by BioJava");
155                sb.append("\n");
156                return sb.toString();
157        }
158}