/* Copyright (c) 2009 The Regents of the University of California. All rights reserved. Permission is hereby granted, without written agreement and without license or royalty fees, to use, copy, modify, and distribute this software and its documentation for any purpose, provided that the above copyright notice and the following two paragraphs appear in all copies of this software. IN NO EVENT SHALL THE UNIVERSITY OF CALIFORNIA BE LIABLE TO ANY PARTY FOR DIRECT, INDIRECT, SPECIAL, INCIDENTAL, OR CONSEQUENTIAL DAMAGES ARISING OUT OF THE USE OF THIS SOFTWARE AND ITS DOCUMENTATION, EVEN IF THE UNIVERSITY OF CALIFORNIA HAS BEEN ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. THE UNIVERSITY OF CALIFORNIA SPECIFICALLY DISCLAIMS ANY WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE. THE SOFTWARE PROVIDED HEREUNDER IS ON AN "AS IS" BASIS, AND THE UNIVERSITY OF CALIFORNIA HAS NO OBLIGATION TO PROVIDE MAINTENANCE, SUPPORT, UPDATES, ENHANCEMENTS, OR MODIFICATIONS.. */ package sequencing; import java.awt.AlphaComposite; import java.awt.event.MouseEvent; import java.util.Date; import java.text.DateFormat; import java.text.SimpleDateFormat; import java.awt.Font; import java.awt.Color; import java.awt.Dimension; import java.awt.Graphics; import java.awt.Graphics2D; import java.awt.RenderingHints; import java.awt.Shape; import java.awt.BasicStroke; import java.awt.Composite; import java.awt.Point; import java.awt.Rectangle; import java.awt.event.MouseListener; import java.awt.geom.Rectangle2D; import java.awt.geom.RoundRectangle2D; import java.awt.geom.Line2D; import java.util.ArrayList; import javax.swing.JFrame; import javax.swing.JPanel; import java.awt.font.FontRenderContext; import java.awt.font.TextLayout; import java.awt.geom.GeneralPath; import java.awt.image.BufferedImage; import sequencing.TraceExtract.AlignedABI; /** * * @author Ben Bubenheim */ public class InteractiveReport extends JPanel { public InteractiveReport(TraceExtract finalTarget, String author, String sName, String resultenum) { auth=author; sampleName=sName; extract = finalTarget; numAbis = extract.getABIList().size(); //setting up window width double totalbases = finalTarget.getSeq().length(); pixPerBase=600/totalbases; for(AlignedABI abi : extract.getABIList()) { System.out.println(abi.name + " " + abi.startOnTarget + " " + abi.endOnTarget); } System.out.println("start and end of target " + finalTarget.getMergeStart() + " " + finalTarget.getMergeEnd()); //***** //pixPerBase=700/(double)t; //***** seqStart=finalTarget.getMergeStart()*pixPerBase; seqSize=(finalTarget.getMergeEnd()-finalTarget.getMergeStart())*pixPerBase; nextRect=(int)seqStart; result=resultenum; //calculate window size y=numAbis*(abiHeight+3)+200; imX=x; imY=y; addMouseListener(new ClickListen()); } private class ClickListen implements MouseListener { public void mouseClicked(MouseEvent e) { Point pointy =e.getPoint(); //If the click was in the sequence box (the green box) show the best trace data for that region if(targetSeqBox.contains(pointy)) { int xpos = pointy.x; double dpos = Math.floor(xpos/pixPerBase); int ipos = (int) dpos; try { showTrace(ipos); } catch(Exception err) { } } } public void mousePressed(MouseEvent e) { } public void mouseReleased(MouseEvent e) { } public void mouseEntered(MouseEvent e) { } public void mouseExited(MouseEvent e) { } } private void showTrace(int position) { System.out.println("Creating a trace at postioin " + position); System.out.println("Num of abis " + extract.getABIList().size()); int bestScore = 0; for(AlignedABI abi : extract.getABIList()) { int indexinABI = position - abi.startOnTarget; if(indexinABI<0) { continue; } if(position>abi.endOnTarget) { continue; } System.out.println("indexinABI " + abi.name + " " + indexinABI); int score = abi.trace.getQualities()[indexinABI]; if(score>bestScore) { bestScore = score; currentTrace = abi; if(abi.isForward) { currPosOnABI = indexinABI; } else { currPosOnABI = abi.trace.getTrace().getSequence().length() - indexinABI; } } } currPosOnTarget = position; System.out.println("best trace is " + currentTrace.name); setBackground(Color.WHITE); repaint(); this.getParent().repaint(); } private void disp(){ frame = new JFrame("Sequencing Report"); frame.getContentPane().add(this); frame.setPreferredSize(new Dimension(630,200+(abiHeight+6)*numAbis)); frame.setDefaultCloseOperation(JFrame.DISPOSE_ON_CLOSE); frame.pack(); frame.setVisible(true); frame.setResizable(false); } //returns the JPanel that would be put into the JFrame, which is what is displayed by the Sequence Analyzer public JPanel getPanel() { return this; } public void showReport(){ disp(); } @Override public void paintComponent(Graphics g) { //Initiate Graphics2D, it's composite, and font context g2 = (Graphics2D) g; g2.setRenderingHint(RenderingHints.KEY_INTERPOLATION, RenderingHints.VALUE_INTERPOLATION_BILINEAR); g2.setRenderingHint(RenderingHints.KEY_ANTIALIASING, RenderingHints.VALUE_ANTIALIAS_ON); FontRenderContext context = g2.getFontRenderContext(); Composite regularOverlay = g2.getComposite(); int targetXStart = (int) (extract.getMergeStart() * pixPerBase); int targetWidth = (int) ((extract.getMergeEnd() - extract.getMergeStart()) * pixPerBase); //Draw the green box for the comparison sequence g2.setColor(green); Rectangle2D.Double readSeq=new Rectangle2D.Double(targetXStart, numAbis*(abiHeight+8), targetWidth, 15); targetSeqBox =readSeq; g2.fill(readSeq); //Paint each mutation red box over the comparison sequence box short[] scores = extract.getBaseScores(); for(int i=extract.getMergeStart(); i<extract.getMergeEnd(); i++) { switch(scores[i]) { case 1: break; //For mutations case 0: g2.setColor(Color.RED); Rectangle2D.Double red=new Rectangle2D.Double(i*pixPerBase, numAbis*(abiHeight+8), 2*pixPerBase, 15); g2.fill(red); break; //For ambiguities case -1: g2.setColor(Color.YELLOW); Rectangle2D.Double yeller=new Rectangle2D.Double(i*pixPerBase, numAbis*(abiHeight+8), 2*pixPerBase, 15); g2.fill(yeller); break; //For insertions case 2: g2.setColor(Color.BLACK); Rectangle2D.Double blacker=new Rectangle2D.Double(i*pixPerBase, numAbis*(abiHeight+8), 2*pixPerBase, 15); g2.fill(blacker); break; //For deletions case 3: g2.setColor(Color.WHITE); Rectangle2D.Double whiter=new Rectangle2D.Double(i*pixPerBase, numAbis*(abiHeight+8), 2*pixPerBase, 15); g2.fill(whiter); break; } } nextRect=(int)seqStart; //Add ruler underneath the comparison sequence int tickSize=50; if(extract.getSeq().length()>1000){ tickSize=100; } double ticktemp=(extract.getMergeEnd()-extract.getMergeStart())/(double)tickSize; int ticks=0; if(ticktemp==(int)ticktemp){ ticks=(int)ticktemp; } else{ ticks=((int)ticktemp)+1; } int rulerL=(int)((ticks*tickSize)*pixPerBase); g2.setColor(Color.black); Line2D.Double ruler=new Line2D.Double(targetXStart, numAbis*(abiHeight+6)+25, seqStart+rulerL, numAbis*(abiHeight+6)+25); g2.draw(ruler); //Draw the ticks and labels below them for(int i=0; i<ticks+1; i++){ //Put in tick lines Line2D.Double temp=new Line2D.Double((i*tickSize*pixPerBase)+targetXStart, numAbis*(abiHeight+6)+25, (i*tickSize*pixPerBase)+seqStart, numAbis*(abiHeight+6)+30); g2.draw(temp); //Put in base count labels TextLayout layout = new TextLayout(""+(i*tickSize), arial10, context); Double textbarwidthdiv2 = layout.getBounds().getWidth()/2; int xpos = (int) (((i * tickSize * pixPerBase) + seqStart) - textbarwidthdiv2 + 1); int ypos = numAbis*(abiHeight+6)+41; layout.draw(g2, xpos, ypos); } //For each ABI, put in shape, label, and ticks for(int i=0; i<numAbis; i++){ AlignedABI alabi = extract.getABIList().get(i); boolean isRevComp = alabi.isForward; int xStart=(int)(alabi.startOnTarget*pixPerBase); int xWidth=(int)((alabi.endOnTarget-alabi.startOnTarget)*pixPerBase); int yStart=((numAbis-(i+1))*(abiHeight+6))+3; //Put in the background round rectangle for the ABI RoundRectangle2D.Double r=new RoundRectangle2D.Double(xStart, yStart, xWidth, abiHeight, 15, 25); g2.setColor(lightBlue); g2.fill(r); //Put in ambiguous base ticks g2.setComposite(AlphaComposite.getInstance(AlphaComposite.SRC_OVER, 0.3f)); g2.setColor(Color.WHITE); for(int n=0; n< alabi.trace.getSeq().length(); n++){ if(alabi.trace.getSeq().charAt(n)=='N'){ int xpos = xStart+(int)(n*pixPerBase); int ypos = ((numAbis-(i+1))*(abiHeight+6))+4; Rectangle2D.Double yell=new Rectangle2D.Double(xpos,ypos , 2*pixPerBase, abiHeight-2); g2.fill(yell); } } g2.setComposite(regularOverlay); //Draw the round rectangle surrounding each ABI shape g2.setColor(darkBlue); g2.setStroke(new BasicStroke(2)); g2.draw(r); //Put in name labels of ABI's g2.setColor(Color.BLACK); String abiLabel = alabi.name; TextLayout layout = new TextLayout(abiLabel, arial12bold, context); int width = (int) (alabi.trace.getSeq().length() * pixPerBase); int xpos= (int) (xStart + width / 2 - layout.getBounds().getWidth() / 2); int ypos=yStart+16; layout.draw(g2, xpos, ypos); //Put in directionality triangles GeneralPath gp = new GeneralPath(); ypos-=7; if(isRevComp) { xpos-=width/2 - layout.getBounds().getWidth()/2 - 10; gp.moveTo(xpos, ypos); gp.lineTo(xpos+5, ypos+3); } else { xpos+=width/2 + layout.getBounds().getWidth()/2 - 7; gp.moveTo(xpos, ypos); gp.lineTo(xpos-5, ypos+3); } gp.lineTo(xpos, ypos+6); gp.lineTo(xpos, ypos); g2.draw (gp); } //Put in result text int fat=(int)((x/2)-seqStart); int xpos = (int)seqStart; int ypos=numAbis*(abiHeight+6) + 90; TextLayout layout = new TextLayout("Sample: "+sampleName, arial12bold, context); layout.draw(g2, xpos, ypos); ypos += 15; layout = new TextLayout("Result: "+result, arial12bold, context); layout.draw(g2, xpos, ypos); ypos += 15; layout = new TextLayout("Author: "+auth, arial12bold, context); layout.draw(g2, xpos, ypos); ypos += 15; layout = new TextLayout(getDateTime(), arial12bold, context); layout.draw(g2, xpos, ypos); //Put in trace paintTrace(g2); } private void paintTrace(Graphics2D g2) { if(g2==null) { System.out.println("Have no G2"); return; } if(currentTrace==null) { System.out.println("Have no trace"); return; } //Paint a white background int ypos=numAbis*(abiHeight+6) + 55; Rectangle boundingBox = new Rectangle(220,ypos, 320,105); g2.setColor(Color.WHITE); g2.fill(boundingBox); //Paint the trace over that try { ABITrace abi = currentTrace.trace.getTrace(); BufferedImage bi = abi.getImage(105, 2, currPosOnABI); g2.drawImage(bi, null, 220,ypos); g2.setColor(Color.BLACK); g2.setStroke(new BasicStroke(1)); g2.draw(boundingBox); } catch(Exception e) { return; } //Put the sequence over that FontRenderContext context = g2.getFontRenderContext(); for(int i=0; i<13; i++) { if(currPosOnTarget<7) { continue; } if(currPosOnTarget>currentTrace.trace.getTrace().getSequence().length()-6) { continue; } char base = extract.getSeq().charAt(currPosOnTarget + i -4); TextLayout layout = new TextLayout(""+base, arial12bold, context); layout.draw(g2, 230+23*i, ypos+95); } } private String getDateTime() { DateFormat dateFormat = new SimpleDateFormat("MM/dd/yyyy"); Date date = new Date(); return dateFormat.format(date); } /////////////////////////////////////////////////////////////////// //// private variables //// private String result, auth, sampleName; TraceExtract extract; private int numAbis; private ArrayList<TextLayout> toBeDeleted; private Graphics2D g2; private JFrame frame; private int nextRect; private int length, paintNum; private Shape targetSeqBox; private AlignedABI currentTrace; private int currPosOnTarget; private int currPosOnABI; private ArrayList<Shape> regions = new ArrayList<Shape>(); private int y; private double pixPerBase; private double seqStart; private double seqSize; private int x; private int imX, imY; private static final int abiHeight=23; private static final Color green=new Color(8, 153, 72); private static final Color lightBlue=new Color(183, 226, 224); private static final Color darkBlue=new Color(108, 172, 183); private static final Color yellow=new Color(231, 208, 62); private static final Font arial10=new Font("Arial", Font.PLAIN, 10); private static final Font arial12=new Font("Arial", Font.PLAIN, 12); private static final Font arial14=new Font("Arial", Font.PLAIN, 14); private static final Font arial12bold=new Font("Arial", Font.BOLD, 12); }