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

ngmf.ui.graph.PlotView Maven / Gradle / Ivy

/*
 * $Id: PlotView.java 3fabf23f909a 2018-12-08 [email protected] $
 * 
 * This file is part of the Object Modeling System (OMS),
 * 2007-2012, Olaf David and others, Colorado State University.
 *
 * OMS is free software: you can redistribute it and/or modify
 * it under the terms of the GNU Lesser General Public License as published by
 * the Free Software Foundation, version 2.1.
 *
 * OMS 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 Lesser General Public License for more details.
 *
 * You should have received a copy of the GNU Lesser General Public License
 * along with OMS.  If not, see .
 */
package ngmf.ui.graph;

import java.awt.Color;
import java.io.File;
import java.io.FileReader;
import java.io.IOException;
import java.util.ArrayList;
import java.util.Calendar;
import java.util.Date;
import java.util.GregorianCalendar;
import java.util.List;
import javax.swing.JComponent;
import oms3.Conversions;
import oms3.io.CSProperties;
import oms3.io.CSTable;
import oms3.io.DataIO;
import oms3.util.Dates;
import org.jfree.chart.ChartFactory;
import org.jfree.chart.ChartPanel;
import org.jfree.chart.JFreeChart;
import org.jfree.chart.axis.AxisLocation;
import org.jfree.chart.axis.DateAxis;
import org.jfree.chart.axis.NumberAxis;
import org.jfree.chart.labels.StandardXYToolTipGenerator;
import org.jfree.chart.plot.CombinedDomainXYPlot;
import org.jfree.chart.plot.PlotOrientation;
import org.jfree.chart.plot.XYPlot;
import org.jfree.chart.renderer.xy.XYLineAndShapeRenderer;
import org.jfree.data.general.SeriesException;
import org.jfree.data.time.TimeSeries;
import org.jfree.data.time.TimeSeriesCollection;
import org.jfree.data.xy.XYDataset;
import org.jfree.data.time.Millisecond;
import org.jfree.data.xy.XYSeries;
import org.jfree.data.xy.XYSeriesCollection;
import org.omscentral.modules.analysis.esp.ESPTimeSeries;
import org.omscentral.modules.analysis.esp.ESPToolPanel;
import org.omscentral.modules.analysis.esp.EnsembleData;
import org.omscentral.modules.analysis.esp.ModelDateTime;

/**
 *
 * @author od
 */
public class PlotView {
    
    private static final Color[] col = {Color.RED, Color.BLUE, Color.GREEN, Color.BLACK, Color.PINK, Color.MAGENTA,  Color.CYAN};

    static XYDataset createDatasetCombined(Date[] date, List label, List vals) {
        TimeSeriesCollection tsc = new TimeSeriesCollection();

        for (int i = 0; i < label.size(); i++) {
            String n = label.get(i);
            Double[] v = vals.get(i);

            TimeSeries series = new TimeSeries(n, Millisecond.class);
            for (int j = 0; j < v.length; j++) {
                try {
//                    series.add(new Millisecond(date[j]), v[j]);
                     series.add(new Millisecond(date[j]), v[j] == -9999 ? null : v[j]);
                } catch (SeriesException E) {
                    System.err.println("Error adding to series " + date[j] + " " + v[j]);
                }
            }
            tsc.addSeries(series);
        }
        return tsc;
    }

    static XYDataset createDataset(Date[] date, List label, List vals) {
        TimeSeriesCollection tsc = new TimeSeriesCollection();
        for (int i = 0; i < label.size(); i++) {
            String n = label.get(i);
            Double[] v = vals.get(i);
            TimeSeries series = new TimeSeries(n, Millisecond.class);
            for (int j = 0; j < v.length; j++) {
                try {
//                    series.add(new Millisecond(date[j]), v[j]);
                     series.add(new Millisecond(date[j]), v[j] == -9999 ? null : v[j]);
                } catch (SeriesException E) {
                    System.err.println("Error adding to series " + date[j] + " " + v[j]);
                }
            }
            tsc.addSeries(series);
        }
        return tsc;
    }

    static XYDataset[] createDatasets(Date[] date, List label, List vals) {
        XYDataset[] sets = new XYDataset[vals.size()];
        for (int i = 0; i < sets.length; i++) {
            sets[i] = new TimeSeriesCollection();
            String n = label.get(i);
            Double[] v = vals.get(i);
            TimeSeries series = new TimeSeries(n, Millisecond.class);
            for (int j = 0; j < v.length; j++) {
                try {
//                    series.add(new Millisecond(date[j]), v[j]);
                    series.add(new Millisecond(date[j]), v[j] == -9999 ? null : v[j]);
                } catch (SeriesException E) {
                    System.err.println("Error adding to series " + date[j] + " " + v[j]);
                }
            }
            ((TimeSeriesCollection) sets[i]).addSeries(series);
        }
        return sets;
    }

