All Downloads are FREE. Search and download functionalities are using the official Maven repository.

org.meteoinfo.chart.plot.XY1DPlot Maven / Gradle / Ivy

There is a newer version: 3.8
Show newest version
/*
 * To change this license header, choose License Headers in Project Properties.
 * To change this template file, choose Tools | Templates
 * and open the template in the editor.
 */
package org.meteoinfo.chart.plot;

import org.meteoinfo.chart.ChartLegend;
import org.meteoinfo.chart.Location;
import org.meteoinfo.chart.axis.TimeAxis;
import org.meteoinfo.common.Extent;
import org.meteoinfo.common.MIMath;
import org.meteoinfo.common.PointF;
import org.meteoinfo.common.colors.ColorUtil;
import org.meteoinfo.data.*;
import org.meteoinfo.geo.drawing.Draw;
import org.meteoinfo.geometry.legend.*;
import org.meteoinfo.geometry.shape.ShapeTypes;

import java.awt.*;
import java.awt.geom.AffineTransform;
import java.awt.geom.Line2D;
import java.awt.geom.Rectangle2D;
import java.util.ArrayList;
import java.util.List;
import java.util.logging.Level;
import java.util.logging.Logger;

/**
 *
 * @author yaqiang
 */
public class XY1DPlot extends AbstractPlot2D {

    // 
    private XYListDataset dataset;
    private ChartPlotMethod chartPlotMethod;
    private List seriesLegends;
    private boolean useBreak2D;
    private float barWidth;
    private boolean autoBarWidth;

    // 
    // 
    /**
     * Constructor
     */
    public XY1DPlot() {
        super();

        this.dataset = new XYListDataset();
        this.chartPlotMethod = ChartPlotMethod.LINE;
        this.useBreak2D = false;
        this.seriesLegends = new ArrayList<>();
        this.barWidth = 0.8f;
        this.autoBarWidth = true;
    }

    /**
     * Constructor
     *
     * @param dateset Dataset
     */
    public XY1DPlot(XYDataset dateset) {
        this();
        this.setDataset(dateset);
    }

    /**
     * Constructor
     *
     * @param orientation Plot orientation
     * @param dateset Dataset
     */
    public XY1DPlot(PlotOrientation orientation, XYDataset dateset) {
        this();
        this.setPlotOrientation(orientation);
        this.setDataset(dateset);
    }

    /**
     * Constructor
     *
     * @param isTime If x axis is time
     * @param dateset Dataset
     */
    public XY1DPlot(boolean isTime, XYDataset dateset) {
        this();
        if (isTime) {
            try {
                this.setXAxis(new TimeAxis("X", true));
            } catch (CloneNotSupportedException ex) {
                Logger.getLogger(XY1DPlot.class.getName()).log(Level.SEVERE, null, ex);
            }
            this.getAxis(Location.TOP).setDrawTickLabel(false);
        }
        //this.getXAxis().setTimeAxis(isTime);
        this.setDataset(dateset);
    }

    /**
     * Constructor
     *
     * @param isTime If x axis is time
     * @param cpMethod Plot method
     * @param dateset Dataset
     */
    public XY1DPlot(boolean isTime, ChartPlotMethod cpMethod, XYDataset dateset) {
        this();
        if (isTime) {
            try {
                this.setXAxis(new TimeAxis("X", true));
            } catch (CloneNotSupportedException ex) {
                Logger.getLogger(XY1DPlot.class.getName()).log(Level.SEVERE, null, ex);
            }
            this.getAxis(Location.TOP).setDrawTickLabel(false);
        }
        //this.getXAxis().setTimeAxis(isTime);
        this.setChartPlotMethod(cpMethod);
        this.setDataset(dateset);
    }

    /**
     * Constructor
     *
     * @param isTime If x axis is time
     * @param orientation Plot orientation
     * @param dateset Dataset
     */
    public XY1DPlot(boolean isTime, PlotOrientation orientation, XYDataset dateset) {
        this();
        if (isTime) {
            try {
                this.setXAxis(new TimeAxis("X", true));
            } catch (CloneNotSupportedException ex) {
                Logger.getLogger(XY1DPlot.class.getName()).log(Level.SEVERE, null, ex);
            }
            this.getAxis(Location.TOP).setDrawTickLabel(false);
        }
        //this.getXAxis().setTimeAxis(isTime);
        this.setPlotOrientation(orientation);
        this.setDataset(dateset);
    }

