JAL-2629 hmmsearch # of new sequences now only displayed when specified
[jalview.git] / src / jalview / ws / ebi / HmmerJSONProcessor.java
index d0f3c18..428c498 100644 (file)
@@ -7,9 +7,12 @@ import jalview.datamodel.SequenceFeature;
 import jalview.datamodel.SequenceGroup;
 import jalview.datamodel.SequenceI;
 import jalview.io.FileParse;
+import jalview.viewmodel.AlignmentViewport;
 
 import java.io.IOException;
+import java.util.ArrayList;
 import java.util.HashMap;
+import java.util.List;
 import java.util.Map;
 
 import org.json.simple.JSONArray;
@@ -18,8 +21,16 @@ import org.json.simple.parser.JSONParser;
 
 public class HmmerJSONProcessor
 {
+  /**
+   * result to be annotated. may not be null
+   */
   AlignmentI resultAl;
 
+  /**
+   * viewport on the alignment. may be null at construction time
+   */
+  AlignmentViewport viewAl = null;
+
   public HmmerJSONProcessor(AlignmentI searchResult)
   {
     resultAl = searchResult;
@@ -143,11 +154,16 @@ public class HmmerJSONProcessor
               .get("alimodel"), ppline = (String) dhit.get("alippline");
       //
       int found = 0;
+      SequenceI firsthit = null;
       for (SequenceI hitseq : hits)
       {
         // match alisqfrom,alisqto,seq
         if (hitseq.getStart() == alisqfrom && hitseq.getEnd() == alisqto)
         {
+          if (found == 0)
+          {
+            firsthit = hitseq;
+          }
           found++; // annotated a sequence
           AlignmentAnnotation alipp = parsePosteriorProb(ppline);
           AlignmentAnnotation pval = new AlignmentAnnotation("p-value",
@@ -165,7 +181,9 @@ public class HmmerJSONProcessor
           hitseq.addSequenceFeature(new SequenceFeature(
                   "Pfam Domain Architecture", (hmmrhit.get("archindex"))
                           + " " + (arch = (String) hmmrhit.get("arch")), 0,
-                  0, Integer.valueOf((String) hmmrhit.get("archScore")),
+                  0,
+                  (hmmrhit.get("archScore") != null ? Integer
+                          .valueOf((String) hmmrhit.get("archScore")) : 0f),
                   "HMMER3"));
           addArchGroup(hitseq, arch);
           alipp.setScore(Double.valueOf("" + dhit.get("bitscore")));
@@ -176,17 +194,90 @@ public class HmmerJSONProcessor
           alipp.validateRangeAndDisplay();
         }
       }
-      if (found == 0)
+      // look for other sequences represented by this hit and create rep groups
+      // could be in "pdbs", or ..
+      addRedundantSeqGroup(firsthit, alisqfrom, alisqto,
+              (JSONArray) hmmrhit.get("seqs"), true);
+    }
+  }
+
+  /**
+   * series of operations to perform for the viewpanel associated with the
+   * alignment
+   */
+  private List<Runnable> viewOps = new ArrayList<Runnable>();
+
+  public void updateView(AlignmentViewport view)
+  {
+    viewAl = view;
+    for (Runnable op : viewOps)
+    {
+      op.run();
+    }
+  }
+
+  private void addRedundantSeqGroup(final SequenceI firsthit,
+          long alisqfrom, long alisqto, JSONArray others, boolean justDelete)
+  {
+    if (others != null)
+    {
+      final SequenceGroup repgroup = new SequenceGroup();
+      repgroup.setSeqrep(firsthit);
+      repgroup.addOrRemove(firsthit, false);
+      repgroup.setStartRes(0);
+      repgroup.setEndRes(resultAl.getWidth() - 1);
+      for (Object otherseq : others.toArray(new JSONObject[0]))
       {
-        System.err.println("Warn - no match for json hit " + sname + "/"
-                + alisqfrom + "-" + alisqto);
+        String repseq = (String) ((JSONObject) otherseq).get("dn");
+        SequenceI[] other = resultAl.findSequenceMatch(repseq);
+        if (other != null && other.length > 0)
+        {
+          if (justDelete)
+          {
+            for (SequenceI oth : other)
+            {
+              resultAl.deleteSequence(oth);
+            }
+            ;
+          }
+          else
+          {
+            int ofound = 0;
+            for (SequenceI oth : other)
+            {
+              if (oth.getStart() == alisqfrom && oth.getEnd() == alisqto)
+              {
+                ofound++;
+                repgroup.addSequence(oth, false);
+              }
+            }
+            if (ofound == 0)
+            {
+              System.err.println("Warn - no match for redundant hit "
+                      + repseq + "/" + alisqfrom + "-" + alisqto);
+            }
+            if (ofound > 1)
+            {
+              System.err
+                      .println("Warn - multiple matches for redundant hit "
+                              + repseq + "/" + alisqfrom + "-" + alisqto);
+            }
+          }
+        }
       }
-      if (found > 1)
+      if (repgroup.getSequences().size() > 1)
       {
-        System.err.println("Warn - multiple matches for json hit " + sname
-                + "/" + alisqfrom + "-" + alisqto);
+        // queue a hide operation
+        final HmmerJSONProcessor me = this;
+        viewOps.add(new Runnable()
+        {
+          @Override
+          public void run()
+          {
+            me.viewAl.hideRepSequences(firsthit, repgroup);
+          }
+        });
       }
-      // look for other sequences represented by this hit and create
     }
   }