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

edu.cmu.tetradapp.editor.BayesImNodeEditingTableObs Maven / Gradle / Ivy

There is a newer version: 7.6.6
Show newest version
///////////////////////////////////////////////////////////////////////////////
// For information as to what this class does, see the Javadoc, below.       //
// Copyright (C) 1998, 1999, 2000, 2001, 2002, 2003, 2004, 2005, 2006,       //
// 2007, 2008, 2009, 2010, 2014, 2015, 2022 by Peter Spirtes, Richard        //
// Scheines, Joseph Ramsey, and Clark Glymour.                               //
//                                                                           //
// This program 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 2 of the License, or         //
// (at your option) any later version.                                       //
//                                                                           //
// This program 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 this program; if not, write to the Free Software               //
// Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA  02111-1307  USA //
///////////////////////////////////////////////////////////////////////////////

package edu.cmu.tetradapp.editor;

import edu.cmu.tetrad.bayes.BayesIm;
import edu.cmu.tetrad.bayes.BayesPm;
import edu.cmu.tetrad.bayes.MlBayesImObs;
import edu.cmu.tetrad.graph.Node;
import edu.cmu.tetrad.graph.NodeType;
import edu.cmu.tetrad.util.JOptionUtils;
import edu.cmu.tetrad.util.NumberFormatUtil;
import org.jetbrains.annotations.NotNull;

import javax.swing.*;
import javax.swing.table.AbstractTableModel;
import javax.swing.table.TableCellEditor;
import javax.swing.table.TableColumn;
import javax.swing.table.TableModel;
import java.awt.*;
import java.awt.event.MouseAdapter;
import java.awt.event.MouseEvent;
import java.beans.PropertyChangeListener;
import java.beans.PropertyChangeSupport;
import java.text.NumberFormat;
import java.util.ArrayList;

/**
 * This is the JTable which displays the getModel parameter set (an Model).
 *
 * @author josephramsey
 */

////////////////////////////////////////////////////////
// display and edit a JPD
////////////////////////////////////////////////////////

class BayesImNodeEditingTableObs extends JTable {
    private int focusRow;
    private int focusCol;

    /**
     * Constructs a new editing table from a given editing table model.
     */
    public BayesImNodeEditingTableObs(BayesIm bayesIm) {
        if (bayesIm == null) {
            throw new NullPointerException();
        }

        Model model = new Model(bayesIm, this);
        model.addPropertyChangeListener(evt -> {
            if ("modelChanged".equals(evt.getPropertyName())) {
                firePropertyChange("modelChanged", null, null);
            }
        });
        setModel(model);

        ////////////////////////////////////////////////////////////////

        setDefaultEditor(Number.class, new NumberCellEditor());
        setDefaultRenderer(Number.class, new NumberCellRenderer());
        getTableHeader().setReorderingAllowed(false);
        getTableHeader().setResizingAllowed(true);
        setAutoResizeMode(JTable.AUTO_RESIZE_OFF);
        setCellSelectionEnabled(true);

        ListSelectionModel rowSelectionModel = getSelectionModel();

        rowSelectionModel.addListSelectionListener(e -> {
            ListSelectionModel m = (ListSelectionModel) (e.getSource());
            setFocusRow(m.getAnchorSelectionIndex());
        });

        ListSelectionModel columnSelectionModel = getColumnModel()
                .getSelectionModel();

        columnSelectionModel.addListSelectionListener(
                e -> setFocusColumn());

        addMouseListener(new MouseAdapter() {
            public void mousePressed(MouseEvent e) {
                if (SwingUtilities.isRightMouseButton(e) || e.isControlDown()) {
                    showPopup(e);
                }
            }
        });

        setFocusRow(0);
        setFocusColumn();
    }

    public void createDefaultColumnsFromModel() {
        super.createDefaultColumnsFromModel();

        if (getModel() instanceof Model) {
            FontMetrics fontMetrics = getFontMetrics(getFont());
            Model model = (Model) getModel();

            for (int i = 0; i < model.getColumnCount(); i++) {
                TableColumn column = getColumnModel().getColumn(i);
                String columnName = model.getColumnName(i);
                int currentWidth = column.getPreferredWidth();

                if (columnName != null) {
                    int minimumWidth = fontMetrics.stringWidth(columnName) + 8;

                    if (minimumWidth > currentWidth) {
                        column.setPreferredWidth(minimumWidth);
                    }
                }
            }
        }
    }

    private void showPopup(MouseEvent e) {
        JPopupMenu popup = new JPopupMenu();

        JMenuItem randomizeEntireTable =
                new JMenuItem("Randomize entire table");
        JMenuItem clearEntireTable = new JMenuItem("Clear entire table");

        randomizeEntireTable.addActionListener(e1 -> {
            /*if (existsCompleteRow(bayesIm, nodeIndex)) {*/
            int ret = JOptionPane.showConfirmDialog(
                    JOptionUtils.centeringComp(),
                    "This will modify all values in the table. " +
                            "Continue?", "Warning",
                    JOptionPane.YES_NO_OPTION);

            if (ret == JOptionPane.NO_OPTION) {
                return;
            }
            /*}*/

            BayesImNodeEditingTableObs editingTable =
                    BayesImNodeEditingTableObs.this;
            TableCellEditor cellEditor = editingTable.getCellEditor();

            if (cellEditor != null) {
                cellEditor.cancelCellEditing();
            }

            // randomize the jpd
            //getBayesIm().getJPD().createRandomCellTable();
            getBayesIm().createRandomCellTable();

            getEditingTableModel().fireTableDataChanged();
            firePropertyChange("modelChanged", null, null);
        });

        clearEntireTable.addActionListener(e12 -> {
            //if (existsCompleteRow(bayesIm, nodeIndex)) {
            int ret = JOptionPane.showConfirmDialog(
                    JOptionUtils.centeringComp(),
                    "This will delete all values in the table. " +
                            "Continue?", "Warning",
                    JOptionPane.YES_NO_OPTION);

            if (ret == JOptionPane.NO_OPTION) {
                return;
            }
            //}

            BayesImNodeEditingTableObs editingTable =
                    BayesImNodeEditingTableObs.this;
            TableCellEditor cellEditor = editingTable.getCellEditor();

            if (cellEditor != null) {
                cellEditor.cancelCellEditing();
            }

            // clear the jpd
            getBayesIm().getJPD().clearCellTable();

            getEditingTableModel().fireTableDataChanged();
            firePropertyChange("modelChanged", null, null);
        });

        popup.add(randomizeEntireTable);
        popup.add(clearEntireTable);

        popup.show((Component) e.getSource(), e.getX(), e.getY());
    }

    public void setModel(@NotNull TableModel model) {
        super.setModel(model);
    }

    /**
     * Sets the focus row to the anchor row currently being selected.
     */
    private void setFocusRow(int row) {
        if (row == -1) {
            return;
        }

        Model editingTableModel = (Model) getModel();
        int failedRow = editingTableModel.getFailedRow();

        if (failedRow != -1) {
            row = failedRow;
            editingTableModel.resetFailedRow();
        }

        this.focusRow = row;

        if (this.focusRow < getRowCount()) {
            setRowSelectionInterval(this.focusRow, this.focusRow);
            editCellAt(this.focusRow, this.focusCol);
        }
    }

    /**
     * Sets the focus column to the anchor column currently being selected.
     */
    private void setFocusColumn() {
        Model editingTableModel = (Model) getModel();
        int failedCol = editingTableModel.getFailedCol();

        if (failedCol != -1) {
            editingTableModel.resetFailedCol();
        }

        // always focus on the last column of each row (the joint probability
        // of the combination of variable values denoted by the other columns
        // in that row)
        this.focusCol = getColumnCount() - 1;

        setColumnSelectionInterval(this.focusCol, this.focusCol);
        editCellAt(this.focusRow, this.focusCol);
    }

    private Model getEditingTableModel() {
        return (Model) getModel();
    }

    private MlBayesImObs getBayesIm() {
        return getEditingTableModel().getBayesIm();
    }

    //////////////////////////////////////////
    // The abstract table model of the jpd
    //////////////////////////////////////////

    static final class Model extends AbstractTableModel {

        /**
         * The BayesIm being edited.
         */
        private final MlBayesImObs bayesIm;
        private final java.util.List obsNodes = new ArrayList<>();
        private int failedRow = -1;
        private int failedCol = -1;
        private PropertyChangeSupport pcs;

        /////////////////////////////////////////////////////////////
        // construct a new editing table model for a given bayesIm
        //
        public Model(BayesIm bayesIm, JComponent messageAnchor) {
            if (bayesIm == null) {
                throw new NullPointerException("Bayes IM must not be null.");
            }

            if (messageAnchor == null) {
                throw new NullPointerException(
                        "Message anchor must not be null.");
            }

            // cast the bayesIm to MlBayesImObs
            this.bayesIm = (MlBayesImObs) bayesIm;

            for (int i = 0; i < bayesIm.getNumNodes(); i++) {
                Node nodeO = bayesIm.getNode(i);
                if (nodeO.getNodeType() == NodeType.MEASURED) {
                    this.obsNodes.add(nodeO);
                }
            }

            // This does not work: it gives a different ordering of the nodes
			/*
			obsNodes = bayesIm.getMeasuredNodes();
			*/

        }

        /**
         * @return the name of the given column.
         */
        public String getColumnName(int col) {
            if (col < this.obsNodes.size()) {
                return this.obsNodes.get(col).getName();
            } else if (col == this.obsNodes.size()) {
                return "Probability";  // last column is the joint probability
            } else {
                return null;
            }
        }

        //////////////////////////////////////////////////////////////////*
        // number of rows in the table.
        public int getRowCount() {
            return getBayesIm().getNumRows();
        }

        //////////////////////////////////////////////////////////////////
        // number of columns in the table
        // (number of variables in the bayeIm plus one for the probability
        // value)
        public int getColumnCount() {
            return this.obsNodes.size() + 1;
        }

        ////////////////////////////////////////////////////////////////////
        // Returns the value of the table at the given row and column. 
        // The type of value returned depends on the column.
        // If there are n variables in the bayesIm, the first n columns
        // have String values, representing the combination of node values.
        // The last column is a Double representing the probability of
        // this combination of node values.
        //
        public Object getValueAt(int tableRow, int tableCol) {

            if (tableCol < this.obsNodes.size()) {
                int categoryIndex = getBayesIm().getRowValues(tableRow)[tableCol];
                BayesPm bayesPm = getBayesIm().getBayesPm();
                return bayesPm.getCategory(this.obsNodes.get(tableCol),
                        categoryIndex);
            } else if (tableCol == this.obsNodes.size()) {
                return getBayesIm().getProbability(tableRow);
            } else {
                return null;
            }
        }

        ////////////////////////////////////////////////////////////////////
        // Determine whether a cell is in the column range to allow for editing
        // (only the last column (representing the prob value) can be edited.)
        public boolean isCellEditable(int row, int col) {
            return (col >= this.obsNodes.size());
        }

        ////////////////////////////////////////////////////////////////////
        // Set the value of the cell at (row, col) to 'aValue'.
        // Perform some error checking to make sure the probabilities add up.
        //
        public void setValueAt(Object aValue, int row, int col) {
            if ("".equals(aValue) || aValue == null) {    // cell is cleared
                getBayesIm().setProbability(row, Double.NaN);
                fireTableRowsUpdated(row, row);
                getPcs().firePropertyChange("modelChanged", null, null);
                return;
            }

            try {
                NumberFormat nf = NumberFormatUtil.getInstance().getNumberFormat();

                double probability = Double.parseDouble((String) aValue);
//                probability = Double.parseDouble(nf.format(probability));
                double sum = sumProb(row) + probability;

                double oldProbability = getBayesIm().getProbability(row);

                if (!Double.isNaN(oldProbability)) { // there is an old value 
                    oldProbability = Double.parseDouble(nf.format(oldProbability));
                }

                if (probability == oldProbability) { // value is retained
                    return;
                }

                if (probabilityOutOfRange(probability)) {
                    JOptionPane.showMessageDialog(JOptionUtils.centeringComp(),
                            "Probabilities must be in range [0.0, 1.0].");
                    this.failedRow = row;
                    this.failedCol = col;
                } else if (sum > 1.00005) {
                    JOptionPane.showMessageDialog(JOptionUtils.centeringComp(),
                            "Sum of probabilities in the column must not exceed 1.0.");
                    this.failedRow = row;
                    this.failedCol = col;
                } else {
                    getBayesIm().setProbability(row, probability);

                    // all the rows are filled in and the total is less than 1
                    if ((numNanRows() == 0) && (sum < 0.99995)) {
                        //if (sumInRow < 0.99995 || sumInRow > 1.00005) {
                        // emptyRow(row);   // too harsh

                        // only two rows in the table: filling in one row will
                        // cause the other row to be filled in
                        if (getBayesIm().getNumRows() == 2) {
                            getBayesIm().setProbability(row, probability);
                            fillInSingleRemainingRow();
                            fireTableRowsUpdated(row, row);
                            getPcs().firePropertyChange("modelChanged", null,
                                    null);
                        } else {
                            // set the probability back to before editing
                            getBayesIm().setProbability(row, oldProbability);
                            JOptionPane.showMessageDialog(
                                    JOptionUtils.centeringComp(),
                                    "Probabilities in the column must sum up to 1.0.\n"
                                            + "Leave one row (or two) blank while working.");
                            this.failedRow = row;
                            this.failedCol = col;
                        }

                    } else // things are ok
                    {
                        fillInSingleRemainingRow();
                        fillInZerosIfSumIsOne();
                        fireTableRowsUpdated(row, row);
                        getPcs().firePropertyChange("modelChanged", null,
                                null);
                    }
                }
            } catch (NumberFormatException e) {  // not a number
                e.printStackTrace();
                JOptionPane.showMessageDialog(JOptionUtils.centeringComp(),
                        "Could not interpret '" + aValue + "'");
                this.failedRow = row;
                this.failedCol = col;
            }
        }

        public void addPropertyChangeListener(PropertyChangeListener l) {
            getPcs().addPropertyChangeListener(l);
        }

        private PropertyChangeSupport getPcs() {
            if (this.pcs == null) {
                this.pcs = new PropertyChangeSupport(this);
            }
            return this.pcs;
        }

        // fill in the last remaining row so that the rows sum up to 1
        private void fillInSingleRemainingRow() {
            int leftOverRow = uniqueNanRow();

            if (leftOverRow != -1) {
                double difference = 1.0 - sumProb(leftOverRow);
                getBayesIm().setProbability(leftOverRow, difference);
            }
        }

        // make all remaining rows 0 if the filled in rows already sum up to 1
        private void fillInZerosIfSumIsOne() {
            double sum = sumProb(-1);  // sum all the rows without skipping

            if (sum > 0.9995 && sum < 1.0005) {
                int numRows = getBayesIm().getNumRows();

                for (int i = 0; i < numRows; i++) {
                    double probability = getBayesIm().getProbability(i);

                    if (Double.isNaN(probability)) {
                        getBayesIm().setProbability(i, 0.0);
                    }
                }
            }
        }

        private boolean probabilityOutOfRange(double value) {
            return value < 0.0 || value > 1.0;
        }

        // one empty row only
        private int uniqueNanRow() {
            int numNanRows = 0;
            int lastNanRow = -1;

            for (int i = 0; i < getBayesIm().getNumRows(); i++) {
                double probability = getBayesIm().getProbability(i);
                if (Double.isNaN(probability)) {
                    numNanRows++;
                    lastNanRow = i;
                }
            }

            return numNanRows == 1 ? lastNanRow : -1;
        }

        // number of empty rows
        private int numNanRows() {
            int numNanRows = 0;

            for (int i = 0; i < getBayesIm().getNumRows(); i++) {
                double probability = getBayesIm().getProbability(i);
                if (Double.isNaN(probability)) {
                    numNanRows++;
                }
            }

            return numNanRows;
        }


        // sum of all probability entries in the table except the row rowToSkip
        // To sum every row, make rowToSkip an impossible value, e.g. -1
        private double sumProb(int rowToSkip) {
            double sum = 0.0;

            for (int i = 0; i < getBayesIm().getNumRows(); i++) {
                double probability = getBayesIm().getProbability(i);

                if (i != rowToSkip && !Double.isNaN(probability)) {

                    NumberFormat nf = NumberFormatUtil.getInstance().getNumberFormat();
                    probability = Double.parseDouble(nf.format(probability));

                    sum += probability;
                }
            }

            return sum;
        }

        ///////////////////////////////////////
        // last column is a number, the others general objects
        // This is used to determine the formatting of the cell
        public Class getColumnClass(int col) {
            return col == getColumnCount() - 1 ? Number.class : Object.class;
        }

        // cast the bayesIm to MlBayesImObs
        public MlBayesImObs getBayesIm() {
            return this.bayesIm;
        }

        public int getFailedRow() {
            return this.failedRow;
        }

        public int getFailedCol() {
            return this.failedCol;
        }

        public void resetFailedRow() {
            this.failedRow = -1;
        }

        public void resetFailedCol() {
            this.failedCol = -1;
        }

    }
}










© 2015 - 2025 Weber Informatics LLC | Privacy Policy