    // 
    // 
    @Override
    public XYDataset getDataset() {
        return dataset;
    }

    @Override
    public void setDataset(Dataset value) {
        dataset = (XYListDataset) value;
        Extent extent = this.getAutoExtent();
        this.setDrawExtent(extent);
        this.setExtent(extent);
        this.updateSeriesLegend();
    }

    private void updateSeriesLegend() {
        //this.seriesLegends.clear();
        int si = this.seriesLegends.size();
        if (si > dataset.getSeriesCount()) {
            si = 0;
        }
        for (int i = si; i < dataset.getSeriesCount(); i++) {
            Color color = ColorUtil.getCommonColor(i);
            switch (this.chartPlotMethod) {
                case LINE:
                case LINE_POINT:
                    PolylineBreak plb = new PolylineBreak();
                    if (this.chartPlotMethod == ChartPlotMethod.LINE) {
                        plb.setDrawSymbol(false);
                    } else {
                        plb.setDrawSymbol(true);
                        plb.setSymbolColor(color);
                        plb.setSymbolFillColor(color);
                    }
                    plb.setColor(color);
                    plb.setCaption(dataset.getSeriesKey(i));
                    seriesLegends.add(new SeriesLegend(plb));
                    break;
                case POINT:
                    PointBreak pb = new PointBreak();
                    pb.setColor(color);
                    pb.setCaption(dataset.getSeriesKey(i));
                    seriesLegends.add(new SeriesLegend(pb));
                    break;
                case BAR:
                    PolygonBreak pgb = new PolygonBreak();
                    pgb.setColor(color);
                    pgb.setCaption(dataset.getSeriesKey(i));
                    seriesLegends.add(new SeriesLegend(pgb));
                    break;
            }
        }
    }

    /**
     * Get chart plot method
     *
     * @return Chart plot method
     */
    public ChartPlotMethod getChartPlotMethod() {
        return this.chartPlotMethod;
    }

    /**
     * Set chart plot method
     *
     * @param value Chart plot method
     */
    public void setChartPlotMethod(ChartPlotMethod value) {
        this.chartPlotMethod = value;
        if (this.dataset != null) {
            this.updateSeriesLegend();
        }
    }

    @Override
    public PlotType getPlotType() {
        return PlotType.XY;
    }

    /**
     * Get Series legend breaks
     *
     * @return Series legend breaks
     */
    public List getLegendBreaks() {
        return this.seriesLegends;
    }

//    /**
//     * Get point breaks
//     *
//     * @return Point breaks
//     */
//    public PointBreak[] getPointBreaks() {
//        return this.pointBreaks;
//    }
    /**
     * If use item 2D point breaks
     *
     * @return Boolean
     */
    public boolean isUseBreak2D() {
        return this.useBreak2D;
    }

    /**
     * Set if use item 2D point breaks
     *
     * @param value Boolean
     */
    public void setUseBeak2D(boolean value) {
        this.useBreak2D = value;
    }

    /**
     * Get bar width ratio
     *
     * @return Bar width ratio
     */
    public float getBarWidth() {
        return this.barWidth;
    }

    /**
     * Set bar width ratio
     *
     * @param value Bar width ratio
     */
    public void setBarWidth(float value) {
        this.barWidth = value;
    }

    /**
     * Get if automatically decide bar width
     *
     * @return Boolean
     */
    public boolean isAutoBarWidth() {
        return this.autoBarWidth;
    }

    /**
     * Set if automatically decide bar height
     *
     * @param value Boolean
     */
    public void setAutoBarWidth(boolean value) {
        this.autoBarWidth = value;
    }

