X-Git-Url: http://source.jalview.org/gitweb/?a=blobdiff_plain;f=src%2Fjalview%2Fappletgui%2FAlignmentPanel.java;h=365f8865390807ed2be027df0d42b42dfb197f0d;hb=3d2750d886f1b6015b2fc3f8671a6cf3e32bb07a;hp=e4b9b595442aebd4e812b23f39a24551cafbf647;hpb=2143ce8c4c9af885bc6d455655775d1ecb62acae;p=jalview.git diff --git a/src/jalview/appletgui/AlignmentPanel.java b/src/jalview/appletgui/AlignmentPanel.java old mode 100755 new mode 100644 index e4b9b59..365f886 --- a/src/jalview/appletgui/AlignmentPanel.java +++ b/src/jalview/appletgui/AlignmentPanel.java @@ -1,31 +1,53 @@ /* - * Jalview - A Sequence Alignment Editor and Viewer (Version 2.6) - * Copyright (C) 2010 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.appletgui; -import java.awt.*; -import java.awt.event.*; -import java.util.Hashtable; -import java.util.Vector; - +import jalview.analysis.AnnotationSorter; +import jalview.api.AlignViewportI; import jalview.api.AlignmentViewPanel; -import jalview.datamodel.*; - -public class AlignmentPanel extends Panel implements AdjustmentListener, AlignmentViewPanel +import jalview.bin.JalviewLite; +import jalview.datamodel.AlignmentI; +import jalview.datamodel.SearchResultsI; +import jalview.datamodel.SequenceI; +import jalview.structure.StructureSelectionManager; +import jalview.viewmodel.ViewportListenerI; +import jalview.viewmodel.ViewportRanges; + +import java.awt.BorderLayout; +import java.awt.Color; +import java.awt.Dimension; +import java.awt.FontMetrics; +import java.awt.Frame; +import java.awt.Graphics; +import java.awt.Panel; +import java.awt.Scrollbar; +import java.awt.event.AdjustmentEvent; +import java.awt.event.AdjustmentListener; +import java.awt.event.ComponentAdapter; +import java.awt.event.ComponentEvent; +import java.beans.PropertyChangeEvent; +import java.util.List; + +public class AlignmentPanel extends Panel + implements AdjustmentListener, AlignmentViewPanel, ViewportListenerI { public AlignViewport av; @@ -46,6 +68,8 @@ public class AlignmentPanel extends Panel implements AdjustmentListener, Alignme AnnotationLabels alabels; + ViewportRanges vpRanges; + // this value is set false when selection area being dragged boolean fastPaint = true; @@ -61,6 +85,7 @@ public class AlignmentPanel extends Panel implements AdjustmentListener, Alignme alignFrame = af; this.av = av; + vpRanges = av.getRanges(); seqPanel = new SeqPanel(av, this); idPanel = new IdPanel(av, this); scalePanel = new ScalePanel(av, this); @@ -69,10 +94,9 @@ public class AlignmentPanel extends Panel implements AdjustmentListener, Alignme annotationPanelHolder.add(annotationPanel, BorderLayout.CENTER); sequenceHolderPanel.add(annotationPanelHolder, BorderLayout.SOUTH); - alabels = new AnnotationLabels(this); - setAnnotationVisible(av.showAnnotation); + setAnnotationVisible(av.isShowAnnotation()); idPanelHolder.add(idPanel, BorderLayout.CENTER); idSpaceFillerPanel1.add(idwidthAdjuster, BorderLayout.CENTER); @@ -89,11 +113,37 @@ public class AlignmentPanel extends Panel implements AdjustmentListener, Alignme addComponentListener(new ComponentAdapter() { + @Override public void componentResized(ComponentEvent evt) { - setScrollValues(av.getStartRes(), av.getStartSeq()); + // reset the viewport ranges when the alignment panel is resized + // in particular, this initialises the end residue value when Jalview + // is initialised + if (av.getWrapAlignment()) + { + int widthInRes = seqPanel.seqCanvas + .getWrappedCanvasWidth(seqPanel.seqCanvas.getWidth()); + vpRanges.setViewportWidth(widthInRes); + } + else + { + int widthInRes = seqPanel.seqCanvas.getWidth() + / av.getCharWidth(); + int heightInSeq = seqPanel.seqCanvas.getHeight() + / av.getCharHeight(); + + vpRanges.setViewportWidth(widthInRes); + vpRanges.setViewportHeight(heightInSeq); + } + // setScrollValues(vpRanges.getStartRes(), vpRanges.getStartSeq()); + if (getSize().height > 0 + && annotationPanelHolder.getSize().height > 0) + { + validateAnnotationDimensions(false); + } repaint(); } + }); Dimension d = calculateIdWidth(); @@ -108,6 +158,7 @@ public class AlignmentPanel extends Panel implements AdjustmentListener, Alignme final AlignmentPanel ap = this; av.addPropertyChangeListener(new java.beans.PropertyChangeListener() { + @Override public void propertyChange(java.beans.PropertyChangeEvent evt) { if (evt.getPropertyName().equals("alignment")) @@ -117,7 +168,13 @@ public class AlignmentPanel extends Panel implements AdjustmentListener, Alignme } } }); + av.getRanges().addPropertyChangeListener(this); + } + @Override + public AlignViewportI getAlignViewport() + { + return av; } public SequenceRenderer getSequenceRenderer() @@ -125,11 +182,20 @@ public class AlignmentPanel extends Panel implements AdjustmentListener, Alignme return seqPanel.seqCanvas.sr; } - public FeatureRenderer getFeatureRenderer() + @Override + public jalview.api.FeatureRenderer getFeatureRenderer() { return seqPanel.seqCanvas.fr; } + @Override + public jalview.api.FeatureRenderer cloneFeatureRenderer() + { + FeatureRenderer nfr = new FeatureRenderer(av); + nfr.transferSettings(seqPanel.seqCanvas.fr); + return nfr; + } + public void alignmentChanged() { av.alignmentChanged(this); @@ -151,34 +217,32 @@ public class AlignmentPanel extends Panel implements AdjustmentListener, Alignme idPanel.idCanvas.image = null; FontMetrics fm = getFontMetrics(av.getFont()); - scalePanel.setSize(new Dimension(10, av.charHeight + fm.getDescent())); - idwidthAdjuster.setSize(new Dimension(10, av.charHeight - + fm.getDescent())); + scalePanel.setSize( + new Dimension(10, av.getCharHeight() + fm.getDescent())); + idwidthAdjuster.setSize( + new Dimension(10, av.getCharHeight() + fm.getDescent())); av.updateSequenceIdColours(); annotationPanel.image = null; int ap = annotationPanel.adjustPanelHeight(false); Dimension d = calculateIdWidth(); d.setSize(d.width + 4, seqPanel.seqCanvas.getSize().height); alabels.setSize(d.width + 4, ap); + idPanel.idCanvas.setSize(d); hscrollFillerPanel.setSize(d); - + validateAnnotationDimensions(false); annotationPanel.repaint(); validate(); repaint(); - - if (overviewPanel != null) - { - overviewPanel.updateOverviewImage(); - } } public void setIdWidth(int w, int h) { idPanel.idCanvas.setSize(w, h); idPanelHolder.setSize(w, idPanelHolder.getSize().height); - annotationSpaceFillerHolder.setSize(w,annotationSpaceFillerHolder.getSize().height); + annotationSpaceFillerHolder.setSize(w, + annotationSpaceFillerHolder.getSize().height); alabels.setSize(w, alabels.getSize().height); validate(); } @@ -234,7 +298,7 @@ public class AlignmentPanel extends Panel implements AdjustmentListener, Alignme * Highlight the given results on the alignment. * */ - public void highlightSearchResults(SearchResults results) + public void highlightSearchResults(SearchResultsI results) { scrollToPosition(results); seqPanel.seqCanvas.highlightSearchResults(results); @@ -247,7 +311,7 @@ public class AlignmentPanel extends Panel implements AdjustmentListener, Alignme * @param results * @return false if results were not found */ - public boolean scrollToPosition(SearchResults results) + public boolean scrollToPosition(SearchResultsI results) { return scrollToPosition(results, true); } @@ -261,35 +325,75 @@ public class AlignmentPanel extends Panel implements AdjustmentListener, Alignme * - when set, the overview will be recalculated (takes longer) * @return false if results were not found */ - public boolean scrollToPosition(SearchResults results, + public boolean scrollToPosition(SearchResultsI results, boolean redrawOverview) { + return scrollToPosition(results, 0, redrawOverview, false); + } + + /** + * scroll the view to show the position of the highlighted region in results + * (if any) + * + * @param results + * @param redrawOverview + * - when set, the overview will be recalculated (takes longer) + * @return false if results were not found + */ + public boolean scrollToPosition(SearchResultsI results, + int verticalOffset, boolean redrawOverview, boolean centre) + { // do we need to scroll the panel? - if (results != null && results.getSize() > 0) + if (results != null && results.getCount() > 0) { - int seqIndex = av.alignment.findIndex(results); + AlignmentI alignment = av.getAlignment(); + int seqIndex = alignment.findIndex(results); if (seqIndex == -1) { return false; } - SequenceI seq = av.alignment.getSequenceAt(seqIndex); - int[] r = results.getResults(seq, 0,av.alignment.getWidth()); + /* + * allow for offset of target sequence (actually scroll to one above it) + */ + + SequenceI seq = alignment.getSequenceAt(seqIndex); + int[] r = results.getResults(seq, 0, alignment.getWidth()); if (r == null) { - if (av.applet.debug) {// DEBUG - System.out.println("DEBUG: scroll didn't happen - results not within alignment : "+seq.getStart()+","+seq.getEnd()); + if (JalviewLite.debug) + {// DEBUG + System.out.println( + "DEBUG: scroll didn't happen - results not within alignment : " + + seq.getStart() + "," + seq.getEnd()); } return false; } - if (av.applet.debug) { + if (JalviewLite.debug) + { // DEBUG - System.out.println("DEBUG: scroll didn't happen: start=" + r[0] - + " av.getStartRes()=" + av.getStartRes() + " end=" + r[1] - + " seq.end=" + seq.getEnd() + " av.getEndRes()=" - + av.getEndRes() + " hextent=" + hextent); + /* + * System.out.println("DEBUG: scroll: start=" + r[0] + + * " av.getStartRes()=" + av.getStartRes() + " end=" + r[1] + + * " seq.end=" + seq.getEnd() + " av.getEndRes()=" + av.getEndRes() + + * " hextent=" + hextent); + */ } int start = r[0]; int end = r[1]; + + /* + * To centre results, scroll to positions half the visible width + * left/right of the start/end positions + */ + if (centre) + { + int offset = (vpRanges.getEndRes() - vpRanges.getStartRes() + 1) / 2 + - 1; + start = Math.max(start - offset, 0); + end = end + offset - 1; + // end = Math.min(end + offset, seq.getEnd() - 1); + } + if (start < 0) { return false; @@ -298,67 +402,119 @@ public class AlignmentPanel extends Panel implements AdjustmentListener, Alignme { return false; } - if (av.hasHiddenColumns) + + /* + * allow for offset of target sequence (actually scroll to one above it) + */ + seqIndex = Math.max(0, seqIndex - verticalOffset); + return scrollTo(start, end, seqIndex, false, redrawOverview); + } + return true; + } + + public boolean scrollTo(int ostart, int end, int seqIndex, + boolean scrollToNearest, boolean redrawOverview) + { + int startv, endv, starts, ends, width; + + int start = -1; + if (av.hasHiddenColumns()) + { + AlignmentI al = av.getAlignment(); + start = al.getHiddenColumns().absoluteToVisibleColumn(ostart); + end = al.getHiddenColumns().absoluteToVisibleColumn(end); + if (start == end) { - start = av.getColumnSelection().findColumnPosition(start); - end = av.getColumnSelection().findColumnPosition(end); - if (start == end) + if (!scrollToNearest && !al.getHiddenColumns().isVisible(ostart)) { - if (!av.colSel.isVisible(r[0])) - { - // don't scroll - position isn't visible - return false; - } + // don't scroll - position isn't visible + return false; } } - if (!av.wrapAlignment) + } + else + { + start = ostart; + } + + if (!av.getWrapAlignment()) + { + /* + * int spos=av.getStartRes(),sqpos=av.getStartSeq(); if ((startv = + * av.getStartRes()) >= start) { spos=start-1; // seqIn // + * setScrollValues(start - 1, seqIndex); } else if ((endv = + * av.getEndRes()) <= end) { // setScrollValues(spos=startv + 1 + end - + * endv, seqIndex); spos=startv + 1 + end - endv; } else if ((starts = + * av.getStartSeq()) > seqIndex) { setScrollValues(av.getStartRes(), + * seqIndex); } else if ((ends = av.getEndSeq()) <= seqIndex) { + * setScrollValues(av.getStartRes(), starts + seqIndex - ends + 1); } + */ + + // below is scrolling logic up to Jalview 2.8.2 + // if ((av.getStartRes() > end) + // || (av.getEndRes() < start) + // || ((av.getStartSeq() > seqIndex) || (av.getEndSeq() < seqIndex))) + // { + // if (start > av.getAlignment().getWidth() - hextent) + // { + // start = av.getAlignment().getWidth() - hextent; + // if (start < 0) + // { + // start = 0; + // } + // + // } + // if (seqIndex > av.getAlignment().getHeight() - vextent) + // { + // seqIndex = av.getAlignment().getHeight() - vextent; + // if (seqIndex < 0) + // { + // seqIndex = 0; + // } + // } + // setScrollValues(start, seqIndex); + // } + // logic copied from jalview.gui.AlignmentPanel: + if ((startv = vpRanges.getStartRes()) >= start) { - if ((av.getStartRes() > end) - || (av.getEndRes() < start) - || ((av.getStartSeq() > seqIndex) || (av.getEndSeq() < seqIndex))) - { - if (start > av.alignment.getWidth() - hextent) - { - start = av.alignment.getWidth() - hextent; - if (start < 0) - { - start = 0; - } - - } - if (seqIndex > av.alignment.getHeight() - vextent) - { - seqIndex = av.alignment.getHeight() - vextent; - if (seqIndex < 0) - { - seqIndex = 0; - } - } - setScrollValues(start, seqIndex); - } + /* + * Scroll left to make start of search results visible + */ + setScrollValues(start - 1, seqIndex); + } + else if ((endv = vpRanges.getEndRes()) <= end) + { + /* + * Scroll right to make end of search results visible + */ + setScrollValues(startv + 1 + end - endv, seqIndex); + } + else if ((starts = vpRanges.getStartSeq()) > seqIndex) + { + /* + * Scroll up to make start of search results visible + */ + setScrollValues(vpRanges.getStartRes(), seqIndex); } - else + else if ((ends = vpRanges.getEndSeq()) <= seqIndex) { - scrollToWrappedVisible(start); + /* + * Scroll down to make end of search results visible + */ + setScrollValues(vpRanges.getStartRes(), + starts + seqIndex - ends + 1); } + /* + * Else results are already visible - no need to scroll + */ } - if (redrawOverview && overviewPanel != null) + else { - overviewPanel.setBoxPosition(); + vpRanges.scrollToWrappedVisible(start); } - paintAlignment(redrawOverview); - return true; - } - void scrollToWrappedVisible(int res) - { - int cwidth = seqPanel.seqCanvas - .getWrappedCanvasWidth(seqPanel.seqCanvas.getSize().width); - if (res <= av.getStartRes() || res >= (av.getStartRes() + cwidth)) - { - vscroll.setValue(res / cwidth); - av.startRes = vscroll.getValue() * cwidth; - } + paintAlignment(redrawOverview, false); + return true; } public OverviewPanel getOverviewPanel() @@ -373,76 +529,105 @@ public class AlignmentPanel extends Panel implements AdjustmentListener, Alignme public void setAnnotationVisible(boolean b) { - if (!av.wrapAlignment) + if (!av.getWrapAlignment()) { annotationSpaceFillerHolder.setVisible(b); annotationPanelHolder.setVisible(b); } + else + { + annotationSpaceFillerHolder.setVisible(false); + annotationPanelHolder.setVisible(false); + } validate(); repaint(); } - + /** * automatically adjust annotation panel height for new annotation whilst * ensuring the alignment is still visible. */ + @Override public void adjustAnnotationHeight() { // TODO: display vertical annotation scrollbar if necessary // this is called after loading new annotation onto alignment if (alignFrame.getSize().height == 0) { - System.out.println("NEEDS FIXING"); + System.out.println( + "adjustAnnotationHeight frame size zero NEEDS FIXING"); } fontChanged(); validateAnnotationDimensions(true); apvscroll.addNotify(); hscroll.addNotify(); validate(); - addNotify(); - repaint(); + paintAlignment(true, false); } + /** - * calculate the annotation dimensions and refresh slider values accordingly. - * need to do repaints/notifys afterwards. + * Calculate the annotation dimensions and refresh slider values accordingly. + * Need to do repaints/notifys afterwards. */ - protected void validateAnnotationDimensions(boolean adjustPanelHeight) { - int height = annotationPanel.calcPanelHeight(); - if (hscroll.isVisible()) + protected void validateAnnotationDimensions(boolean adjustPanelHeight) + { + int rowHeight = av.getCharHeight(); + int alignmentHeight = rowHeight * av.getAlignment().getHeight(); + int annotationHeight = av.calcPanelHeight(); + + int mheight = annotationHeight; + Dimension d = sequenceHolderPanel.getSize(); + + int availableHeight = d.height - scalePanelHolder.getHeight(); + + if (adjustPanelHeight) { - height += hscroll.getPreferredSize().height; + /* + * If not enough vertical space, maximize annotation height while keeping + * at least two rows of alignment visible + */ + if (annotationHeight + alignmentHeight > availableHeight) + { + annotationHeight = Math.min(annotationHeight, + availableHeight - 2 * rowHeight); + } } - int mheight = height; - // sets initial preferred height - if ((height+40) > getSize().height / 2) + else { - height = getSize().height / 2; + // maintain same window layout whilst updating sliders + annotationHeight = annotationPanelHolder.getSize().height; } - if (!adjustPanelHeight) + + if (availableHeight - annotationHeight < 5) { - // maintain same window layout whilst updating sliders - height=seqPanelHolder.getSize().height; + annotationHeight = availableHeight; } - Dimension d=seqPanelHolder.getSize(),e=idPanel.getSize(); - annotationPanel.setSize(new Dimension(d.width,height)); - alabels.setSize(e.width,height); - annotationSpaceFillerHolder.setSize(new Dimension(e.width, height)); - annotationPanelHolder.setSize(new Dimension(d.width, height)); - seqPanelHolder.setSize(d.width,d.height-height); - int s=apvscroll.getValue(); - if (s>mheight-height) + + annotationPanel.setSize(new Dimension(d.width, annotationHeight)); + annotationPanelHolder.setSize(new Dimension(d.width, annotationHeight)); + // seqPanelHolder.setSize(d.width, seqandannot - height); + seqPanel.seqCanvas.setSize(d.width, + seqPanel.seqCanvas.getSize().height); + + Dimension e = idPanel.getSize(); + alabels.setSize(new Dimension(e.width, annotationHeight)); + annotationSpaceFillerHolder + .setSize(new Dimension(e.width, annotationHeight)); + + int s = apvscroll.getValue(); + if (s > mheight - annotationHeight) { s = 0; } - apvscroll.setValues(apvscroll.getValue(), height, 0, mheight); - annotationPanel.setScrollOffset(apvscroll.getValue()); - alabels.setScrollOffset(apvscroll.getValue()); + apvscroll.setValues(s, annotationHeight, 0, mheight); + annotationPanel.setScrollOffset(apvscroll.getValue(), false); + alabels.setScrollOffset(apvscroll.getValue(), false); } - + public void setWrapAlignment(boolean wrap) { - av.startSeq = 0; - av.startRes = 0; + vpRanges.setStartEndSeq(0, vpRanges.getVisibleAlignmentHeight()); + vpRanges.setStartRes(0); scalePanelHolder.setVisible(!wrap); hscroll.setVisible(!wrap); @@ -453,7 +638,7 @@ public class AlignmentPanel extends Panel implements AdjustmentListener, Alignme annotationPanelHolder.setVisible(false); annotationSpaceFillerHolder.setVisible(false); } - else if (av.showAnnotation) + else if (av.isShowAnnotation()) { annotationPanelHolder.setVisible(true); annotationSpaceFillerHolder.setVisible(true); @@ -464,6 +649,7 @@ public class AlignmentPanel extends Panel implements AdjustmentListener, Alignme fontChanged(); // This is so that the scalePanel is resized correctly validate(); + sequenceHolderPanel.validate(); repaint(); } @@ -472,204 +658,273 @@ public class AlignmentPanel extends Panel implements AdjustmentListener, Alignme int vextent = 0; - // return value is true if the scroll is valid - public boolean scrollUp(boolean up) + public void setScrollValues(int xpos, int ypos) { - if (up) + int x = xpos; + int y = ypos; + + if (av.getWrapAlignment()) { - if (vscroll.getValue() < 1) - { - return false; - } - setScrollValues(hscroll.getValue(), vscroll.getValue() - 1); + setScrollingForWrappedPanel(x); } else { - if (vextent + vscroll.getValue() >= av.getAlignment().getHeight()) + int width = av.getAlignment().getVisibleWidth(); + int height = av.getAlignment().getHeight(); + + if (x < 0) { - return false; + x = 0; } - setScrollValues(hscroll.getValue(), vscroll.getValue() + 1); - } - repaint(); - return true; - } + hextent = seqPanel.seqCanvas.getSize().width / av.getCharWidth(); + vextent = seqPanel.seqCanvas.getSize().height / av.getCharHeight(); - public boolean scrollRight(boolean right) - { - if (!right) - { - if (hscroll.getValue() < 1) + if (hextent > width) { - return false; + hextent = width; } - setScrollValues(hscroll.getValue() - 1, vscroll.getValue()); - } - else - { - if (hextent + hscroll.getValue() >= av.getAlignment().getWidth()) + + if (vextent > height) { - return false; + vextent = height; } - setScrollValues(hscroll.getValue() + 1, vscroll.getValue()); - } - - repaint(); - return true; - } - public void setScrollValues(int x, int y) - { - int width = av.alignment.getWidth(); - int height = av.alignment.getHeight(); + if ((hextent + x) > width) + { + System.err.println("hextent was " + hextent + " and x was " + x); - if (av.hasHiddenColumns) - { - width = av.getColumnSelection().findColumnPosition(width); - } + x = width - hextent; + } - av.setStartRes(x); - av.setEndRes((x + (seqPanel.seqCanvas.getSize().width / av.charWidth)) - 1); + if ((vextent + y) > height) + { + y = height - vextent; + } - hextent = seqPanel.seqCanvas.getSize().width / av.charWidth; - vextent = seqPanel.seqCanvas.getSize().height / av.charHeight; - - if (hextent > width) - { - hextent = width; - } + if (y < 0) + { + y = 0; + } - if (vextent > height) - { - vextent = height; - } + if (x < 0) + { + System.err.println("x was " + x); + x = 0; + } - if ((hextent + x) > width) - { - x = width - hextent; - } + hscroll.setValues(x, hextent, 0, width); + vscroll.setValues(y, vextent, 0, height); - if ((vextent + y) > height) - { - y = height - vextent; - } + // AWT scrollbar does not fire adjustmentValueChanged for setValues + // so also call adjustment code! + adjustHorizontal(x); + adjustVertical(y); - if (y < 0) - { - y = 0; + sendViewPosition(); } + } - if (x < 0) + /** + * Respond to adjustment event when horizontal or vertical scrollbar is + * changed + * + * @param evt + * adjustment event encoding whether apvscroll, hscroll or vscroll + * changed + */ + @Override + public void adjustmentValueChanged(AdjustmentEvent evt) + { + // Note that this event is NOT fired by the AWT scrollbar when setValues is + // called. Instead manually call adjustHorizontal and adjustVertical + // directly. + if (evt == null || evt.getSource() == apvscroll) { - x = 0; + annotationPanel.setScrollOffset(apvscroll.getValue(), false); + alabels.setScrollOffset(apvscroll.getValue(), false); } - - av.setStartSeq(y); - - int endSeq = y + vextent; - if (endSeq > av.alignment.getHeight()) + if (evt == null || evt.getSource() == hscroll) { - endSeq = av.alignment.getHeight(); + int x = hscroll.getValue(); + adjustHorizontal(x); } - av.setEndSeq(endSeq); - hscroll.setValues(x, hextent, 0, width); - vscroll.setValues(y, vextent, 0, height); - - if (overviewPanel != null) + if (evt == null || evt.getSource() == vscroll) { - overviewPanel.setBoxPosition(); + int offy = vscroll.getValue(); + adjustVertical(offy); } } - public void adjustmentValueChanged(AdjustmentEvent evt) + private void adjustHorizontal(int x) { - int oldX = av.getStartRes(); - int oldY = av.getStartSeq(); - - if (evt == null || evt.getSource() == apvscroll) + int oldX = vpRanges.getStartRes(); + int oldwidth = vpRanges.getViewportWidth(); + int width = seqPanel.seqCanvas.getWidth() / av.getCharWidth(); + + // if we're scrolling to the position we're already at, stop + // this prevents infinite recursion of events when the scroll/viewport + // ranges values are the same + if ((x == oldX) && (width == oldwidth)) { - annotationPanel.setScrollOffset(apvscroll.getValue()); - alabels.setScrollOffset(apvscroll.getValue()); - // annotationPanel.image=null; - // alabels.image=null; - // alabels.repaint(); - // annotationPanel.repaint(); + return; } - if (evt == null || evt.getSource() == hscroll) + vpRanges.setViewportStartAndWidth(x, width); + + if (av.getWrapAlignment() || !fastPaint) { - int x = hscroll.getValue(); - av.setStartRes(x); - av.setEndRes(x + seqPanel.seqCanvas.getSize().width - / av.getCharWidth() - 1); + repaint(); } + sendViewPosition(); + } - if (evt == null || evt.getSource() == vscroll) + private void adjustVertical(int newY) + { + if (av.getWrapAlignment()) { - int offy = vscroll.getValue(); - if (av.getWrapAlignment()) + /* + * if we're scrolling to the position we're already at, stop + * this prevents infinite recursion of events when the scroll/viewport + * ranges values are the same + */ + int oldX = vpRanges.getStartRes(); + int oldY = vpRanges.getWrappedScrollPosition(oldX); + if (oldY == newY) { - int rowSize = seqPanel.seqCanvas - .getWrappedCanvasWidth(seqPanel.seqCanvas.getSize().width); - av.setStartRes(vscroll.getValue() * rowSize); - av.setEndRes((vscroll.getValue() + 1) * rowSize); + return; } - else + if (newY > -1) { - av.setStartSeq(offy); - av.setEndSeq(offy + seqPanel.seqCanvas.getSize().height - / av.getCharHeight()); + /* + * limit page up/down to one width's worth of positions + */ + int rowSize = vpRanges.getViewportWidth(); + int newX = newY > oldY ? oldX + rowSize : oldX - rowSize; + vpRanges.setViewportStartAndWidth(Math.max(0, newX), rowSize); } } - - if (overviewPanel != null) + else { - overviewPanel.setBoxPosition(); + int height = seqPanel.seqCanvas.getHeight() / av.getCharHeight(); + int oldY = vpRanges.getStartSeq(); + int oldheight = vpRanges.getViewportHeight(); + + // if we're scrolling to the position we're already at, stop + // this prevents infinite recursion of events when the scroll/viewport + // ranges values are the same + if ((newY == oldY) && (height == oldheight)) + { + return; + } + vpRanges.setViewportStartAndHeight(newY, height); } - - int scrollX = av.startRes - oldX; - int scrollY = av.startSeq - oldY; - - if (av.getWrapAlignment() || !fastPaint || av.MAC) + if (av.getWrapAlignment() || !fastPaint) { repaint(); } - else + sendViewPosition(); + } + + /** + * A helper method to return the AlignmentPanel in the other (complementary) + * half of a SplitFrame view. Returns null if not in a SplitFrame. + * + * @return + */ + private AlignmentPanel getComplementPanel() + { + AlignmentPanel ap = null; + if (alignFrame != null) { - // Make sure we're not trying to draw a panel - // larger than the visible window - if (scrollX > av.endRes - av.startRes) + SplitFrame sf = alignFrame.getSplitFrame(); + if (sf != null) { - scrollX = av.endRes - av.startRes; - } - else if (scrollX < av.startRes - av.endRes) - { - scrollX = av.startRes - av.endRes; + AlignFrame other = sf.getComplement(alignFrame); + if (other != null) + { + ap = other.alignPanel; + } } + } + return ap; + } - idPanel.idCanvas.fastPaint(scrollY); - seqPanel.seqCanvas.fastPaint(scrollX, scrollY); - - scalePanel.repaint(); - if (av.getShowAnnotation()) + /** + * Follow a scrolling change in the (cDNA/Protein) complementary alignment. + * The aim is to keep the two alignments 'lined up' on their centre columns. + * + * @param sr + * holds mapped region(s) of this alignment that we are scrolling + * 'to'; may be modified for sequence offset by this method + * @param seqOffset + * the number of visible sequences to show above the mapped region + */ + protected void scrollToCentre(SearchResultsI sr, int seqOffset) + { + /* + * To avoid jumpy vertical scrolling (if some sequences are gapped or not + * mapped), we can make the scroll-to location a sequence above the one + * actually mapped. + */ + SequenceI mappedTo = sr.getResults().get(0).getSequence(); + List seqs = av.getAlignment().getSequences(); + + /* + * This is like AlignmentI.findIndex(seq) but here we are matching the + * dataset sequence not the aligned sequence + */ + boolean matched = false; + for (SequenceI seq : seqs) + { + if (mappedTo == seq.getDatasetSequence()) { - annotationPanel.fastPaint(av.getStartRes() - oldX); + matched = true; + break; } } + if (!matched) + { + return; // failsafe, shouldn't happen + } + /* + * Scroll to position but centring the target residue. Also set a state flag + * to prevent adjustmentValueChanged performing this recursively. + */ + scrollToPosition(sr, seqOffset, true, true); } - public void paintAlignment(boolean updateOverview) + private void sendViewPosition() { + StructureSelectionManager.getStructureSelectionManager(av.applet) + .sendViewPosition(this, vpRanges.getStartRes(), + vpRanges.getEndRes(), vpRanges.getStartSeq(), + vpRanges.getEndSeq()); + } + + /** + * Repaint the alignment and annotations, and, optionally, any overview window + */ + @Override + public void paintAlignment(boolean updateOverview, + boolean updateStructures) + { + final AnnotationSorter sorter = new AnnotationSorter(getAlignment(), + av.isShowAutocalculatedAbove()); + sorter.sort(getAlignment().getAlignmentAnnotation(), + av.getSortAnnotationsBy()); repaint(); - if (updateOverview) + if (updateStructures) { jalview.structure.StructureSelectionManager - .getStructureSelectionManager().sequenceColoursChanged(this); - + .getStructureSelectionManager(av.applet) + .sequenceColoursChanged(this); + } + if (updateOverview) + { if (overviewPanel != null) { overviewPanel.updateOverviewImage(); @@ -677,48 +932,56 @@ public class AlignmentPanel extends Panel implements AdjustmentListener, Alignme } } + @Override public void update(Graphics g) { paint(g); } + @Override public void paint(Graphics g) { invalidate(); Dimension d = idPanel.idCanvas.getSize(); - idPanel.idCanvas.setSize(d.width, seqPanel.seqCanvas.getSize().height); - - if (av.getWrapAlignment()) + final int canvasHeight = seqPanel.seqCanvas.getSize().height; + if (canvasHeight != d.height) { - int maxwidth = av.alignment.getWidth(); - - if (av.hasHiddenColumns) - { - maxwidth = av.getColumnSelection().findColumnPosition(maxwidth) - 1; - } + idPanel.idCanvas.setSize(d.width, canvasHeight); + } - int canvasWidth = seqPanel.seqCanvas - .getWrappedCanvasWidth(seqPanel.seqCanvas.getSize().width); + setScrollValues(vpRanges.getStartRes(), vpRanges.getStartSeq()); - if (canvasWidth > 0) + seqPanel.seqCanvas.repaint(); + idPanel.idCanvas.repaint(); + if (!av.getWrapAlignment()) + { + if (av.isShowAnnotation()) { - int max = maxwidth / canvasWidth; - vscroll.setMaximum(1 + max); - vscroll.setUnitIncrement(1); - vscroll.setVisibleAmount(1); + alabels.repaint(); + annotationPanel.repaint(); } - } - else - { - setScrollValues(av.getStartRes(), av.getStartSeq()); + scalePanel.repaint(); } - alabels.repaint(); + } - seqPanel.seqCanvas.repaint(); - scalePanel.repaint(); - annotationPanel.repaint(); - idPanel.idCanvas.repaint(); + /** + * Set vertical scroll bar parameters for wrapped panel + * + * @param topLeftColumn + * the column position at top left (0..) + */ + private void setScrollingForWrappedPanel(int topLeftColumn) + { + int scrollPosition = vpRanges.getWrappedScrollPosition(topLeftColumn); + int maxScroll = vpRanges.getWrappedMaxScroll(topLeftColumn); + + /* + * a scrollbar's value can be set to at most (maximum-extent) + * so we add extent (1) to the maxScroll value + */ + vscroll.setUnitIncrement(1); + vscroll.setValues(scrollPosition, 1, 0, maxScroll + 1); } protected Panel sequenceHolderPanel = new Panel(); @@ -729,72 +992,58 @@ public class AlignmentPanel extends Panel implements AdjustmentListener, Alignme protected Panel seqPanelHolder = new Panel(); - BorderLayout borderLayout1 = new BorderLayout(); - - BorderLayout borderLayout3 = new BorderLayout(); - protected Panel scalePanelHolder = new Panel(); protected Panel idPanelHolder = new Panel(); - BorderLayout borderLayout5 = new BorderLayout(); - protected Panel idSpaceFillerPanel1 = new Panel(); public Panel annotationSpaceFillerHolder = new Panel(); - BorderLayout borderLayout6 = new BorderLayout(); - - BorderLayout borderLayout7 = new BorderLayout(); - - Panel hscrollHolder = new Panel(); - - BorderLayout borderLayout10 = new BorderLayout(); - protected Panel hscrollFillerPanel = new Panel(); - BorderLayout borderLayout11 = new BorderLayout(); - - BorderLayout borderLayout4 = new BorderLayout(); - - BorderLayout borderLayout2 = new BorderLayout(); - Panel annotationPanelHolder = new Panel(); protected Scrollbar apvscroll = new Scrollbar(); - BorderLayout borderLayout12 = new BorderLayout(); + /* + * Flag set while scrolling to follow complementary cDNA/protein scroll. When + * false, suppresses invoking the same method recursively. + */ + private boolean scrollComplementaryPanel = true; private void jbInit() throws Exception { // idPanelHolder.setPreferredSize(new Dimension(70, 10)); - this.setLayout(borderLayout7); + this.setLayout(new BorderLayout()); - //sequenceHolderPanel.setPreferredSize(new Dimension(150, 150)); - sequenceHolderPanel.setLayout(borderLayout3); - seqPanelHolder.setLayout(borderLayout1); + // sequenceHolderPanel.setPreferredSize(new Dimension(150, 150)); + sequenceHolderPanel.setLayout(new BorderLayout()); + seqPanelHolder.setLayout(new BorderLayout()); scalePanelHolder.setBackground(Color.white); // scalePanelHolder.setPreferredSize(new Dimension(10, 30)); - scalePanelHolder.setLayout(borderLayout6); - idPanelHolder.setLayout(borderLayout5); + scalePanelHolder.setLayout(new BorderLayout()); + idPanelHolder.setLayout(new BorderLayout()); idSpaceFillerPanel1.setBackground(Color.white); // idSpaceFillerPanel1.setPreferredSize(new Dimension(10, 30)); - idSpaceFillerPanel1.setLayout(borderLayout11); + idSpaceFillerPanel1.setLayout(new BorderLayout()); annotationSpaceFillerHolder.setBackground(Color.white); // annotationSpaceFillerHolder.setPreferredSize(new Dimension(10, 80)); - annotationSpaceFillerHolder.setLayout(borderLayout4); + annotationSpaceFillerHolder.setLayout(new BorderLayout()); hscroll.setOrientation(Scrollbar.HORIZONTAL); - hscrollHolder.setLayout(borderLayout10); + + Panel hscrollHolder = new Panel(); + hscrollHolder.setLayout(new BorderLayout()); hscrollFillerPanel.setBackground(Color.white); apvscroll.setOrientation(Scrollbar.VERTICAL); apvscroll.setVisible(true); apvscroll.addAdjustmentListener(this); annotationPanelHolder.setBackground(Color.white); - annotationPanelHolder.setLayout(borderLayout12); + annotationPanelHolder.setLayout(new BorderLayout()); annotationPanelHolder.add(apvscroll, BorderLayout.EAST); // hscrollFillerPanel.setPreferredSize(new Dimension(70, 10)); hscrollHolder.setBackground(Color.white); @@ -827,74 +1076,85 @@ public class AlignmentPanel extends Panel implements AdjustmentListener, Alignme public void updateAnnotation(boolean applyGlobalSettings) { - // TODO: this should be merged with other annotation update stuff - that - // sits on AlignViewport - boolean updateCalcs = false; - boolean conv = av.isShowGroupConservation(); - boolean cons = av.isShowGroupConsensus(); - boolean showprf = av.isShowSequenceLogo(); - boolean showConsHist = av.isShowConsensusHistogram(); + updateAnnotation(applyGlobalSettings, false); + } - boolean sortg = true; + public void updateAnnotation(boolean applyGlobalSettings, + boolean preserveNewGroupSettings) + { + av.updateGroupAnnotationSettings(applyGlobalSettings, + preserveNewGroupSettings); + adjustAnnotationHeight(); + } - // remove old automatic annotation - // add any new annotation + @Override + public AlignmentI getAlignment() + { + return av.getAlignment(); + } - Vector gr = av.alignment.getGroups(); // OrderedBy(av.alignment.getSequencesArray()); - // intersect alignment annotation with alignment groups + @Override + public String getViewName() + { + return getName(); + } - AlignmentAnnotation[] aan = av.alignment.getAlignmentAnnotation(); - Hashtable oldrfs = new Hashtable(); - if (aan != null) - { - for (int an = 0; an < aan.length; an++) - { - if (aan[an].autoCalculated && aan[an].groupRef != null) - { - oldrfs.put(aan[an].groupRef, aan[an].groupRef); - av.alignment.deleteAnnotation(aan[an]); - aan[an] = null; - } - } - } - SequenceGroup sg; - if (gr != null) - { - for (int g = 0; g < gr.size(); g++) - { - updateCalcs = false; - sg = (SequenceGroup) gr.elementAt(g); - if (applyGlobalSettings || !oldrfs.containsKey(sg)) - { - // set defaults for this group's conservation/consensus - sg.setshowSequenceLogo(showprf); - sg.setShowConsensusHistogram(showConsHist); - } - if (conv) - { - updateCalcs = true; - av.alignment.addAnnotation(sg.getConservationRow(), 0); - } - if (cons) - { - updateCalcs = true; - av.alignment.addAnnotation(sg.getConsensus(), 0); - } - // refresh the annotation rows - if (updateCalcs) - { - sg.recalcConservation(); - } - } - } - oldrfs.clear(); - adjustAnnotationHeight(); + @Override + public StructureSelectionManager getStructureSelectionManager() + { + return StructureSelectionManager + .getStructureSelectionManager(av.applet); } @Override - public AlignmentI getAlignment() + public void raiseOOMWarning(String string, OutOfMemoryError error) { - return av.alignment; + // TODO: JAL-960 + System.err.println("Out of memory whilst '" + string + "'"); + error.printStackTrace(); + } + + /** + * Set a flag to say we are scrolling to follow a (cDNA/protein) complement. + * + * @param b + */ + protected void setToScrollComplementPanel(boolean b) + { + this.scrollComplementaryPanel = b; + } + + /** + * Get whether to scroll complement panel + * + * @return true if cDNA/protein complement panels should be scrolled + */ + protected boolean isSetToScrollComplementPanel() + { + return this.scrollComplementaryPanel; + } + + @Override + /** + * Property change event fired when a change is made to the viewport ranges + * object associated with this alignment panel's viewport + */ + public void propertyChange(PropertyChangeEvent evt) + { + // update this panel's scroll values based on the new viewport ranges values + int x = vpRanges.getStartRes(); + int y = vpRanges.getStartSeq(); + setScrollValues(x, y); + + // now update any complementary alignment (its viewport ranges object + // is different so does not get automatically updated) + if (isSetToScrollComplementPanel()) + { + setToScrollComplementPanel(false); + av.scrollComplementaryAlignment(getComplementPanel()); + setToScrollComplementPanel(true); + } + } }