    static XYDataset createXYDataset(Integer[] x, List label, List vals) {
        XYSeriesCollection tsc = new XYSeriesCollection();
        for (int i = 0; i < label.size(); i++) {
            String n = label.get(i);
            Double[] v = vals.get(i);
            XYSeries series = new XYSeries(n);
            for (int j = 0; j < v.length; j++) {
                try {
//                    series.add(x[j], v[j]);
                     series.add(x[j], v[j] == -9999 ? null : v[j]);
                } catch (SeriesException E) {
                    System.err.println("Error adding to series " + x[j] + " " + v[j]);
                }
            }
            tsc.addSeries(series);
        }
        return tsc;
    }

    static XYDataset createXYScatterDataset(Double[] x, Double[] y) {
        XYSeriesCollection tsc = new XYSeriesCollection();
        XYSeries series = new XYSeries("");
        for (int j = 0; j < x.length; j++) {
            try {
//                series.add(x[j], y[j]);
                     series.add(x[j], y[j] == -9999 ? null : y[j]);
            } catch (SeriesException E) {
                System.err.println("Error adding to series " + x[j] + " " + y[j]);
            }
        }
        tsc.addSeries(series);
        return tsc;
    }

    static JComponent comp(JFreeChart chart) {
        chart.getPlot().setBackgroundPaint(new Color(233, 232, 226));
        chart.setTextAntiAlias(true);
        ChartPanel cp = new ChartPanel(chart);
        cp.setDomainZoomable(true);
        cp.setRangeZoomable(true);
        return cp;
    }

    public static JComponent createTSChart(String title, Date[] date, List label,
            List vals, int view, List y) {
        
        XYDataset[] dataset = createDatasets(date, label, vals);
        if (view == 0) {  // stacked
            DateAxis timeAxis = new DateAxis(label.get(0));
            CombinedDomainXYPlot plot = new CombinedDomainXYPlot(timeAxis);
            plot.setGap(10.0);
            for (int i = 0; i < dataset.length; i++) {
                NumberAxis valueAxis = new NumberAxis(label.get(i));
                XYPlot subplot = new XYPlot(dataset[i], timeAxis, valueAxis, null);
                subplot.setRangeAxisLocation(AxisLocation.TOP_OR_LEFT);
                
                XYLineAndShapeRenderer renderer = new XYLineAndShapeRenderer();
                renderer.setSeriesPaint(0, col[i % col.length]);
                renderer.setSeriesLinesVisible(0, y.get(i).isLine());
                renderer.setSeriesShapesVisible(0, y.get(i).isShape());
                renderer.setBaseToolTipGenerator(new StandardXYToolTipGenerator());
                subplot.setRenderer(renderer);
                plot.add(subplot, 1);
            }
            plot.setOrientation(PlotOrientation.VERTICAL);
            JFreeChart chart = new JFreeChart(title, JFreeChart.DEFAULT_TITLE_FONT, plot, true);
            return comp(chart);
        } else if (view == 1) {  // multi
            JFreeChart chart = ChartFactory.createTimeSeriesChart(title, "Date", label.get(0), dataset[0], true, true, false);
            XYPlot plot = (XYPlot) chart.getPlot();
            
            plot.setOrientation(PlotOrientation.VERTICAL);
            plot.getRangeAxis().setFixedDimension(15.0);
            plot.getRangeAxis().setLabelPaint(col[0]);


            for (int i = 0; i < dataset.length; i++) {
                NumberAxis axis = new NumberAxis(label.get(i));
                axis.setLabelPaint(col[i % col.length]);
                plot.setRangeAxis(i, axis);
                plot.setRangeAxisLocation(i, AxisLocation.BOTTOM_OR_LEFT);
                plot.setDataset(i, dataset[i]);
                plot.mapDatasetToRangeAxis(i, i);
//                XYItemRenderer rend = new StandardXYItemRenderer();
//                rend.setSeriesPaint(0, col[i % col.length]);
//                plot.setRenderer(i, rend);

                XYLineAndShapeRenderer renderer = new XYLineAndShapeRenderer();
//                renderer.setSeriesPaint(i, col[i % col.length]);
//                renderer.setSeriesLinesVisible(i, y.get(i).isLine());
//                renderer.setSeriesShapesVisible(i, y.get(i).isShape());
                renderer.setSeriesPaint(0, col[i % col.length]);
                renderer.setSeriesLinesVisible(0, y.get(i).isLine());
                renderer.setSeriesShapesVisible(0, y.get(i).isShape());
                renderer.setBaseToolTipGenerator(new StandardXYToolTipGenerator());
                plot.setRenderer(i, renderer);
            }
            
            return comp(chart);
        } else if (view == 2) {  // combined
            XYDataset dataset1 = createDatasetCombined(date, label, vals);
            JFreeChart chart = ChartFactory.createTimeSeriesChart(title, "Date", null, dataset1, true, false, false);
            XYPlot plot = (XYPlot) chart.getPlot();
            XYLineAndShapeRenderer renderer = new XYLineAndShapeRenderer();
            for (int i = 0; i < dataset.length; i++) {
                renderer.setSeriesLinesVisible(i, y.get(i).isLine());
                renderer.setSeriesShapesVisible(i, y.get(i).isShape());
            }
            renderer.setBaseToolTipGenerator(new StandardXYToolTipGenerator());
            plot.setRenderer(renderer);

            return comp(chart);
        } else {
            throw new IllegalArgumentException("Illegal view type");
        }
    }

//    public static JComponent createTSChart(String title, Date[] date, List label, List vals) {
//        XYDataset dataset = createDataset(date, label, vals);
//        JFreeChart chart = ChartFactory.createTimeSeriesChart(title, "Date", null, dataset, true, true, false);
//        return comp(chart);
//    }
    public static JComponent createLineChart(String title, Integer[] date, List label, List vals) {
        XYDataset dataset = createXYDataset(date, label, vals);
        JFreeChart chart = ChartFactory.createXYLineChart(title, "%", null, dataset, PlotOrientation.VERTICAL, true, true, false);
        return comp(chart);
    }