    // 
    //     
    /**
     * Add a series data
     *
     * @param seriesKey Series key
     * @param xvs X value array
     * @param yvs Y value array
     */
    public void addSeries(String seriesKey, double[] xvs, double[] yvs) {
        ((XYListDataset) this.dataset).addSeries(seriesKey, xvs, yvs);
        PolylineBreak plb = new PolylineBreak();
        plb.setColor(ColorUtil.getCommonColor(this.dataset.getSeriesCount()));
        plb.setCaption(seriesKey);
        seriesLegends.add(new SeriesLegend(plb));

        Extent extent = this.getAutoExtent();
        this.setDrawExtent(extent);
    }

    /**
     * Remove last series
     */
    public void removeLastSeries() {
        XYListDataset ds = (XYListDataset) this.dataset;
        ds.removeSeries(dataset.getSeriesCount() - 1);
        this.seriesLegends.remove(this.seriesLegends.size() - 1);

        Extent extent = this.getAutoExtent();
        this.setDrawExtent(extent);
    }

    private PointF[] getScreenPoints(double[] xdata, double[] ydata, List mvIdx, Rectangle2D area) {
        int len = xdata.length;
        PointF[] points = new PointF[len];
        double[] xy;
        if (this.getPlotOrientation() == PlotOrientation.VERTICAL) {
            for (int j = 0; j < len; j++) {
                xy = this.projToScreen(xdata[j], ydata[j], area);
                points[j] = new PointF((float) xy[0], (float) xy[1]);
            }
        } else {
            for (int j = 0; j < len; j++) {
                xy = this.projToScreen(xdata[j], ydata[j], area);
                points[j] = new PointF((float) xy[0], (float) xy[1]);
            }
        }
//        if (this.getYAxis().isInverse()) {
//            PointF[] npoints = new PointF[len];
//            PointF p;
//            float y;
//            for (int j = 0; j < len; j++) {
//                p = points[len - j - 1];
//                y = (float) area.getHeight() - p.Y;
//                npoints[j] = new PointF(p.X, y);
//            }
//            points = npoints;
//            if (!mvIdx.isEmpty()) {
//                for (int j = 0; j < mvIdx.size(); j++) {
//                    mvIdx.set(j, len - mvIdx.get(j) - 1);
//                }
//            }
//        }
//        if (this.getXAxis().isInverse()) {
//            PointF[] npoints = new PointF[len];
//            PointF p;
//            float x;
//            for (int j = 0; j < len; j++) {
//                p = points[len - j - 1];
//                x = (float) area.getWidth() - p.X;
//                npoints[j] = new PointF(x, p.Y);
//            }
//            points = npoints;
//            if (!mvIdx.isEmpty()) {
//                for (int j = 0; j < mvIdx.size(); j++) {
//                    mvIdx.set(j, len - mvIdx.get(j) - 1);
//                }
//            }
//        }

        return points;
    }

