fi.smaa.jsmaa.gui.views.CriterionView.java Source code

Java tutorial

Introduction

Here is the source code for fi.smaa.jsmaa.gui.views.CriterionView.java

Source

/*
This file is part of JSMAA.
JSMAA is distributed from http://smaa.fi/.
    
(c) Tommi Tervonen, 2009-2010.
(c) Tommi Tervonen, Gert van Valkenhoef 2011.
(c) Tommi Tervonen, Gert van Valkenhoef, Joel Kuiper, Daan Reid 2012.
(c) Tommi Tervonen, Gert van Valkenhoef, Joel Kuiper, Daan Reid, Raymond Vermaas 2013.
    
JSMAA 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 3 of the License, or
(at your option) any later version.
    
JSMAA 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 JSMAA.  If not, see <http://www.gnu.org/licenses/>.
*/
package fi.smaa.jsmaa.gui.views;

import java.awt.Color;
import java.awt.Component;
import java.beans.PropertyChangeEvent;
import java.beans.PropertyChangeListener;

import javax.swing.BorderFactory;
import javax.swing.JComponent;
import javax.swing.JPanel;

import org.drugis.common.gui.LayoutUtil;
import org.drugis.common.gui.ViewBuilder;
import org.jfree.chart.ChartFactory;
import org.jfree.chart.ChartPanel;
import org.jfree.chart.JFreeChart;
import org.jfree.chart.axis.ValueAxis;
import org.jfree.chart.labels.StandardXYToolTipGenerator;
import org.jfree.chart.plot.PlotOrientation;
import org.jfree.chart.plot.XYPlot;
import org.jfree.chart.renderer.xy.XYLineAndShapeRenderer;
import org.jfree.data.Range;
import org.jfree.util.ShapeUtilities;

import com.jgoodies.binding.PresentationModel;
import com.jgoodies.binding.adapter.BasicComponentFactory;
import com.jgoodies.binding.value.ValueHolder;
import com.jgoodies.forms.builder.PanelBuilder;
import com.jgoodies.forms.layout.CellConstraints;
import com.jgoodies.forms.layout.FormLayout;

import fi.smaa.jsmaa.gui.IntervalFormat;
import fi.smaa.jsmaa.gui.components.MeasurementPanel;
import fi.smaa.jsmaa.gui.components.MeasurementPanel.MeasurementType;
import fi.smaa.jsmaa.gui.components.ValueFunctionMouseListener;
import fi.smaa.jsmaa.gui.components.ValueFunctionPointsPanel;
import fi.smaa.jsmaa.gui.jfreechart.UtilityFunctionDataset;
import fi.smaa.jsmaa.gui.presentation.ImpactMatrixPresentationModel;
import fi.smaa.jsmaa.gui.presentation.OrdinalCriterionMeasurementsPM;
import fi.smaa.jsmaa.model.CardinalCriterion;
import fi.smaa.jsmaa.model.CardinalMeasurement;
import fi.smaa.jsmaa.model.Criterion;
import fi.smaa.jsmaa.model.IndependentMeasurements;
import fi.smaa.jsmaa.model.OrdinalCriterion;
import fi.smaa.jsmaa.model.OutrankingCriterion;
import fi.smaa.jsmaa.model.SMAAModel;
import fi.smaa.jsmaa.model.SMAATRIModel;
import fi.smaa.jsmaa.model.ScaleCriterion;

public class CriterionView implements ViewBuilder {
    protected Criterion criterion;
    protected SMAAModel model;
    private Component parent;

    public CriterionView(Criterion crit, SMAAModel model, Component parent) {
        this.criterion = crit;
        this.model = model;
        this.parent = parent;
    }