    public static JComponent createScatterChart(String title, String labelX, String labelY, Double[] x, Double[] y) {
        XYDataset dataset = createXYScatterDataset(x, y);
        JFreeChart chart = ChartFactory.createScatterPlot(title, labelX, labelY, dataset, PlotOrientation.VERTICAL, false, true, false);
        return comp(chart);
    }

    public static ESPToolPanel createESPTraces(String title, String dir, String var, String[] obs) throws Exception {
        ESPResultInfo info = load(dir);
        ESPTimeSeries init = getInitTimeSeries(info, dir, var);
        ArrayList output = new ArrayList();
        for (int i = 0; i < info.output.length; i++) {
            ESPTimeSeries ts = getOutputTimeSeries(info, dir, i, var);
            output.add(ts);
        }
        ArrayList forecasts = new ArrayList();
        for (int i = 0; i < info.output.length; i++) {
            ESPTimeSeries ts = getForecastTimeSeries(info, dir, i, var);
            forecasts.add(ts);
        }
        EnsembleData ed = new EnsembleData(var, init, forecasts, output, null);
        ESPToolPanel p = new ESPToolPanel(ed, obs);
        p.setResult(new File(dir, "result.csv").toString());
        return p;
    }

    private static ESPResultInfo load(String dir) throws IOException {
        File result = new File(dir, "result.csv");
        if (!result.exists()) {
            throw new IllegalArgumentException("Not found: " + result);
        }
        FileReader fr = new FileReader(result);
        CSProperties p = DataIO.properties(fr, "Result");
        fr.close();
        
        ESPResultInfo info = new ESPResultInfo();
        
        
//        // old
//        info.initstart = Conversions.convert(p.getInfo().get("initstart"), Date.class);
//        info.initend = Conversions.convert(p.getInfo().get("initend"), Date.class);
//        info.forecastend = Conversions.convert(p.getInfo().get("forecastend"), Date.class);
//        info.firsthistoricalyear = Conversions.convert(p.getInfo().get("firstyear"), Integer.class);
//        info.lasthistoricalyear = Conversions.convert(p.getInfo().get("lastyear"), Integer.class);

//         new 
        String esp_dates = p.getInfo().get(DataIO.KEY_ESP_DATES);
        String hist_years = p.getInfo().get(DataIO.KEY_HIST_YEARS);
        int y[] = Dates.parseHistoricalYears(hist_years);
        Date[] dates = Dates.parseESPDates(esp_dates);
        
        info.initstart = dates[0];
        info.initend = Dates.dayBefore(dates[1]); // init end is the day before forecast 
        info.forecastend = dates[2];
        info.firsthistoricalyear = y[0];
        info.lasthistoricalyear = y[1];
        
        Trace[] t = new Trace[info.lasthistoricalyear - info.firsthistoricalyear + 1];
        for (int i = info.firsthistoricalyear; i <= info.lasthistoricalyear; i++) {
            int idx = i - info.firsthistoricalyear;
            t[idx] = new Trace();
            t[idx].year = i;
            t[idx].file = p.getInfo().get("trace." + i);
        }
        info.output = t;
        return info;
    }

