sascha@422: package de.intevation.gnv.jfreechart; sascha@422: sascha@779: import java.awt.BasicStroke; sascha@422: import java.awt.Color; sascha@779: import java.awt.Font; sascha@779: import java.awt.FontMetrics; sascha@422: import java.awt.Graphics2D; sascha@422: import java.awt.Paint; sascha@422: import java.awt.Shape; sascha@447: sascha@422: import java.awt.geom.GeneralPath; sascha@779: sascha@779: import java.awt.geom.Rectangle2D.Double; sascha@779: sascha@422: import java.awt.geom.Rectangle2D; sascha@779: sascha@779: import java.util.ArrayList; sascha@779: sascha@779: import org.apache.log4j.Logger; sascha@779: sascha@779: import org.jfree.chart.axis.ValueAxis; sascha@422: sascha@422: import org.jfree.data.Range; sascha@422: sascha@779: import org.jfree.text.TextUtilities; sascha@450: sascha@450: import org.jfree.ui.RectangleEdge; sascha@450: sascha@422: /** ingo@795: * This renderer is used to draw polygons into a Graphics object. ingo@795: * sascha@780: * @author Ingo Weinzierl sascha@780: * @author Sascha L. Teichmann sascha@422: */ sascha@422: public class PolygonRenderer sascha@422: { sascha@447: private static Logger log = Logger.getLogger( sascha@447: PolygonRenderer.class); sascha@447: ingo@795: /** ingo@795: * This interfaces describes a single method to retrieve a Paint object ingo@795: * for a given index. ingo@795: */ ingo@795: public interface PaintLookup { sascha@422: ingo@795: /** ingo@795: * ingo@795: * @param index Index. ingo@795: * @return Paint ingo@795: */ ingo@795: Paint getPaint(int index); sascha@422: sascha@422: } // interface PaintLookup sascha@422: ingo@795: /** ingo@795: * This class is used to generate labels for a given series. ingo@795: */ ingo@795: public static class DefaultLabelGenerator sascha@450: implements PolygonSeriesLabelGenerator sascha@450: { ingo@795: /** ingo@795: * Construts an empty DefaultLabelGenerator. ingo@795: */ ingo@795: public DefaultLabelGenerator() { sascha@450: } sascha@422: ingo@795: /** ingo@795: * ingo@795: * @param series A PolygonSeries. ingo@795: * @return The label of series. ingo@795: */ ingo@795: public String generateLabel(PolygonSeries series) { sascha@450: Object label = series.getAttribute("label"); sascha@450: return label != null sascha@450: ? toString(label) sascha@450: : null; sascha@450: } sascha@422: ingo@795: /** ingo@795: * ingo@795: * @param label Object ingo@795: * @return String representaton of label. ingo@795: */ ingo@795: protected String toString(Object label) { sascha@450: return label.toString(); sascha@450: } sascha@450: } // class DefaultLabelGenerator sascha@422: ingo@795: /** ingo@795: * Constructor. ingo@795: */ ingo@795: public static final PolygonSeriesLabelGenerator sascha@450: DEFAULT_LABEL_GENERATOR_INSTANCE = new DefaultLabelGenerator(); sascha@450: ingo@795: /** ingo@795: * ingo@795: */ ingo@795: protected PaintLookup lookup; ingo@795: /** ingo@795: * ingo@795: */ sascha@450: protected PolygonSeriesLabelGenerator labelGenerator; sascha@422: ingo@795: /** ingo@795: * ingo@795: * @param lookup ingo@795: */ sascha@422: public PolygonRenderer(PaintLookup lookup) { sascha@450: this(lookup, null); sascha@422: } sascha@422: ingo@795: /** ingo@795: * ingo@795: * @param lookup ingo@795: * @param labelGenerator ingo@795: */ sascha@450: public PolygonRenderer( sascha@778: PaintLookup lookup, sascha@450: PolygonSeriesLabelGenerator labelGenerator sascha@450: ) { sascha@450: this.lookup = lookup; sascha@450: this.labelGenerator = labelGenerator; sascha@422: } sascha@422: ingo@795: /** ingo@795: * This method draws polygons of each series in dataset into ingo@795: * the given graphics object. If a polygon has no attribute 'fill', we ingo@795: * expect that it is a line, otherwise the polygon is filled. ingo@795: * ingo@795: * @param graphics ingo@795: * @param plot ingo@795: * @param area ingo@795: * @param dataset ingo@795: */ sascha@450: public void drawPolygons( sascha@422: Graphics2D graphics, sascha@451: PolygonPlot plot, sascha@451: Rectangle2D area, sascha@422: PolygonDataset dataset sascha@422: ) { sascha@422: int seriesCount = dataset.getSeriesCount(); sascha@422: for (int i = 0; i < seriesCount; i++) { sascha@422: PolygonSeries series = dataset.getSeries(i); sascha@422: Integer colorIdx = (Integer)series.getAttribute("fill"); sascha@422: sascha@422: if (colorIdx != null) { sascha@422: Paint paint = lookup.getPaint(colorIdx.intValue()); sascha@422: graphics.setPaint(paint != null ? paint : Color.black); sascha@451: graphics.fill(constructShape(plot, area, series, true)); sascha@422: } sascha@422: else { sascha@447: Number lineWidth = (Number)series.getAttribute("line.width"); sascha@447: BasicStroke stroke = new BasicStroke( sascha@447: lineWidth != null ? lineWidth.floatValue() : 1f); sascha@447: graphics.setStroke(stroke); sascha@422: graphics.setPaint(Color.black); sascha@451: graphics.draw(constructShape(plot, area, series, false)); sascha@422: } sascha@422: } sascha@450: } sascha@450: ingo@795: /** ingo@795: * Draw labels at each item of a series in the given dataset. If the series ingo@795: * has no label attritue, no label is drawn. ingo@795: * ingo@795: * @param graphics ingo@795: * @param plot ingo@795: * @param area ingo@795: * @param dataset ingo@795: */ sascha@450: public void drawLabels( sascha@450: final Graphics2D graphics, sascha@450: final PolygonPlot plot, sascha@450: final Rectangle2D area, sascha@450: PolygonDataset dataset sascha@450: ) { sascha@450: if (labelGenerator == null) { sascha@450: return; sascha@450: } sascha@450: sascha@450: final ArrayList bboxes = new ArrayList(); sascha@450: sascha@450: Font font = graphics.getFont(); sascha@450: font = font.deriveFont(Font.PLAIN, Math.max(8, font.getSize()-3)); sascha@450: graphics.setFont(font); sascha@450: FontMetrics metrics = graphics.getFontMetrics(font); sascha@450: sascha@450: for (int i = dataset.getSeriesCount()-1; i >= 0; --i) { sascha@450: PolygonSeries series = dataset.getSeries(i); sascha@450: sascha@450: String label = labelGenerator.generateLabel(series); sascha@450: if (label == null) { sascha@450: continue; sascha@450: } sascha@450: sascha@450: final Rectangle2D box = TextUtilities.getTextBounds( sascha@450: label, graphics, metrics); sascha@450: sascha@450: for (int j = series.getItemCount()-1; j >= 0; --j) { sascha@450: final CompactXYItems ring = series.getItem(j); sascha@450: LevelOrderIndices loi = new LevelOrderIndices(ring.size()-1); sascha@450: Rectangle2D r = (Rectangle2D)loi.visit( sascha@778: new LevelOrderIndices.Visitor() sascha@450: { sascha@450: ValueAxis da = plot.getDomainAxis(); sascha@450: ValueAxis ra = plot.getRangeAxis(); sascha@450: RectangleEdge de = plot.getDomainAxisEdge(); sascha@450: RectangleEdge re = plot.getRangeAxisEdge(); sascha@450: Rectangle2D.Double r = new Rectangle2D.Double( sascha@450: 0d, 0d, box.getWidth(), box.getHeight()); sascha@450: sascha@450: public Object visit(int index) { sascha@778: r.x = da.valueToJava2D(ring.getX(index), area, de) sascha@450: - 0.5*box.getWidth(); sascha@450: r.y = ra.valueToJava2D(ring.getY(index), area, re) sascha@450: + 0.5*box.getHeight(); sascha@450: sascha@450: for (Rectangle2D b: bboxes) { sascha@450: if (b.intersects(r)) { sascha@450: return null; sascha@450: } sascha@450: } sascha@450: return r; sascha@450: } sascha@450: }); sascha@450: sascha@450: if (r != null) { sascha@450: bboxes.add(r); sascha@450: graphics.drawString( sascha@450: label, (float)r.getX(), (float)r.getY()); sascha@450: } sascha@450: } // for all items in series sascha@450: } // for all series sascha@422: } sascha@422: ingo@795: /** ingo@795: * Creates a shape made up of the CompactXYItems object stored in the given ingo@795: * series. ingo@795: * ingo@795: * @param plot The plot. ingo@795: * @param area The boundary. ingo@795: * @param series The series storing the items. ingo@795: * @param close Specifies if the polygon should be closed or not. ingo@795: * @return ingo@795: */ sascha@451: protected Shape constructShape( sascha@451: PolygonPlot plot, sascha@451: Rectangle2D area, sascha@778: PolygonSeries series, sascha@451: boolean close sascha@451: ) { sascha@451: ValueAxis da = plot.getDomainAxis(); sascha@451: ValueAxis ra = plot.getRangeAxis(); sascha@451: RectangleEdge de = plot.getDomainAxisEdge(); sascha@451: RectangleEdge re = plot.getRangeAxisEdge(); sascha@451: sascha@422: CompactXYItems [] rings = series.getRings(); sascha@451: GeneralPath path = new GeneralPath(); sascha@451: sascha@422: for (int i = 0; i < rings.length; ++i) { sascha@451: sascha@422: CompactXYItems ring = rings[i]; sascha@451: sascha@422: double [] data = ring.getData(); sascha@451: sascha@422: if (data.length >= 2) { sascha@451: path.moveTo( sascha@451: (float)da.valueToJava2D(data[0], area, de), sascha@451: (float)ra.valueToJava2D(data[1], area, re)); sascha@422: } sascha@422: for (int j = 2; j < data.length;) { sascha@451: path.lineTo( sascha@451: (float)da.valueToJava2D(data[j++], area, de), sascha@451: (float)ra.valueToJava2D(data[j++], area, re)); sascha@422: } sascha@437: if (close) { sascha@437: path.closePath(); sascha@437: } sascha@422: } sascha@422: return path; sascha@422: } sascha@422: ingo@795: /** ingo@795: * Retrieves the bounding box of a dataset. ingo@795: * ingo@795: * @param dataset ingo@795: * @return ingo@795: */ sascha@422: public Rectangle2D getBoundingBox(PolygonDataset dataset) { sascha@422: Rectangle2D bbox = null; sascha@422: sascha@422: for (int i = 0, N = dataset.getSeriesCount(); i < N; i++) { sascha@422: Range domain = dataset.getSeries(i).getDomainBounds(); sascha@422: Range range = dataset.getSeries(i).getRangeBounds(); sascha@422: sascha@422: double x = domain.getLowerBound(); sascha@422: double y = range.getLowerBound(); sascha@422: double w = Math.abs(domain.getUpperBound() - x); sascha@422: double h = Math.abs(range.getUpperBound() - y); sascha@422: sascha@422: if (bbox == null) { sascha@422: bbox = new Rectangle2D.Double(x, y, w, h); sascha@422: } sascha@422: else { sascha@422: bbox.add(new Rectangle2D.Double(x, y, w, h)); sascha@422: } sascha@422: } sascha@422: sascha@422: return bbox; sascha@422: } sascha@422: ingo@795: /** ingo@795: * ingo@795: * @param series ingo@795: * @return the bounds of a series. ingo@795: */ sascha@422: public Rectangle2D getBounds(PolygonSeries series) { sascha@422: sascha@422: Range domain = series.getDomainBounds(); sascha@422: Range range = series.getRangeBounds(); sascha@422: sascha@422: return new Rectangle2D.Double( sascha@422: domain.getLowerBound(), range.getLowerBound(), sascha@422: domain.getUpperBound(), range.getUpperBound() sascha@422: ); sascha@422: } sascha@422: } sascha@422: // vim:set ts=4 sw=4 si et sta sts=4 fenc=utf-8 :