    public JComponent buildPanel() {
        FormLayout layout = new FormLayout("pref:grow:fill", "p, 3dlu, p, 3dlu, p");

        PanelBuilder builder = new PanelBuilder(layout);
        builder.setBorder(BorderFactory.createEmptyBorder());
        CellConstraints cc = new CellConstraints();

        builder.add(buildOverviewPart(), cc.xy(1, 1));

        int row = 3;
        if (criterion instanceof OutrankingCriterion) {
            LayoutUtil.addRow(layout);
            builder.addSeparator("Thresholds", cc.xy(1, 3));
            LayoutUtil.addRow(layout);
            builder.add(buildThresholdsPart(), cc.xy(1, 5));
            row = 7;
        }
        builder.addSeparator("Measurements", cc.xy(1, row));

        JComponent measPanel = null;
        if (criterion instanceof CardinalCriterion && model.getMeasurements() instanceof IndependentMeasurements) {
            ImpactMatrixPresentationModel iModel = new ImpactMatrixPresentationModel(
                    (IndependentMeasurements) model.getMeasurements());
            measPanel = new CardinalCriterionMeasurementsView((CardinalCriterion) criterion, iModel).buildPanel();
        } else if (criterion instanceof OrdinalCriterion
                && model.getMeasurements() instanceof IndependentMeasurements) {
            OrdinalCriterionMeasurementsPM pm = new OrdinalCriterionMeasurementsPM((OrdinalCriterion) criterion,
                    (IndependentMeasurements) model.getMeasurements());
            measPanel = new OrdinalCriterionMeasurementsView(pm).buildPanel();
        }
        builder.add(measPanel, cc.xy(1, row + 2));

        row += 4;
        if (criterion instanceof ScaleCriterion) {
            LayoutUtil.addRow(layout);
            builder.addSeparator("Value function", cc.xy(1, row));
            LayoutUtil.addRow(layout);

            final JPanel chartPanel = buildValueFunctionChartPanel((ScaleCriterion) criterion);

            builder.add(chartPanel, cc.xy(1, row + 2));
            row += 4;
        }

        if (model instanceof SMAATRIModel) {
            LayoutUtil.addRow(layout);
            builder.addSeparator("Profiles (category boundaries)", cc.xy(1, row));
            LayoutUtil.addRow(layout);
            builder.add(new ProfilesView((OutrankingCriterion) criterion, (SMAATRIModel) model).buildPanel(),
                    cc.xy(1, row + 2));
        }

        return builder.getPanel();
    }

    private JPanel buildValueFunctionChartPanel(ScaleCriterion criterion) {
        UtilityFunctionDataset dataset = new UtilityFunctionDataset(criterion);

        JFreeChart chart = ChartFactory.createXYLineChart("", "x", "v(x)", dataset, PlotOrientation.VERTICAL, false,
                true, true);

        final XYPlot plot = chart.getXYPlot();
        XYLineAndShapeRenderer renderer = new XYLineAndShapeRenderer();
        plot.setRenderer(0, renderer);
        renderer.setSeriesPaint(0, Color.black);
        renderer.setSeriesShape(0, ShapeUtilities.createDiamond(3.0f));
        renderer.setBaseToolTipGenerator(new StandardXYToolTipGenerator());

        ValueAxis rAxis = plot.getRangeAxis();
        rAxis.setAutoRange(false);
        rAxis.setRange(new Range(-0.03, 1.03));
        ValueAxis dAxis = plot.getDomainAxis();
        dAxis.setLowerMargin(0.03);
        dAxis.setUpperMargin(0.03);

        ChartPanel chartPanel = new ChartPanel(chart, false, true, true, false, true);
        chartPanel.addChartMouseListener(new ValueFunctionMouseListener(chartPanel, criterion, parent));

        chartPanel.setDomainZoomable(false);
        chartPanel.setRangeZoomable(false);
        chartPanel.setDisplayToolTips(true);
        chartPanel.setToolTipText("Click to add/remove partial value function points");
        chartPanel.setMouseWheelEnabled(false);
        chartPanel.setMouseZoomable(false);

        plot.setDomainCrosshairLockedOnData(false);
        plot.setRangeCrosshairLockedOnData(false);
        plot.setDomainCrosshairVisible(true);
        plot.setRangeCrosshairVisible(true);

        FormLayout layout = new FormLayout("left:pref", "p, 3dlu, p");

        PanelBuilder builder = new PanelBuilder(layout);
        CellConstraints cc = new CellConstraints();
        builder.add(chartPanel, cc.xy(1, 1));
        builder.add(new ValueFunctionPointsPanel(criterion), cc.xy(1, 3));

        return builder.getPanel();
    }