    private static ESPTimeSeries getInitTimeSeries(ESPResultInfo info, String dir, String var) throws IOException {
        String file = info.output[0].file;

//        File filename = new File(dir,file);
        File filename = new File(file);
        CSTable table = DataIO.table(filename, "efc");
        
        double[] values = DataIO.getColumnDoubleValuesInterval(info.initstart, info.initend, table, var, DataIO.DAILY);
        double dates[] = new double[values.length];

        ModelDateTime start = new ModelDateTime();
        start.setTime(info.initstart);
        ModelDateTime end = new ModelDateTime();
        end.setTime(info.initend);

        ModelDateTime current = new ModelDateTime();
        GregorianCalendar cal = new GregorianCalendar();
        cal.setTime(info.initstart);
        for (int i = 0; i < values.length; i++) {
            current.setTime(cal.getTime());
            dates[i] = current.getJulian();
            cal.add(Calendar.DATE, 1);
//            values[i] = (Double) table.getValueAt(i, col);
        }
        ESPTimeSeries ts = new ESPTimeSeries("init", dates, values, start, end, "", filename.toString(), "");
        return ts;
    }

    private static ESPTimeSeries getOutputTimeSeries(ESPResultInfo info, String dir, int trace, String var) throws IOException {
        int year = info.output[trace].year;
        String file = info.output[trace].file;

        File filename = new File(file);
//        File filename = new File(dir, file);
        CSTable table = DataIO.table(filename, "efc");

        Double v[] = DataIO.getColumnDoubleValues(table, var);
        Date d[] = DataIO.getColumnDateValues(table, "date");
        double values[] = new double[v.length];
        double dates[] = new double[v.length];

        ModelDateTime start = new ModelDateTime();
        start.setTime(info.initstart);

        ModelDateTime end = new ModelDateTime();
        end.setTime(info.forecastend);

        ModelDateTime current = new ModelDateTime();
        for (int i = 0; i < dates.length; i++) {
            current.setTime(d[i]);
            dates[i] = current.getJulian();
            values[i] = v[i].doubleValue();
        }
        return new ESPTimeSeries(Integer.toString(year), dates, values, start, end, "", filename.toString(), "");
    }

    private static ESPTimeSeries getForecastTimeSeries(ESPResultInfo info, String dir, int trace, String var) throws IOException {
        int year = info.output[trace].year;
        String file = info.output[trace].file;

        File filename = new File(file);
//        File filename = new File(dir, file);
        CSTable table = DataIO.table(filename, "efc");

        GregorianCalendar cal = new GregorianCalendar();
        cal.setTime(info.initend);
        cal.add(Calendar.DATE, 1);
 
        double[] values = DataIO.getColumnDoubleValuesInterval(cal.getTime(), info.forecastend, table, var, DataIO.DAILY);
        double dates[] = new double[values.length];

        ModelDateTime start = new ModelDateTime();
        start.setTime(info.initend);
        start.setJul2Greg(start.getJulian() + 1.0);

        ModelDateTime end = new ModelDateTime();
        end.setTime(info.forecastend);

        ModelDateTime current = new ModelDateTime();
        for (int i = 0; i < dates.length; i++) {
            current.setTime(cal.getTime());
            dates[i] = current.getJulian();
            cal.add(Calendar.DATE, 1);
        }

        return new ESPTimeSeries(Integer.toString(year), dates, values, start, end, "", filename.toString(), "");
    }

    public static class Trace {

        int year;
        String file;
    }

    public static class ESPResultInfo {

        Date initstart;
        Date initend;
        Date forecastend;
        int firsthistoricalyear;
        int lasthistoricalyear;
        Trace[] output;
    }
    
//    public static void main(String[] args) throws Exception {
//        UIManager.setLookAndFeel(UIManager.getSystemLookAndFeelClassName());
//        final String title = "\u20A2\u20A2\u20A2\u20A3\u20A4\u20A5\u20A6\u20A7\u20A8\u20A9\u20AA";
//        JFrame f = new JFrame("test");
//        f.setDefaultCloseOperation(JFrame.EXIT_ON_CLOSE);
//        final XYDataset dataset = createDataset();
//        final JFreeChart chart = createChart(dataset);
//        final ChartPanel chartPanel = new ChartPanel(chart);
//        chartPanel.setPreferredSize(new java.awt.Dimension(500, 270));
//        chartPanel.setMouseZoomable(true, false);
//        f.setContentPane(chartPanel);
//        f.pack();
//        f.setVisible(true);
//    }
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy