Mercurial > dive4elements > river
view flys-artifacts/src/main/java/de/intevation/flys/jfree/ShapeRenderer.java @ 4241:49cb65d5932d
Improved the historical discharge calculation.
The calculation now creates new HistoricalWQKms (new subclass of WQKms). Those WQKms are used
to create new facets from (new) type 'HistoricalDischargeCurveFacet'. The chart generator is
improved to support those facets.
author | Ingo Weinzierl <ingo.weinzierl@intevation.de> |
---|---|
date | Wed, 24 Oct 2012 14:34:35 +0200 |
parents | 0cf647fe2a96 |
children |
line wrap: on
line source
package de.intevation.flys.jfree; /** * Copyright (c) 2006, 2012 by Intevation GmbH * * @author Sascha L. Teichmann (teichmann@intevation.de) * * This program is free software under the LGPL (>=v2.1) * Read the file LGPL coming with FLYS for details. */ import java.awt.Font; import java.awt.Graphics2D; import java.awt.Paint; import java.awt.Shape; import java.awt.geom.Point2D; import java.awt.geom.Rectangle2D; import java.util.ArrayList; import java.util.List; import java.util.Map; import org.jfree.chart.axis.ValueAxis; import org.jfree.chart.labels.ItemLabelPosition; import org.jfree.chart.labels.XYItemLabelGenerator; import org.jfree.chart.plot.CrosshairState; import org.jfree.chart.plot.PlotOrientation; import org.jfree.chart.plot.PlotRenderingInfo; import org.jfree.chart.plot.XYPlot; import org.jfree.chart.renderer.xy.StandardXYItemRenderer; import org.jfree.chart.renderer.xy.XYItemRendererState; import org.jfree.data.xy.XYDataset; import org.jfree.text.TextUtilities; import org.jfree.ui.RectangleEdge; public class ShapeRenderer extends StandardXYItemRenderer { public static class Entry { protected Shape shape; protected Shape frame; protected Paint paint; protected boolean filled; public Entry( Shape shape, Paint paint, boolean filled ) { this.shape = shape; this.paint = paint; this.filled = filled; } public Entry( Shape shape, Shape frame, Paint paint, boolean filled ) { this.shape = shape; this.frame = frame; this.paint = paint; this.filled = filled; } public Shape getShape() { return shape; } public void setShape(Shape shape) { this.shape = shape; } public Paint getPaint() { return paint; } public void setPaint(Paint paint) { this.paint = paint; } public boolean getFilled() { return filled; } public void setFilled(boolean filled) { this.filled = filled; } public boolean equals(Object other) { Entry entry = (Entry)other; return filled == entry.filled && paint.equals(entry.paint) && shape.equals(entry.shape); } public int hashCode() { return shape.hashCode() ^ paint.hashCode() ^ (filled ? 1231 : 1237); } } // class Entry public interface LabelGenerator { String createLabel(Entry entry); } // interface EntryLabelGenerator protected Entry [] entries; protected List<Rectangle2D> labelBoundingBoxes; protected Rectangle2D area; public ShapeRenderer() { this(SHAPES); } public ShapeRenderer(int type) { super(type); } public ShapeRenderer(Map<Entry, Integer> map) { super(SHAPES); setEntries(map); } public void setEntries(Entry [] entries) { this.entries = entries; } public void setEntries(Map<Entry, Integer> map) { Entry [] entries = new Entry[map.size()]; for (Map.Entry<Entry, Integer> entry: map.entrySet()) { entries[entry.getValue()] = entry.getKey(); } setEntries(entries); } @Override public Shape getSeriesShape(int series) { return entries[series].shape; } public Shape getSeriesFrame(int series) { return entries[series].frame; } @Override public Paint getSeriesPaint(int series) { return entries[series].paint; } @Override public boolean getItemShapeFilled(int series, int item) { return entries[series].filled; } @Override public XYItemRendererState initialise( Graphics2D g2, Rectangle2D dataArea, XYPlot plot, XYDataset data, PlotRenderingInfo info ) { if (labelBoundingBoxes == null) { labelBoundingBoxes = new ArrayList<Rectangle2D>(32); } else { labelBoundingBoxes.clear(); } area = dataArea; return super.initialise(g2, dataArea, plot, data, info); } @Override public void drawItem( Graphics2D g2, XYItemRendererState state, Rectangle2D dataArea, PlotRenderingInfo info, XYPlot plot, ValueAxis domainAxis, ValueAxis rangeAxis, XYDataset dataset, int series, int item, CrosshairState crosshairState, int pass ) { if (!getItemVisible(series, item)) { return; } // get the data point... double x1 = dataset.getXValue(series, item); double y1 = dataset.getYValue(series, item); if (Double.isNaN(x1) || Double.isNaN(y1)) { return; } RectangleEdge xAxisLocation = plot.getDomainAxisEdge(); RectangleEdge yAxisLocation = plot.getRangeAxisEdge(); double x = domainAxis.valueToJava2D(x1, dataArea, xAxisLocation); double y = rangeAxis.valueToJava2D(y1, dataArea, yAxisLocation); if (dataArea.contains(x, y)) super.drawItem( g2, state, dataArea, info, plot, domainAxis, rangeAxis, dataset, series, item, crosshairState, pass); } protected Point2D shiftBox(Rectangle2D box) { double cx1 = area.getX(); double cy1 = area.getY(); double cx2 = cx1 + area.getWidth(); double cy2 = cy1 + area.getHeight(); double bx1 = box.getX(); double by1 = box.getY(); double bx2 = bx1 + box.getWidth(); double by2 = by1 + box.getHeight(); double dx; double dy; if (bx1 < cx1) { dx = cx1 - bx1; } else if (bx2 > cx2) { dx = cx2 - bx2; } else { dx = 0d; } if (by1 < cy1) { dy = cy1 - by1; } else if (by2 > cy2) { dy = cy2 - by2; } else { dy = 0d; } return new Point2D.Double(dx, dy); } @Override protected void drawItemLabel( Graphics2D g2, PlotOrientation orientation, XYDataset dataset, int series, int item, double x, double y, boolean negative ) { XYItemLabelGenerator generator = getItemLabelGenerator(series, item); if (generator == null) { return; } Font labelFont = getItemLabelFont(series, item); Paint paint = getItemLabelPaint(series, item); g2.setFont(labelFont); g2.setPaint(paint); String label = generator.generateLabel(dataset, series, item); ATTEMPS: for (int attempt = 0; attempt < 2; ++attempt) { // get the label position.. ItemLabelPosition position = null; boolean pos; switch (attempt) { case 0: pos = negative; break; case 1: pos = !negative; break; default: break ATTEMPS; } if (pos) { position = getNegativeItemLabelPosition(series, item); } else { position = getPositiveItemLabelPosition(series, item); } // work out the label anchor point... Point2D anchorPoint = calculateLabelAnchorPoint( position.getItemLabelAnchor(), x, y, orientation); Shape labelShape = TextUtilities.calculateRotatedStringBounds( label, g2, (float)anchorPoint.getX(), (float)anchorPoint.getY(), position.getTextAnchor(), position.getAngle(), position.getRotationAnchor()); Rectangle2D bbox = labelShape.getBounds2D(); Point2D shift = shiftBox(bbox); bbox = new Rectangle2D.Double( bbox.getX() + shift.getX(), bbox.getY() + shift.getY(), bbox.getWidth(), bbox.getHeight()); if (labelBoundingBoxes != null) { for (Rectangle2D old: labelBoundingBoxes) { if (old.intersects(bbox)) { continue ATTEMPS; } } labelBoundingBoxes.add(bbox); } TextUtilities.drawRotatedString( label, g2, (float)(anchorPoint.getX() + shift.getX()), (float)(anchorPoint.getY() + shift.getY()), position.getTextAnchor(), position.getAngle(), position.getRotationAnchor()); break; } } } // vim:set ts=4 sw=4 si et sta sts=4 fenc=utf8 :