/* * CharactersPainter.java * * Copyright (C) 2006-2014 Andrew Rambaut * * This program 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 2 * of the License, or (at your option) any later version. * * This program 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 this program; if not, write to the Free Software * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA. */ package figtree.treeviewer.painters; import jebl.evolution.alignments.Alignment; import jebl.evolution.graphs.Node; import jebl.evolution.sequences.AminoAcids; import jebl.evolution.sequences.Sequence; import jebl.evolution.trees.Tree; import figtree.treeviewer.painters.BasicLabelPainter; import java.awt.*; import java.awt.geom.Rectangle2D; /** * @author Andrew Rambaut * @version $Id$ * * $HeadURL$ * * $LastChangedBy$ * $LastChangedDate$ * $LastChangedRevision$ */ public class CharactersPainter extends BasicLabelPainter { public CharactersPainter(Alignment characters) { super(BasicLabelPainter.PainterIntent.TIP); this.characters = characters; paints[AminoAcids.A_STATE.getIndex()] = new Color(204, 255, 255); paints[AminoAcids.C_STATE.getIndex()] = new Color(0, 255, 255); paints[AminoAcids.D_STATE.getIndex()] = new Color(255, 204, 153); paints[AminoAcids.E_STATE.getIndex()] = new Color(255, 204, 0); paints[AminoAcids.F_STATE.getIndex()] = new Color(0, 204, 255); paints[AminoAcids.G_STATE.getIndex()] = new Color(0, 255, 0); paints[AminoAcids.H_STATE.getIndex()] = new Color(255, 255, 153); paints[AminoAcids.I_STATE.getIndex()] = new Color(0, 0, 128); paints[AminoAcids.K_STATE.getIndex()] = new Color(198, 66, 0); paints[AminoAcids.L_STATE.getIndex()] = new Color(51, 102, 255); paints[AminoAcids.M_STATE.getIndex()] = new Color(153, 204, 255); paints[AminoAcids.N_STATE.getIndex()] = new Color(255, 153, 0); paints[AminoAcids.P_STATE.getIndex()] = new Color(255, 255, 0); paints[AminoAcids.Q_STATE.getIndex()] = new Color(255, 102, 0); paints[AminoAcids.R_STATE.getIndex()] = new Color(230, 6, 6); paints[AminoAcids.S_STATE.getIndex()] = new Color(204, 255, 153); paints[AminoAcids.T_STATE.getIndex()] = new Color(0, 255, 153); paints[AminoAcids.V_STATE.getIndex()] = new Color(0, 0, 255); paints[AminoAcids.W_STATE.getIndex()] = new Color(204, 153, 255); paints[AminoAcids.Y_STATE.getIndex()] = new Color(204, 255, 204); paints[AminoAcids.B_STATE.getIndex()] = Color.DARK_GRAY; paints[AminoAcids.Z_STATE.getIndex()] = Color.DARK_GRAY; paints[AminoAcids.X_STATE.getIndex()] = Color.GRAY; paints[AminoAcids.UNKNOWN_STATE.getIndex()] = Color.GRAY; paints[AminoAcids.STOP_STATE.getIndex()] = Color.GRAY; paints[AminoAcids.GAP_STATE.getIndex()] = Color.GRAY; }; public Rectangle2D calibrate(Graphics2D g2, Node node) { super.calibrate(g2, node); int siteCount = characters.getSiteCount(); blockSize = super.getPreferredHeight() / 2; blockGap = blockSize / 2; width = (siteCount * (blockSize + blockGap)) + blockGap + blockGap; return new Rectangle2D.Double(0.0, 0.0, getPreferredWidth(), getPreferredHeight()); } public double getPreferredWidth() { return super.getPreferredWidth() + width; } public void paint(Graphics2D g2, Node node, Justification justification, Rectangle2D bounds) { Tree tree = getTree(); Stroke oldStroke = g2.getStroke(); Paint oldPaint = g2.getPaint(); float yOffset = (float)getPreferredHeight() / 2; switch (justification) { case FLUSH: case LEFT: { super.paint(g2, node, justification, new Rectangle2D.Double(bounds.getX() + width, bounds.getY(), bounds.getWidth() - width, bounds.getHeight())); Sequence sequence = characters.getSequence(tree.getTaxon(node)); if (sequence != null) { double x = blockGap; for (int state : sequence.getStateIndices()) { if (state != AminoAcids.GAP_STATE.getIndex()) { // Now create a bounds for the label Rectangle2D block = new Rectangle2D.Double( x, yOffset - (blockSize / 2.0), blockSize, blockSize); g2.setPaint(getStatePaint(state)); g2.fill(block); } x += blockSize + blockGap; } } } break; case RIGHT: { super.paint(g2, node, justification, new Rectangle2D.Double(bounds.getX(), bounds.getY(), bounds.getWidth() - width, bounds.getHeight())); Sequence sequence = characters.getSequence(tree.getTaxon(node)); if (sequence != null) { double x = getPreferredWidth() - blockGap - blockSize; for (int state : sequence.getStateIndices()) { // Now create a bounds for the label Rectangle2D block = new Rectangle2D.Double( x, yOffset - (blockSize / 2.0), blockSize, blockSize); g2.setPaint(getStatePaint(state)); g2.fill(block); x -= (blockSize + blockGap); } } } break; default: throw new IllegalArgumentException("Unrecognized alignment enum option"); } g2.setStroke(oldStroke); g2.setPaint(oldPaint); } private Alignment characters = null; private double width; private double blockSize = 6; private double blockGap = 2; Paint[] paints = new Paint[AminoAcids.getStateCount()]; public Paint getStatePaint(int stateIndex) { return paints[stateIndex]; } }