X-Git-Url: http://source.jalview.org/gitweb/?a=blobdiff_plain;f=src%2Fjalview%2Fgui%2FOverviewPanel.java;h=d66bbf056635f107155c50b86bc9c00b72c41264;hb=fc8a59878075498420187716fde697dfc81ff490;hp=f8299bf79438dad8b790a7d0ead9af0537ed7b85;hpb=a45774ee31d9f35d4eff46d54d7deab719afb092;p=jalview.git diff --git a/src/jalview/gui/OverviewPanel.java b/src/jalview/gui/OverviewPanel.java index f8299bf..d66bbf0 100755 --- a/src/jalview/gui/OverviewPanel.java +++ b/src/jalview/gui/OverviewPanel.java @@ -1,151 +1,209 @@ /* - * Jalview - A Sequence Alignment Editor and Viewer (Version 2.7) - * Copyright (C) 2011 J Procter, AM Waterhouse, G Barton, M Clamp, S Searle + * Jalview - A Sequence Alignment Editor and Viewer ($$Version-Rel$$) + * Copyright (C) $$Year-Rel$$ The Jalview Authors * * This file is part of Jalview. * * Jalview is free software: you can redistribute it and/or * modify it under the terms of the GNU General Public License - * as published by the Free Software Foundation, either version 3 of the License, or (at your option) any later version. - * + * as published by the Free Software Foundation, either version 3 + * of the License, or (at your option) any later version. + * * Jalview is distributed in the hope that it will be useful, but * WITHOUT ANY WARRANTY; without even the implied warranty * of MERCHANTABILITY or FITNESS FOR A PARTICULAR * PURPOSE. See the GNU General Public License for more details. * - * You should have received a copy of the GNU General Public License along with Jalview. If not, see . + * You should have received a copy of the GNU General Public License + * along with Jalview. If not, see . + * The Jalview Authors are detailed in the 'AUTHORS' file. */ package jalview.gui; -import java.awt.*; -import java.awt.event.*; -import java.awt.image.*; -import javax.swing.*; +import jalview.bin.Cache; +import jalview.util.MessageManager; +import jalview.util.Platform; +import jalview.viewmodel.OverviewDimensions; +import jalview.viewmodel.OverviewDimensionsHideHidden; +import jalview.viewmodel.OverviewDimensionsShowHidden; +import jalview.viewmodel.ViewportListenerI; + +import java.awt.BorderLayout; +import java.awt.Cursor; +import java.awt.Dimension; +import java.awt.event.ActionEvent; +import java.awt.event.ActionListener; +import java.awt.event.ComponentAdapter; +import java.awt.event.ComponentEvent; +import java.awt.event.MouseAdapter; +import java.awt.event.MouseEvent; +import java.awt.event.MouseMotionAdapter; +import java.beans.PropertyChangeEvent; + +import javax.swing.JCheckBoxMenuItem; +import javax.swing.JPanel; +import javax.swing.JPopupMenu; +import javax.swing.SwingUtilities; /** - * DOCUMENT ME! + * Panel displaying an overview of the full alignment, with an interactive box + * representing the viewport onto the alignment. * * @author $author$ * @version $Revision$ */ -public class OverviewPanel extends JPanel implements Runnable +public class OverviewPanel extends JPanel + implements Runnable, ViewportListenerI { - BufferedImage miniMe; - - AlignViewport av; - - AlignmentPanel ap; - - float scalew = 1f; - - float scaleh = 1f; - - int width; + private OverviewDimensions od; - int sequencesHeight; + private OverviewCanvas oviewCanvas; - int graphHeight = 20; + private AlignViewport av; - int boxX = -1; + private AlignmentPanel ap; - int boxY = -1; + private JCheckBoxMenuItem displayToggle; - int boxWidth = -1; + private boolean showHidden = true; - int boxHeight = -1; - - boolean resizing = false; - - // Can set different properties in this seqCanvas than - // main visible SeqCanvas - SequenceRenderer sr; - - FeatureRenderer fr; + private boolean draggingBox = false; /** * Creates a new OverviewPanel object. * - * @param ap - * DOCUMENT ME! + * @param alPanel + * The alignment panel which is shown in the overview panel */ - public OverviewPanel(AlignmentPanel ap) + public OverviewPanel(AlignmentPanel alPanel) { - this.av = ap.av; - this.ap = ap; - setLayout(null); + this.av = alPanel.av; + this.ap = alPanel; - sr = new SequenceRenderer(av); - sr.renderGaps = false; - sr.forOverview = true; - fr = new FeatureRenderer(ap); - - // scale the initial size of overviewpanel to shape of alignment - float initialScale = (float) av.alignment.getWidth() - / (float) av.alignment.getHeight(); - - if (av.conservation == null) + showHidden = Cache.getDefault("SHOW_OV_HIDDEN_AT_START", true); + if (showHidden) { - graphHeight = 0; - } - - if (av.alignment.getWidth() > av.alignment.getHeight()) - { - // wider - width = 400; - sequencesHeight = (int) (400f / initialScale); - if (sequencesHeight < 40) - { - sequencesHeight = 40; - } + od = new OverviewDimensionsShowHidden(av.getRanges(), + (av.isShowAnnotation() + && av.getAlignmentConservationAnnotation() != null)); } else { - // taller - width = (int) (400f * initialScale); - sequencesHeight = 300; - - if (width < 120) - { - width = 120; - } + od = new OverviewDimensionsHideHidden(av.getRanges(), + (av.isShowAnnotation() + && av.getAlignmentConservationAnnotation() != null)); } + setSize(od.getWidth(), od.getHeight()); + + oviewCanvas = new OverviewCanvas(od, av); + setLayout(new BorderLayout()); + add(oviewCanvas, BorderLayout.CENTER); + + av.getRanges().addPropertyChangeListener(this); + addComponentListener(new ComponentAdapter() { + @Override public void componentResized(ComponentEvent evt) { - if ((getWidth() != width) - || (getHeight() != (sequencesHeight + graphHeight))) + if ((getWidth() != od.getWidth()) + || (getHeight() != (od.getHeight()))) { updateOverviewImage(); + setBoxPosition(); } } }); addMouseMotionListener(new MouseMotionAdapter() { + @Override public void mouseDragged(MouseEvent evt) { - if (!av.wrapAlignment) + if (!SwingUtilities.isRightMouseButton(evt)) { - // TODO: feature: jv2.5 detect shift drag and update selection from - // it. - boxX = evt.getX(); - boxY = evt.getY(); - checkValid(); + if (draggingBox) + { + // set the mouse position as a fixed point in the box + // and drag relative to that position + od.adjustViewportFromMouse(evt.getX(), evt.getY(), + av.getAlignment().getHiddenSequences(), + av.getAlignment().getHiddenColumns()); + } + else + { + od.updateViewportFromMouse(evt.getX(), evt.getY(), + av.getAlignment().getHiddenSequences(), + av.getAlignment().getHiddenColumns()); + } + } + } + + @Override + public void mouseMoved(MouseEvent evt) + { + if (od.isPositionInBox(evt.getX(), evt.getY())) + { + // display drag cursor at mouse position + setCursor(Cursor.getPredefinedCursor(Cursor.MOVE_CURSOR)); + } + else + { + // reset cursor + setCursor(Cursor.getPredefinedCursor(Cursor.DEFAULT_CURSOR)); } } }); addMouseListener(new MouseAdapter() { + @Override public void mousePressed(MouseEvent evt) { - if (!av.wrapAlignment) + if (SwingUtilities.isRightMouseButton(evt)) { - boxX = evt.getX(); - boxY = evt.getY(); - checkValid(); + if (!Platform.isAMac()) + { + showPopupMenu(evt); + } + } + else + { + if (!od.isPositionInBox(evt.getX(), evt.getY())) + { + // don't do anything if the mouse press is in the overview's box + // (wait to see if it's a drag instead) + // otherwise update the viewport + od.updateViewportFromMouse(evt.getX(), evt.getY(), + av.getAlignment().getHiddenSequences(), + av.getAlignment().getHiddenColumns()); + } + else + { + draggingBox = true; + od.setDragPoint(evt.getX(), evt.getY(), + av.getAlignment().getHiddenSequences(), + av.getAlignment().getHiddenColumns()); + } + } + } + + @Override + public void mouseReleased(MouseEvent evt) + { + if (draggingBox) + { + draggingBox = false; + } + } + + @Override + public void mouseClicked(MouseEvent evt) + { + if (SwingUtilities.isRightMouseButton(evt)) + { + showPopupMenu(evt); } } }); @@ -153,306 +211,128 @@ public class OverviewPanel extends JPanel implements Runnable updateOverviewImage(); } - /** - * DOCUMENT ME! + /* + * Displays the popup menu and acts on user input */ - void checkValid() + private void showPopupMenu(MouseEvent e) { - if (boxY < 0) - { - boxY = 0; - } - - if (boxY > (sequencesHeight - boxHeight)) - { - boxY = sequencesHeight - boxHeight + 1; - } - - if (boxX < 0) + JPopupMenu popup = new JPopupMenu(); + ActionListener menuListener = new ActionListener() { - boxX = 0; - } - - if (boxX > (width - boxWidth)) - { - if (av.hasHiddenColumns) + @Override + public void actionPerformed(ActionEvent event) { - // Try smallest possible box - boxWidth = (int) ((av.endRes - av.startRes + 1) * av.getCharWidth() * scalew); + // switch on/off the hidden columns view + toggleHiddenColumns(); + displayToggle.setSelected(showHidden); } - boxX = width - boxWidth; - } - - int col = (int) (boxX / scalew / av.getCharWidth()); - int row = (int) (boxY / scaleh / av.getCharHeight()); + }; + displayToggle = new JCheckBoxMenuItem( + MessageManager.getString("label.togglehidden")); + displayToggle.setEnabled(true); + displayToggle.setSelected(showHidden); + popup.add(displayToggle); + displayToggle.addActionListener(menuListener); + popup.show(this, e.getX(), e.getY()); + } - if (av.hasHiddenColumns) + /* + * Toggle overview display between showing hidden columns and hiding hidden columns + */ + private void toggleHiddenColumns() + { + if (showHidden) { - if (!av.getColumnSelection().isVisible(col)) - { - return; - } - - col = av.getColumnSelection().findColumnPosition(col); + showHidden = false; + od = new OverviewDimensionsHideHidden(av.getRanges(), + (av.isShowAnnotation() + && av.getAlignmentConservationAnnotation() != null)); } - - if (av.hasHiddenRows) + else { - row = av.alignment.getHiddenSequences().findIndexWithoutHiddenSeqs( - row); + showHidden = true; + od = new OverviewDimensionsShowHidden(av.getRanges(), + (av.isShowAnnotation() + && av.getAlignmentConservationAnnotation() != null)); } - - ap.setScrollValues(col, row); - + oviewCanvas.resetOviewDims(od); + updateOverviewImage(); + setBoxPosition(); } /** - * DOCUMENT ME! + * Updates the overview image when the related alignment panel is updated */ public void updateOverviewImage() { - if (resizing) + if (oviewCanvas == null) { - resizeAgain = true; + /* + * panel has been disposed + */ return; } - resizing = true; - if ((getWidth() > 0) && (getHeight() > 0)) { - width = getWidth(); - sequencesHeight = getHeight() - graphHeight; + od.setWidth(getWidth()); + od.setHeight(getHeight()); } - setPreferredSize(new Dimension(width, sequencesHeight + graphHeight)); + setPreferredSize(new Dimension(od.getWidth(), od.getHeight())); + + if (oviewCanvas.restartDraw()) + { + return; + } Thread thread = new Thread(this); thread.start(); repaint(); } - // This is set true if the user resizes whilst - // the overview is being calculated - boolean resizeAgain = false; - - /** - * DOCUMENT ME! - */ + @Override public void run() { - miniMe = null; - - if (av.showSequenceFeatures) - { - fr.transferSettings(ap.seqPanel.seqCanvas.getFeatureRenderer()); - } - - int alwidth = av.alignment.getWidth(); - int alheight = av.alignment.getHeight() - + av.alignment.getHiddenSequences().getSize(); - - setPreferredSize(new Dimension(width, sequencesHeight + graphHeight)); - - int fullsizeWidth = alwidth * av.getCharWidth(); - int fullsizeHeight = alheight * av.getCharHeight(); - - scalew = (float) width / (float) fullsizeWidth; - scaleh = (float) sequencesHeight / (float) fullsizeHeight; - - miniMe = new BufferedImage(width, sequencesHeight + graphHeight, - BufferedImage.TYPE_INT_RGB); - - Graphics mg = miniMe.getGraphics(); - mg.setColor(Color.orange); - mg.fillRect(0, 0, width, miniMe.getHeight()); - - float sampleCol = (float) alwidth / (float) width; - float sampleRow = (float) alheight / (float) sequencesHeight; - - int lastcol = -1, lastrow = -1; - int color = Color.white.getRGB(); - int row, col; - jalview.datamodel.SequenceI seq; - boolean hiddenRow = false; - for (row = 0; row < sequencesHeight; row++) - { - if ((int) (row * sampleRow) == lastrow) - { - // No need to recalculate the colours, - // Just copy from the row above - for (col = 0; col < width; col++) - { - miniMe.setRGB(col, row, miniMe.getRGB(col, row - 1)); - } - continue; - } - - lastrow = (int) (row * sampleRow); - - hiddenRow = false; - if (av.hasHiddenRows) - { - seq = av.alignment.getHiddenSequences().getHiddenSequence(lastrow); - if (seq == null) - { - int index = av.alignment.getHiddenSequences() - .findIndexWithoutHiddenSeqs(lastrow); - - seq = av.alignment.getSequenceAt(index); - } - else - { - hiddenRow = true; - } - } - else - { - seq = av.alignment.getSequenceAt(lastrow); - } - - if (seq == null) - { - System.out.println(lastrow + " null"); - continue; - } - - for (col = 0; col < width; col++) - { - if ((int) (col * sampleCol) == lastcol - && (int) (row * sampleRow) == lastrow) - { - miniMe.setRGB(col, row, color); - continue; - } - - lastcol = (int) (col * sampleCol); - - if (seq.getLength() > lastcol) - { - color = sr.getResidueBoxColour(seq, lastcol).getRGB(); - - if (av.showSequenceFeatures) - { - color = fr.findFeatureColour(color, seq, lastcol); - } - } - else - { - color = -1; // White - } - - if (hiddenRow - || (av.hasHiddenColumns && !av.getColumnSelection() - .isVisible(lastcol))) - { - color = new Color(color).darker().darker().getRGB(); - } - - miniMe.setRGB(col, row, color); - - } - } - - if (av.conservation != null) - { - for (col = 0; col < width; col++) - { - lastcol = (int) (col * sampleCol); - { - mg.translate(col, sequencesHeight); - ap.annotationPanel.drawGraph(mg, av.conservation, - (int) (sampleCol) + 1, graphHeight, - (int) (col * sampleCol), (int) (col * sampleCol) + 1); - mg.translate(-col, -sequencesHeight); - } - } - } - System.gc(); - - resizing = false; - + oviewCanvas.draw(av.isShowSequenceFeatures(), + (av.isShowAnnotation() + && av.getAlignmentConservationAnnotation() != null), + ap.getSeqPanel().seqCanvas.getFeatureRenderer()); setBoxPosition(); - - if (resizeAgain) - { - resizeAgain = false; - updateOverviewImage(); - } } /** - * DOCUMENT ME! + * Update the overview panel box when the associated alignment panel is + * changed + * */ - public void setBoxPosition() + private void setBoxPosition() { - int fullsizeWidth = av.alignment.getWidth() * av.getCharWidth(); - int fullsizeHeight = (av.alignment.getHeight() + av.alignment - .getHiddenSequences().getSize()) * av.getCharHeight(); - - int startRes = av.getStartRes(); - int endRes = av.getEndRes(); - - if (av.hasHiddenColumns) - { - startRes = av.getColumnSelection().adjustForHiddenColumns(startRes); - endRes = av.getColumnSelection().adjustForHiddenColumns(endRes); - } - - int startSeq = av.startSeq; - int endSeq = av.endSeq; - - if (av.hasHiddenRows) - { - startSeq = av.alignment.getHiddenSequences().adjustForHiddenSeqs( - startSeq); - - endSeq = av.alignment.getHiddenSequences() - .adjustForHiddenSeqs(endSeq); - - } - - scalew = (float) width / (float) fullsizeWidth; - scaleh = (float) sequencesHeight / (float) fullsizeHeight; - - boxX = (int) (startRes * av.getCharWidth() * scalew); - boxY = (int) (startSeq * av.getCharHeight() * scaleh); - - if (av.hasHiddenColumns) - { - boxWidth = (int) ((endRes - startRes + 1) * av.getCharWidth() * scalew); - } - else - { - boxWidth = (int) ((endRes - startRes + 1) * av.getCharWidth() * scalew); - } - - boxHeight = (int) ((endSeq - startSeq) * av.getCharHeight() * scaleh); - + od.setBoxPosition(av.getAlignment().getHiddenSequences(), + av.getAlignment().getHiddenColumns()); repaint(); } + @Override + public void propertyChange(PropertyChangeEvent evt) + { + setBoxPosition(); + } + /** - * DOCUMENT ME! - * - * @param g - * DOCUMENT ME! + * Removes this object as a property change listener, and nulls references */ - public void paintComponent(Graphics g) + protected void dispose() { - if (resizing) + try { - g.setColor(Color.white); - g.fillRect(0, 0, getWidth(), getHeight()); - } - else if (miniMe != null) + av.getRanges().removePropertyChangeListener(this); + } finally { - g.drawImage(miniMe, 0, 0, this); + av = null; + oviewCanvas = null; + ap = null; + od = null; } - - g.setColor(Color.red); - g.drawRect(boxX, boxY, boxWidth, boxHeight); - g.drawRect(boxX + 1, boxY + 1, boxWidth - 2, boxHeight - 2); - } }