    private JComponent buildOverviewPart() {
        FormLayout layout = new FormLayout("right:pref, 3dlu, left:pref", "p, 3dlu, p");

        PanelBuilder builder = new PanelBuilder(layout);
        CellConstraints cc = new CellConstraints();

        PresentationModel<Criterion> pm = new PresentationModel<Criterion>(criterion);

        builder.addLabel("Name:", cc.xy(1, 1));
        builder.add(BasicComponentFactory.createLabel(pm.getModel(Criterion.PROPERTY_NAME)), cc.xy(3, 1));
        builder.addLabel("Type:", cc.xy(1, 3));
        builder.add(BasicComponentFactory.createLabel(pm.getModel(Criterion.PROPERTY_TYPELABEL)), cc.xy(3, 3));

        int row = 5;
        if (criterion instanceof ScaleCriterion) {
            LayoutUtil.addRow(layout);
            ScaleCriterion cardCrit = (ScaleCriterion) criterion;
            PresentationModel<ScaleCriterion> pmc = new PresentationModel<ScaleCriterion>(cardCrit);
            builder.addLabel("Scale:", cc.xy(1, row));
            builder.add(BasicComponentFactory.createLabel(pmc.getModel(ScaleCriterion.PROPERTY_SCALE),
                    new IntervalFormat()), cc.xy(3, row));
            row += 2;
        }

        if (criterion instanceof CardinalCriterion) {
            CardinalCriterion cardCrit = (CardinalCriterion) criterion;
            PresentationModel<CardinalCriterion> pmc = new PresentationModel<CardinalCriterion>(cardCrit);
            LayoutUtil.addRow(layout);
            builder.addLabel("Ascending:", cc.xy(1, row));
            builder.add(BasicComponentFactory.createCheckBox(pmc.getModel(ScaleCriterion.PROPERTY_ASCENDING), null),
                    cc.xy(3, row));
        }

        return builder.getPanel();
    }

    private JComponent buildThresholdsPart() {
        FormLayout layout = new FormLayout("right:pref, 3dlu, left:pref", "p, 3dlu, p");

        PanelBuilder builder = new PanelBuilder(layout);
        CellConstraints cc = new CellConstraints();

        final OutrankingCriterion outrCrit = (OutrankingCriterion) criterion;

        ValueHolder indifHolder = new ValueHolder(outrCrit.getIndifMeasurement());
        indifHolder.addPropertyChangeListener(new PropertyChangeListener() {
            public void propertyChange(PropertyChangeEvent evt) {
                outrCrit.setIndifMeasurement((CardinalMeasurement) evt.getNewValue());
            }
        });
        ValueHolder prefHolder = new ValueHolder(outrCrit.getPrefMeasurement());
        prefHolder.addPropertyChangeListener(new PropertyChangeListener() {
            public void propertyChange(PropertyChangeEvent evt) {
                outrCrit.setPrefMeasurement((CardinalMeasurement) evt.getNewValue());
            }
        });

        MeasurementPanel.MeasurementType[] measVals = new MeasurementPanel.MeasurementType[] {
                MeasurementType.EXACT, MeasurementType.INTERVAL, MeasurementType.GAUSSIAN };
        JPanel indifPanel = new MeasurementPanel(indifHolder, measVals);
        JPanel prefPanel = new MeasurementPanel(prefHolder, measVals);

        builder.addLabel("Indifference:", cc.xy(1, 1));
        builder.add(indifPanel, cc.xy(3, 1));

        builder.addLabel("Preference:", cc.xy(1, 3));
        builder.add(prefPanel, cc.xy(3, 3));

        return builder.getPanel();
    }
}