    @Override
    void drawGraph(Graphics2D g, Rectangle2D area) {
        AffineTransform oldMatrix = g.getTransform();
        Rectangle oldRegion = g.getClipBounds();
        g.setClip(area);
        g.translate(area.getX(), area.getY());

        //Draw background
        if (this.background != null) {
            g.setColor(this.getBackground());
            g.fill(new Rectangle2D.Double(0, 0, area.getWidth(), area.getHeight()));
        }

        double[] xy;
        xy = this.projToScreen(0, 0, area);
        float y0 = (float) xy[1];
        boolean drawBaseline = false;
        for (int i = 0; i < this.dataset.getSeriesCount(); i++) {
            XYSeriesData sdata = this.dataset.getSeriesData(i);
            int len = sdata.dataLength();
            List mvIdx = sdata.getMissingValueIndex();
            PointF[] points = getScreenPoints(sdata.getXdata(), sdata.getYdata(), mvIdx, area);
            SeriesLegend slegend = this.seriesLegends.get(i);
            if (slegend.isLine()) {
                if (mvIdx.isEmpty()) {
                    Draw.drawPolyline(points, (PolylineBreak) slegend.getLegendBreak(), g);
                } else {
                    Draw.drawPolyline(points, (PolylineBreak) slegend.getLegendBreak(), g, mvIdx);
                }
            } else if (slegend.isPoint()) {
                if (slegend.isMutiple()) {
                    for (int j = 0; j < len; j++) {
                        if (!mvIdx.contains(j)) {
                            Draw.drawPoint(points[j], (PointBreak) slegend.getLegendBreak(j), g);
                        }
                    }
                } else {
                    for (int j = 0; j < len; j++) {
                        if (!mvIdx.contains(j)) {
                            Draw.drawPoint(points[j], (PointBreak) slegend.getLegendBreak(), g);
                        }
                    }
                }
            } else if (slegend.isPolygon()) {
                switch (slegend.getPlotMethod()) {
                    case BAR:
                        if (sdata instanceof XYErrorSeriesData) {
                            drawBaseline = true;
                            XYErrorSeriesData esdata = (XYErrorSeriesData) sdata;
                            double[] bottom = esdata.getBottom();
                            float[] yBottoms = null;
                            if (bottom != null) {
                                yBottoms = new float[bottom.length];
                                for (int j = 0; j < bottom.length; j++) {
                                    xy = this.projToScreen(esdata.getBottom(j), esdata.getBottom(j), area);
                                    yBottoms[j] = (float) xy[1];
                                }
                            }
                            float width = this.barWidth;
                            if (this.autoBarWidth) {
                                if (points.length > 1) {
                                    width = (float) ((points[1].X - points[0].X) * 0.5) / this.dataset.getSeriesCount();
                                } else {
                                    width = (float) (area.getWidth() / 10) / this.dataset.getSeriesCount();
                                }
                                float height;
                                PolygonBreak pgb;
                                for (int j = 0; j < len; j++) {
                                    if (!mvIdx.contains(j)) {
                                        pgb = (PolygonBreak) slegend.getLegendBreak(j);
                                        height = Math.abs((float) (points[j].Y - y0));
                                        float yBottom = y0;
                                        if (yBottoms != null) {
                                            if (j < yBottoms.length) {
                                                yBottom = yBottoms[j];
                                            } else {
                                                yBottom = yBottoms[0];
                                            }
                                        }
                                        float yb = yBottom;
                                        if (points[j].Y >= y0) {
                                            yb += height;
                                        }
                                        Draw.drawBar(new PointF(points[j].X - width * this.dataset.getSeriesCount() / 2
                                                + i * width, yb), width, height, pgb, g, false, 5);
                                        if (esdata.getYerror() != null) {
                                            PointF p = (PointF) points[j].clone();
                                            p.Y -= y0 - yBottom;
                                            double elen = 6;
                                            double error = esdata.getYerror(j);
                                            error = this.projYLength(error, area);
                                            double x = p.X - width * this.dataset.getSeriesCount() / 2
                                                    + i * width + width / 2;
                                            g.setColor(slegend.getErrorColor());
                                            g.draw(new Line2D.Double(x, p.Y - error, x, p.Y + error));
                                            g.draw(new Line2D.Double(x - (elen * 0.5), p.Y - error, x + (elen * 0.5), p.Y - error));
                                            g.draw(new Line2D.Double(x - (elen * 0.5), p.Y + error, x + (elen * 0.5), p.Y + error));
                                        }
                                    }
                                }
                            } else {
                                width = (float) this.projXLength(width, area);
                                float height;
                                PolygonBreak pgb;
                                for (int j = 0; j < len; j++) {
                                    if (!mvIdx.contains(j)) {
                                        pgb = (PolygonBreak) slegend.getLegendBreak(j);
                                        height = Math.abs((float) (points[j].Y - y0));
                                        float yBottom = y0;
                                        if (yBottoms != null) {
                                            if (j < yBottoms.length) {
                                                yBottom = yBottoms[j];
                                            } else {
                                                yBottom = yBottoms[0];
                                            }
                                        }
                                        float yb = yBottom;
                                        if (points[j].Y >= y0) {
                                            yb += height;
                                        }
                                        Draw.drawBar(new PointF(points[j].X, yb), width, height, pgb, g, false, 5);
                                        if (esdata.getYerror() != null) {
                                            PointF p = (PointF) points[j].clone();
                                            p.Y -= y0 - yBottom;
                                            double elen = 6;
                                            double error = esdata.getYerror(j);
                                            error = this.projYLength(error, area);
                                            double x = p.X + width / 2;
                                            g.setColor(slegend.getErrorColor());
                                            g.draw(new Line2D.Double(x, p.Y - error, x, p.Y + error));
                                            g.draw(new Line2D.Double(x - (elen * 0.5), p.Y - error, x + (elen * 0.5), p.Y - error));
                                            g.draw(new Line2D.Double(x - (elen * 0.5), p.Y + error, x + (elen * 0.5), p.Y + error));
                                        }
                                    }
                                }
                            }
                        } else {
                            drawBaseline = true;                            
                            float width = this.barWidth;
                            if (this.autoBarWidth) {
                                if (points.length > 1) {
                                    width = (float) ((points[1].X - points[0].X) * 0.5) / this.dataset.getSeriesCount();
                                } else {
                                    width = (float) (area.getWidth() / 10) / this.dataset.getSeriesCount();
                                }
                                float height;
                                PolygonBreak pgb;
                                for (int j = 0; j < len; j++) {
                                    if (!mvIdx.contains(j)) {
                                        pgb = (PolygonBreak) slegend.getLegendBreak(j);
                                        height = Math.abs((float) (points[j].Y - y0));
                                        float yb = y0;
                                        if (points[j].Y >= y0) {
                                            yb += height;
                                        }
                                        Draw.drawBar(new PointF(points[j].X - width * this.dataset.getSeriesCount() / 2
                                                + i * width, yb), width, height, pgb, g, false, 5);
                                    }
                                }
                            } else {
                                width = (float) this.projXLength(width, area);
                                float height;
                                PolygonBreak pgb;
                                for (int j = 0; j < len; j++) {
                                    if (!mvIdx.contains(j)) {
                                        pgb = (PolygonBreak) slegend.getLegendBreak(j);
                                        height = Math.abs((float) (points[j].Y - y0));
                                        float yb = y0;
                                        if (points[j].Y >= y0) {
                                            yb += height;
                                        }
                                        Draw.drawBar(new PointF(points[j].X, yb), width, height, pgb, g, false, 5);                                        
                                    }
                                }
                            }
                        }
                        break;
                    case FILL:
                        XYYSeriesData xyydata = (XYYSeriesData) sdata;
                        PointF[] y2Points = this.getScreenPoints(xyydata.getXdata(), xyydata.getY2data(), mvIdx, area);
                        if (xyydata.getWhere() == null) {
                            PointF[] npoints = new PointF[len * 2];
                            for (int j = 0; j < len; j++) {
                                npoints[j] = points[len - j - 1];
                                npoints[j + len] = y2Points[j];
                            }
                            Draw.drawPolygon(npoints, (PolygonBreak) slegend.getLegendBreak(), g);
                        } else {
                            boolean ob = false;
                            List> idxs = new ArrayList<>();
                            List idx = new ArrayList<>();
                            for (int j = 0; j < len; j++) {
                                if (xyydata.getWhere().get(j)) {
                                    if (!ob) {
                                        idx = new ArrayList<>();
                                    }
                                    idx.add(j);
                                } else if (ob) {
                                    idxs.add(idx);
                                }
                                ob = xyydata.getWhere().get(j);
                            }
                            for (List index : idxs) {
                                int nn = index.size();
                                if (nn >= 2) {
                                    PointF[] npoints = new PointF[nn * 2];
                                    int ii, ii2;
                                    for (int j = 0; j < nn; j++) {
                                        ii = index.get(j);
                                        ii2 = index.get(nn - j - 1);
                                        npoints[j] = points[ii];
                                        npoints[j + index.size()] = y2Points[ii2];
                                    }
                                    Draw.drawPolygon(npoints, (PolygonBreak) slegend.getLegendBreak(), g);
                                }
                            }
                        }
                        break;
                }
            }

            //Draw baseline
            if (drawBaseline) {
                g.setColor(Color.black);
                g.draw(new Line2D.Double(0, y0, area.getWidth(), y0));
            }

            //Draw error bar
            if (sdata instanceof XYErrorSeriesData) {
                XYErrorSeriesData esdata = (XYErrorSeriesData) sdata;
                g.setColor(slegend.getLegendBreak().getColor());
                PointF p;
                double error;
                double elen = 6;
                if (esdata.getYerror() != null) {
                    if (slegend.getPlotMethod() == ChartPlotMethod.BAR) {

                    } else {
                        for (int j = 0; j < len; j++) {
                            if (!mvIdx.contains(j)) {
                                p = points[j];
                                error = esdata.getYerror(j);
                                error = this.projYLength(error, area);
                                g.draw(new Line2D.Double(p.X, p.Y - error, p.X, p.Y + error));
                                g.draw(new Line2D.Double(p.X - (elen * 0.5), p.Y - error, p.X + (elen * 0.5), p.Y - error));
                                g.draw(new Line2D.Double(p.X - (elen * 0.5), p.Y + error, p.X + (elen * 0.5), p.Y + error));
                            }
                        }
                    }
                }
                if (esdata.getXerror() != null) {
                    for (int j = 0; j < len; j++) {
                        if (!mvIdx.contains(j)) {
                            p = points[j];
                            error = esdata.getXerror(j);
                            error = this.projXLength(error, area);
                            g.draw(new Line2D.Double(p.X - error, p.Y, p.X + error, p.Y));
                            g.draw(new Line2D.Double(p.X - error, p.Y - (elen * 0.5), p.X - error, p.Y + (elen * 0.5)));
                            g.draw(new Line2D.Double(p.X + error, p.Y - (elen * 0.5), p.X + error, p.Y + (elen * 0.5)));
                        }
                    }
                }
            }
        }

        //Draw texts
//        for (ChartText text : this.getTexts()) {
//            xy = this.projToScreen(text.getX(), text.getY(), area);
//            float x = (float) xy[0];
//            float y = (float) xy[1];
//            g.setFont(text.getFont());
//            g.setColor(text.getColor());
//            //Dimension dim = Draw.getStringDimension(text.getText(), g);
//            //y -= dim.height * 2 / 3;
//            Draw.drawString(g, text.getText(), x, y);
//        }
        g.setTransform(oldMatrix);
        g.setClip(oldRegion);
    }

    /**
     * Get a item point break
     *
     * @param seriesIdx Series index
     * @param itemIdx Item index
     * @return Item point break;
     */
    public PointBreak getItemPointBreak(int seriesIdx, int itemIdx) {
        return (PointBreak) this.seriesLegends.get(seriesIdx).getLegendBreak(itemIdx);
    }

    /**
     * Set item point break
     *
     * @param seriesIdx Series index
     * @param itemIdx Item index
     * @param pb Item point break
     */
    public void setItemPointBreak(int seriesIdx, int itemIdx, PointBreak pb) {
        this.seriesLegends.get(seriesIdx).setLegendBreak(itemIdx, pb);
    }

    /**
     * Get legend break
     *
     * @param seriesIdx Series index
     * @return Legend break
     */
    public ColorBreak getLegendBreak(int seriesIdx) {
        return this.seriesLegends.get(seriesIdx).getLegendBreak();
    }

    /**
     * Set legend break
     *
     * @param seriesIdx Series index
     * @param cb Legend break
     */
    public void setLegendBreak(int seriesIdx, ColorBreak cb) {
        this.seriesLegends.get(seriesIdx).setLegendBreak(cb);
    }

    /**
     * Set series legend
     *
     * @param seriesIdx Series index
     * @param sLegend SeriesLegend
     */
    public void setLegendBreak(int seriesIdx, SeriesLegend sLegend) {
        this.seriesLegends.set(seriesIdx, sLegend);
    }

//    /**
//     * Get point break
//     *
//     * @param seriesIdx Series index
//     * @return Point break
//     */
//    public PointBreak getPointBreak(int seriesIdx) {
//        return this.pointBreaks[seriesIdx];
//    }
//
//    /**
//     * Set point break
//     *
//     * @param seriesIdx Series index
//     * @param pb Point break
//     */
//    public void setPointBreak(int seriesIdx, PointBreak pb) {
//        this.pointBreaks[seriesIdx] = pb;
//    }
//
//    /**
//     * Get polygon break
//     *
//     * @param seriesIdx Series index
//     * @return Polygon break
//     */
//    public PolygonBreak getPolygonBreak(int seriesIdx) {
//        return this.polygonBreaks[seriesIdx];
//    }
//
//    /**
//     * Set polygon break
//     *
//     * @param seriesIdx Series index
//     * @param pgb Polygon break
//     */
//    public void setPolygonBreak(int seriesIdx, PolygonBreak pgb) {
//        this.polygonBreaks[seriesIdx] = pgb;
//    }
    private double getBarXInterval(int idx) {
        double[] xvalues = this.dataset.getXValues(idx);
        if (xvalues.length == 1) {
            if (xvalues[0] == 0) {
                return 1;
            } else {
                return xvalues[0] / 10;
            }
        } else {
            return xvalues[1] - xvalues[0];
        }
    }

    private int getBarIndex() {
        int idx = -1;
        for (int i = 0; i < this.seriesLegends.size(); i++) {
            if (this.seriesLegends.get(i).getPlotMethod() == ChartPlotMethod.BAR) {
                idx = i;
                break;
            }
        }
        return idx;
    }

    /**
     * Get auto extent
     *
     * @return Auto extent
     */
    @Override
    public Extent getAutoExtent() {
        Extent extent = dataset.getDataExtent();
        if (extent.minX == extent.maxX){
            extent.minX = extent.minX - Math.abs(extent.minX);
            extent.maxX = extent.maxX + Math.abs(extent.minX);
        }
        if (extent.minY == extent.maxY){
            extent.minY = extent.minY - Math.abs(extent.minY);
            extent.maxY = extent.maxY + Math.abs(extent.maxY);
        }
        
        int barIdx = this.getBarIndex();
        if (barIdx >= 0) {
            double dx = getBarXInterval(barIdx);
            extent.minX -= dx;
            extent.maxX += dx;
        }
        double[] xValues;
        if (this.getXAxis() instanceof TimeAxis) {
            //if (this.getXAxis().isTimeAxis()) {
            xValues = (double[]) MIMath.getIntervalValues(extent.minX, extent.maxX, false).get(0);
            xValues[0] = extent.minX;
            xValues[xValues.length - 1] = extent.maxX;
        } else {
            xValues = (double[]) MIMath.getIntervalValues(extent.minX, extent.maxX, true).get(0);
        }
        double[] yValues = (double[]) MIMath.getIntervalValues(extent.minY, extent.maxY, true).get(0);
        if (this.getPlotOrientation() == PlotOrientation.VERTICAL) {
            return new Extent(xValues[0], xValues[xValues.length - 1], yValues[0], yValues[yValues.length - 1]);
        } else {
            return new Extent(yValues[0], yValues[yValues.length - 1], xValues[0], xValues[xValues.length - 1]);
        }
    }

