001/* This class is based on the original FATCAT implementation by 002 * <pre> 003 * Yuzhen Ye & Adam Godzik (2003) 004 * Flexible structure alignment by chaining aligned fragment pairs allowing twists. 005 * Bioinformatics vol.19 suppl. 2. ii246-ii255. 006 * https://www.ncbi.nlm.nih.gov/pubmed/14534198 007 * </pre> 008 * 009 * Thanks to Yuzhen Ye and A. Godzik for granting permission to freely use and redistribute this code. 010 * 011 * This code may be freely distributed and modified under the 012 * terms of the GNU Lesser General Public Licence. This should 013 * be distributed with the code. If you do not have a copy, 014 * see: 015 * 016 * http://www.gnu.org/copyleft/lesser.html 017 * 018 * Copyright for this code is held jointly by the individual 019 * authors. These should be listed in @author doc comments. 020 * 021 * 022 * Created on Jun 17, 2009 023 * Created by Andreas Prlic - RCSB PDB 024 * 025 */ 026 027package org.biojava.nbio.structure.align.fatcat.calc; 028 029import org.biojava.nbio.structure.*; 030import org.biojava.nbio.structure.align.model.AFP; 031import org.biojava.nbio.structure.align.model.AFPChain; 032import org.biojava.nbio.structure.geometry.SuperPositions; 033import org.biojava.nbio.structure.jama.Matrix; 034 035import java.util.ArrayList; 036import java.util.List; 037 038/** 039 * A class that performs calculations on AFPChains 040 * 041 * @author Andreas Prlic 042 * 043 */ 044public class AFPCalculator 045{ 046 public static final boolean debug = FatCatAligner.debug; 047 048 049 public static void extractAFPChains(FatCatParameters params, AFPChain afpChain,Atom[] ca1,Atom[] ca2) throws StructureException { 050 051 List<AFP> afpSet = new ArrayList<>(); 052 afpChain.setAfpSet(afpSet); 053 054 if ( debug ) 055 System.err.println("nr of atoms ca1: " + ca1.length + " ca2: " + ca2.length); 056 057 058 059 int p1, p2; 060 @SuppressWarnings("unused") 061 int n0, n, n1, n2; 062 double filter1; 063 double rmsd = 0; 064 065 Matrix r = new Matrix(3,3); 066 Atom t = new AtomImpl(); 067 068 069 int sparse = params.getSparse(); 070 int maxTra = params.getMaxTra(); 071 int fragLen = params.getFragLen(); 072 double disFilter = params.getDisFilter(); 073 double rmsdCut = params.getRmsdCut(); 074 double badRmsd = params.getBadRmsd(); 075 double fragScore = params.getFragScore(); 076 077 int add = sparse + 1; //if add > 1, use sparse sampling 078 n0 = n = n1 = n2 = 0; 079 080 int minLen = 0; 081 082 int prot1Length = ca1.length; 083 int prot2Length = ca2.length; 084 085 if(prot1Length < prot2Length) 086 minLen = prot1Length; 087 else 088 minLen = prot2Length; 089 afpChain.setMinLen(minLen); 090 091 afpChain.setBlockResList(new int[maxTra+1][2][minLen]); 092 afpChain.setFocusRes1(new int[minLen]); 093 afpChain.setFocusRes2(new int[minLen]); 094 095 for(p1 = 0; p1 < prot1Length - fragLen; p1 += add ) { 096 for(p2 = 0; p2 < prot2Length - fragLen; p2 += add) { 097 n0 ++; 098 filter1 = getEnd2EndDistance(ca1, ca2, p1, p1 + fragLen - 1, p2, p2 + fragLen - 1); 099 //difference bewteen end-to-end distances 100 if(filter1 > disFilter) { n1 ++; continue; } 101 boolean filter2 = filterTerminal(ca1,ca2, p1, p1 + fragLen - 1, p2, p2 + fragLen - 1, fragLen, minLen); 102 if(filter2) { 103 n2 ++; 104 continue; 105 106 } //be cautious to use this filter !! 107 108 // here FATCAT does a a jacobi transformation 109 //rmsd = kearsay(fragLen, ca1[p1], ca2[p2], r, t); 110 // we use the BioJava SVD instead... 111 112 // 113 rmsd = getRmsd(ca1,ca2,fragLen, p1,p2,r,t); 114 115 if(rmsd < rmsdCut) { 116 AFP afptmp = new AFP(); 117 afptmp.setP1(p1); 118 afptmp.setP2(p2); 119 afptmp.setFragLen(fragLen); 120 afptmp.setRmsd(rmsd); 121 afptmp.setM(r); 122 afptmp.setT(t.getCoords()); 123 afptmp.setScore(scoreAfp(afptmp,badRmsd,fragScore)); 124 afpSet.add(afptmp); 125 n ++; 126 } 127 } 128 } 129 130 int afpNum = afpSet.size(); 131 132 if(debug) { 133 String msg = String.format("possible AFP-pairs %d, remain %d after filter 1 remove %d; filter 2 remove %d\n", 134 n0, afpNum, n1, n2); 135 System.err.println(msg); 136 } 137 138 139 } 140 141 /** 142 * filter 1 for AFP extration: the distance of end-to-end 143 * @param p1b 144 * @param p1e 145 * @param p2b 146 * @param p2e 147 * @return 148 */ 149 private static final double getEnd2EndDistance(Atom[] ca1, Atom[] ca2, int p1b, int p1e, int p2b, int p2e) 150 { 151 152 double min = 99; 153 double dist1 = Calc.getDistance(ca1[p1b], ca1[p1e]); 154 double dist2 = Calc.getDistance(ca2[p2b], ca2[p2e]); 155 min = dist1 - dist2; 156 157 return Math.abs(min); 158 } 159 160 /** 161 * filter 2 for AFP extration: the context 162 * @param p1b 163 * @param p1e 164 * @param p2b 165 * @param p2e 166 * @return 167 */ 168 169 private static final boolean filterTerminal(Atom[] ca1, Atom[] ca2, int p1b, int p1e, int p2b, int p2e, int fragLen, int minLen) 170 { 171 int d1 = (p1b < p2b)?p1b:p2b; 172 int d2 = (ca1.length - p1e) < (ca2.length - p2e)?(ca1.length - p1e):(ca2.length - p2e); 173 int d3 = d1 + d2 + fragLen; //maximum alignment length from current AFP 174 175 176 /// DO NOT DO Math.round() this will give different results to FATCAT.... 177 int d4 = (int)(0.3 * minLen); 178 179 return d3 < d4; 180 181 } 182 183 private static final double getRmsd(Atom[] ca1, Atom[] ca2, int fragLen, 184 int p1, int p2, Matrix m, Atom t) throws StructureException { 185 186 187 double rmsd = 99.9; 188 Atom[] catmp1 = getFragment(ca1, p1, fragLen,false); 189 Atom[] catmp2 = getFragment(ca2, p2, fragLen,false); 190 191 if ( catmp1 == null) { 192 System.err.println("could not get fragment for ca1 " + p1 + " " + fragLen ); 193 return rmsd; 194 } 195 196 if ( catmp2 == null) { 197 System.err.println("could not get fragment for ca2 " + p2 + " " + fragLen ); 198 return rmsd; 199 } 200 201 return SuperPositions.getRmsd(Calc.atomsToPoints(catmp1), 202 Calc.atomsToPoints(catmp2)); 203 } 204 205 /** get a continue subset of Atoms based by the starting position and the length 206 * 207 * @param caall 208 * @param pos ... the start position 209 * @param fragmentLength .. the length of the subset to extract. 210 * @param clone: returns a copy of the atom (in case the coordinate get manipulated...) 211 * @return an Atom[] array 212 */ 213 private static final Atom[] getFragment(Atom[] caall, int pos, int fragmentLength , 214 boolean clone){ 215 216 if ( pos+fragmentLength > caall.length) 217 return null; 218 219 Atom[] tmp = new Atom[fragmentLength]; 220 221 for (int i=0;i< fragmentLength;i++){ 222 if (clone){ 223 tmp[i] = (Atom)caall[i+pos].clone(); 224 } else { 225 tmp[i] = caall[i+pos]; 226 } 227 } 228 return tmp; 229 230 } 231 232 233 /** 234 * Assign score to each AFP 235 */ 236 237 private static final double scoreAfp(AFP afp, double badRmsd, double fragScore) 238 { 239 //longer AFP with low rmsd is better 240 double s, w; 241 //s = (rmsdCut - afptmp.rmsd) * afptmp.len; //the same scroing strategy as that in the post-processing 242 w = afp.getRmsd() / badRmsd; 243 w = w * w; 244 s = fragScore * (1.0 - w); 245 return s; 246 } 247 248 //------------------------------------------------------------------ 249 //Sort the AFPs in increase of their diagonals(i,j) 250 //------------------------------------------------------------------ 251 public static final void sortAfps(AFPChain afpChain, Atom[] ca1, Atom[] ca2) 252 { 253 254 255 List<AFP> afpSet = afpChain.getAfpSet(); 256 257 if ( debug) 258 System.err.println("entering sortAfps"); 259 260 int pro1Len = ca1.length; 261 int pro2Len = ca2.length; 262 263 afpChain.setAfpIndex( new int[pro1Len][pro2Len]); //the index of (i,j) pair in AFP list, otherwise -1 264 afpChain.setAfpAftIndex( new int[pro1Len][pro2Len]); //the index of AFP (i,j*) nearest to (i,j), j*<j. if a AFP exits for (i,j), it equals to afpIndex 265 afpChain.setAfpBefIndex( new int[pro1Len][pro2Len]); //the index of AFP (i,j*) nearest to (i,j), j*>j. if a AFP exits for (i,j), it equals to afpIndex 266 267 int[][] afpIndex = afpChain.getAfpIndex(); 268 int[][] afpAftIndex = afpChain.getAfpAftIndex(); 269 int[][] afpBefIndex = afpChain.getAfpBefIndex(); 270 271 for(int i = 0; i < pro1Len; i ++) { 272 for(int j = 0; j < pro2Len; j ++) { 273 274 afpIndex[i][j] = afpAftIndex[i][j] = afpBefIndex[i][j] = -1; 275 } 276 } 277 278 //index the AFP for easy extraction of compatible AFPs 279 int afpNum = afpSet.size(); 280 281 int b0 = 0; 282 for(int a = 0; a < afpNum; a ++) { 283 if(a == afpNum - 1 || afpSet.get(a).getP1() != afpSet.get(a+1).getP1()) { 284 int i = afpSet.get(a).getP1(); 285 for(int b = b0; b <= a; b ++) { 286 int j = afpSet.get(b).getP2(); 287 afpIndex[i][j]=b ; 288 afpBefIndex[i][j]=b; 289 afpAftIndex[i][j]=b; 290 if(afpSet.get(b).getP1() != i) { 291 System.err.println(String.format("Warning: wrong afp index %d %d\n", i, afpSet.get(b).getP1())); 292 return; 293 } 294 } 295 for(int k = 1; k < pro2Len; k ++) { 296 if( afpBefIndex[i][k] == -1){ 297 afpBefIndex[i][k] = afpBefIndex[i][k-1]; 298 } 299 } 300 for(int k = pro2Len - 2; k >= 0; k --) { 301 if(afpAftIndex[i][k] == -1) { 302 afpAftIndex[i][k] = afpAftIndex[i][k+1]; 303 } 304 } 305 b0 = a + 1; 306 } 307 } 308 309 if ( debug) 310 System.err.println("done sortAfps"); 311 312 313 } 314}