    /**
     * Set auto extent
     */
    @Override
    public void setAutoExtent() {
        Extent extent = this.getAutoExtent();
        this.setDrawExtent(extent);
    }

    /**
     * Get legend scheme
     *
     * @return Legend scheme
     */
    public LegendScheme getLegendScheme() {
//        LegendScheme ls = null;
//        switch (this.chartPlotMethod) {
//            case LINE:
//            case LINE_POINT:
//                ls = new LegendScheme(ShapeTypes.Polyline);
//                ls.getLegendBreaks().addAll(Arrays.asList(this.lineBreaks));
//                break;
//            case POINT:
//                ls = new LegendScheme(ShapeTypes.Point);
//                ls.getLegendBreaks().addAll(Arrays.asList(this.pointBreaks));
//                break;
//            case BAR:
//                ls = new LegendScheme(ShapeTypes.Polygon);
//                ls.getLegendBreaks().addAll(Arrays.asList(this.polygonBreaks));
//                break;
//        }
        ShapeTypes stype = ShapeTypes.POLYLINE;
        LegendScheme ls = new LegendScheme(stype);
        for (SeriesLegend slegend : this.seriesLegends) {
            ls.getLegendBreaks().add(slegend.getLegendBreak());
        }
        return ls;
    }

    @Override
    public void updateLegendScheme() {
        if (this.getLegend() == null) {
            this.setLegend(new ChartLegend(this.getLegendScheme()));
        } else {
            this.getLegend().setLegendScheme(this.getLegendScheme());
        }
    }

    //    
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy