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

weka.classifiers.bayes.net.GUI Maven / Gradle / Ivy

/*
 *    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., 675 Mass Ave, Cambridge, MA 02139, USA.
 */

/*
 *    GUI.java
 *    Copyright (C) 2007 University of Waikato, Hamilton, New Zealand
 *
 */
package weka.classifiers.bayes.net;

import weka.classifiers.bayes.net.MarginCalculator.JunctionTreeNode;
import weka.core.FastVector;
import weka.core.Instances;
import weka.core.OptionHandler;
import weka.core.SerializedObject;
import weka.core.Utils;
import weka.core.converters.AbstractFileLoader;
import weka.core.converters.AbstractFileSaver;
import weka.core.converters.ArffSaver;
import weka.core.converters.ConverterUtils;
import weka.gui.ConverterFileChooser;
import weka.gui.ExtensionFileFilter;
import weka.gui.GenericObjectEditor;
import weka.gui.LookAndFeel;
import weka.gui.PropertyDialog;
import weka.gui.graphvisualizer.BIFFormatException;
import weka.gui.graphvisualizer.BIFParser;
import weka.gui.graphvisualizer.GraphNode;
import weka.gui.graphvisualizer.HierarchicalBCEngine;
import weka.gui.graphvisualizer.LayoutCompleteEvent;
import weka.gui.graphvisualizer.LayoutCompleteEventListener;
import weka.gui.graphvisualizer.LayoutEngine;
import weka.gui.visualize.PrintablePanel;

import java.awt.BorderLayout;
import java.awt.Color;
import java.awt.Container;
import java.awt.Dimension;
import java.awt.Font;
import java.awt.FontMetrics;
import java.awt.Frame;
import java.awt.Graphics;
import java.awt.Graphics2D;
import java.awt.GridBagConstraints;
import java.awt.GridBagLayout;
import java.awt.GridLayout;
import java.awt.Insets;
import java.awt.Rectangle;
import java.awt.RenderingHints;
import java.awt.event.ActionEvent;
import java.awt.event.ActionListener;
import java.awt.event.MouseAdapter;
import java.awt.event.MouseEvent;
import java.awt.event.MouseMotionAdapter;
import java.awt.event.WindowAdapter;
import java.awt.event.WindowEvent;
import java.awt.image.BufferedImage;
import java.awt.print.PageFormat;
import java.awt.print.Printable;
import java.awt.print.PrinterException;
import java.awt.print.PrinterJob;
import java.beans.PropertyEditor;
import java.io.File;
import java.io.FileReader;
import java.io.FileWriter;
import java.io.IOException;
import java.util.Random;

import javax.swing.AbstractAction;
import javax.swing.Action;
import javax.swing.BorderFactory;
import javax.swing.ImageIcon;
import javax.swing.JButton;
import javax.swing.JCheckBox;
import javax.swing.JCheckBoxMenuItem;
import javax.swing.JDialog;
import javax.swing.JFileChooser;
import javax.swing.JFrame;
import javax.swing.JLabel;
import javax.swing.JMenu;
import javax.swing.JMenuBar;
import javax.swing.JMenuItem;
import javax.swing.JOptionPane;
import javax.swing.JPanel;
import javax.swing.JPopupMenu;
import javax.swing.JScrollPane;
import javax.swing.JTable;
import javax.swing.JTextField;
import javax.swing.JToolBar;
import javax.swing.KeyStroke;
import javax.swing.table.AbstractTableModel;

/**
 * GUI interface to Bayesian Networks. Allows editing Bayesian networks
 * on screen and provides GUI interface to various Bayesian network facilities
 * in Weka, including random network generation, data set generation and
 * Bayesion network inference.
 * 
 * @author Remco Bouckaert ([email protected])
 * @version $Revision: 4905 $
 */
public class GUI extends JPanel implements LayoutCompleteEventListener {

	/** for serialization */
	private static final long serialVersionUID = -2038911085935515624L;

	/** The current LayoutEngine */
	protected LayoutEngine m_layoutEngine;

	/** Panel actually displaying the graph */
	protected GraphPanel m_GraphPanel;

	/** Container of Bayesian network */
	EditableBayesNet m_BayesNet = new EditableBayesNet(true);

	/** String containing file name storing current network */
	protected String m_sFileName = "";
	/** used for calculating marginals in Bayesian netwowrks */
	MarginCalculator m_marginCalculator = null;

	/**
	 * used for calculating marginals in Bayesian netwowrks when evidence is
	 * present
	 */
	MarginCalculator m_marginCalculatorWithEvidence = null;

	/** flag indicating whether marginal distributions of each of the nodes 
	 * should be shown in display.
	 */
	boolean m_bViewMargins = false;
	boolean m_bViewCliques = false;
	
	/** The menu bar */
	private JMenuBar m_menuBar;

	/** data selected from file. Used to train a Bayesian network on */
	Instances m_Instances = null;

	/** Text field for specifying zoom */
	final JTextField m_jTfZoom;
	/** toolbar containing buttons at top of window */
	final JToolBar m_jTbTools;
	/** status bar at bottom of window */
	final JLabel m_jStatusBar;
	/** TextField for node's width */
	private final JTextField m_jTfNodeWidth = new JTextField(3);
	/** TextField for nodes height */
	private final JTextField m_jTfNodeHeight = new JTextField(3);
	/** this contains the m_GraphPanel GraphPanel */
	JScrollPane m_jScrollPane;

	/** path for icons */
	private final String ICONPATH = "weka/classifiers/bayes/net/icons/";

	/** current zoom value */
	private double m_fScale = 1;

	/** standard width of node */
	private int m_nNodeHeight = 2 * getFontMetrics(getFont()).getHeight();
	/** standard height of node */
	final static int DEFAULT_NODE_WIDTH = 50;
	private int m_nNodeWidth = DEFAULT_NODE_WIDTH;
	/** width of node, allowing for some padding */	
	final static int PADDING = 10;
	private int m_nPaddedNodeWidth = DEFAULT_NODE_WIDTH + PADDING;


	/** used when using zoomIn and zoomOut buttons */
	private int [] m_nZoomPercents = { 10, 25, 50, 75, 100, 125, 150, 175, 200, 225, 250, 275, 300, 350, 400, 450, 500,
			550, 600, 650, 700, 800, 900, 999 };

	/** actions triggered by GUI events */
	Action a_new = new ActionNew();

	Action a_quit = new ActionQuit();
	Action a_save = new ActionSave();
	ActionExport a_export = new ActionExport();
	ActionPrint a_print = new ActionPrint();
	Action a_load = new ActionLoad();
	Action a_zoomin = new ActionZoomIn();
	Action a_zoomout = new ActionZoomOut();
	Action a_layout = new ActionLayout();

	Action a_saveas = new ActionSaveAs();

	Action a_viewtoolbar = new ActionViewToolbar();

	Action a_viewstatusbar = new ActionViewStatusbar();

	Action a_networkgenerator = new ActionGenerateNetwork();

	Action a_datagenerator = new ActionGenerateData();

	Action a_datasetter = new ActionSetData();

	Action a_learn = new ActionLearn();
	Action a_learnCPT = new ActionLearnCPT();

	Action a_help = new ActionHelp();

	Action a_about = new ActionAbout();

	ActionAddNode a_addnode = new ActionAddNode();

	Action a_delnode = new ActionDeleteNode();
	Action a_cutnode = new ActionCutNode();
	Action a_copynode = new ActionCopyNode();
	Action a_pastenode = new ActionPasteNode();
	Action a_selectall = new ActionSelectAll();

	Action a_addarc = new ActionAddArc();

	Action a_delarc = new ActionDeleteArc();

	Action a_undo = new ActionUndo();

	Action a_redo= new ActionRedo();
	
	Action a_alignleft = new ActionAlignLeft();
	Action a_alignright = new ActionAlignRight();
	Action a_aligntop = new ActionAlignTop();
	Action a_alignbottom = new ActionAlignBottom();
	Action a_centerhorizontal = new ActionCenterHorizontal();
	Action a_centervertical = new ActionCenterVertical();
	Action a_spacehorizontal = new ActionSpaceHorizontal();
	Action a_spacevertical = new ActionSpaceVertical();
	
	/** node currently selected through right clicking */
	int m_nCurrentNode = -1;
	/** selection of nodes */
	Selection m_Selection = new Selection();
	/** selection rectangle drawn through dragging with left mouse button */
	Rectangle m_nSelectedRect = null;

	
	class Selection {
		FastVector m_selected;
		public Selection() {
			m_selected = new FastVector();
		} // c'tor
		public FastVector getSelected() {return m_selected;}
		void updateGUI() {
			if (m_selected.size() > 0) {
				a_cutnode.setEnabled(true);
				a_copynode.setEnabled(true);
			} else {
				a_cutnode.setEnabled(false);
				a_copynode.setEnabled(false);
			}
			if (m_selected.size() > 1) {
				a_alignleft.setEnabled(true);
				a_alignright.setEnabled(true);
				a_aligntop.setEnabled(true);
				a_alignbottom.setEnabled(true);
				a_centerhorizontal.setEnabled(true);
				a_centervertical.setEnabled(true);
				a_spacehorizontal.setEnabled(true);
				a_spacevertical.setEnabled(true);
			} else {
				a_alignleft.setEnabled(false);
				a_alignright.setEnabled(false);
				a_aligntop.setEnabled(false);
				a_alignbottom.setEnabled(false);
				a_centerhorizontal.setEnabled(false);
				a_centervertical.setEnabled(false);
				a_spacehorizontal.setEnabled(false);
				a_spacevertical.setEnabled(false);
			}
		} // updateGUI
		
		public void addToSelection(int nNode) {
			for (int iNode = 0; iNode < m_selected.size(); iNode++) {
				if (nNode == (Integer) m_selected.elementAt(iNode)) {
					return;
				}
			}
			m_selected.addElement(nNode);
			updateGUI();
		} // addToSelection
		
		public void addToSelection(int [] iNodes) {
			for (int iNode = 0; iNode < iNodes.length; iNode++) {
				addToSelection(iNodes[iNode]);
			}
			updateGUI();
		} // addToSelection
		
		public void addToSelection(Rectangle selectedRect) {
			for (int iNode = 0; iNode < m_BayesNet.getNrOfNodes(); iNode++) {
				if (contains(selectedRect, iNode)) {
					addToSelection(iNode);
				}
			}
		} // addToSelection
		
		public void selectAll() {
			m_selected.removeAllElements();
			for (int iNode = 0; iNode < m_BayesNet.getNrOfNodes(); iNode++) {
				m_selected.addElement(iNode);
			}
			updateGUI();
		} // selectAll

		boolean contains(Rectangle rect, int iNode) {
			return rect.intersects((m_BayesNet.getPositionX(iNode)) * m_fScale,
					(m_BayesNet.getPositionY(iNode)) * m_fScale,
					m_nPaddedNodeWidth * m_fScale, m_nNodeHeight * m_fScale);
		} // contains
		
		public void removeFromSelection(int nNode) {
			for (int iNode = 0; iNode < m_selected.size(); iNode++) {
				if (nNode == (Integer) m_selected.elementAt(iNode)) {
					m_selected.removeElementAt(iNode);
				}
			}
			updateGUI();
		} // removeFromSelection

		public void toggleSelection(int nNode) {
			for (int iNode = 0; iNode < m_selected.size(); iNode++) {
				if (nNode == (Integer) m_selected.elementAt(iNode)) {
					m_selected.removeElementAt(iNode);
					updateGUI();
					return;
				}
			}
			addToSelection(nNode);
		} // toggleSelection

		public void toggleSelection(Rectangle selectedRect)	{
			for (int iNode = 0; iNode < m_BayesNet.getNrOfNodes(); iNode++) {
				if (contains(selectedRect, iNode)) {
					toggleSelection(iNode);
				}
			}
		} // toggleSelection
		
		public void clear() {
			m_selected.removeAllElements();
			updateGUI();
		}
		
		public void draw(Graphics g) {
			if (m_selected.size() == 0) {
				return;
			}

			for (int iNode = 0; iNode < m_selected.size(); iNode++) {
				int nNode = (Integer) m_selected.elementAt(iNode);
				int nPosX = m_BayesNet.getPositionX(nNode);
				int nPosY = m_BayesNet.getPositionY(nNode);
				g.setColor(Color.BLACK);
				int nXRC = nPosX + m_nPaddedNodeWidth - m_nNodeWidth - (m_nPaddedNodeWidth - m_nNodeWidth) / 2;
				int nYRC = nPosY;
				int d = 5;
				g.fillRect(nXRC, nYRC, d, d); 
				g.fillRect(nXRC, nYRC + m_nNodeHeight, d, d); 
				g.fillRect(nXRC + m_nNodeWidth, nYRC, d, d); 
				g.fillRect(nXRC + m_nNodeWidth, nYRC + m_nNodeHeight, d, d); 
			}
		} // draw
	} // Selection

	ClipBoard m_clipboard = new ClipBoard();
	
	class ClipBoard {
		String m_sText = null;
		public ClipBoard() {
			if (a_pastenode != null) {
				a_pastenode.setEnabled(false);
			}
		}
		public boolean hasText() {return m_sText != null;}
		public String getText() { 
			return m_sText;
		}
		public void setText(String sText) {
			m_sText = sText;
			a_pastenode.setEnabled(true);
		}
	} // class ClipBoard
	
	/** Base class used for definining actions
	 * with a name, tool tip text, possibly an icon and accelerator key. 
	 * */
	class MyAction extends AbstractAction {
		/** for serialization */
		private static final long serialVersionUID = -2038911111935517L;

		public MyAction(String sName, String sToolTipText, String sIcon, String sAcceleratorKey) {
			super(sName);
			//setToolTipText(sToolTipText);
			putValue(Action.SHORT_DESCRIPTION, sToolTipText);
			putValue(Action.LONG_DESCRIPTION, sToolTipText);
			if (sAcceleratorKey.length() > 0) {
				KeyStroke keyStroke = KeyStroke.getKeyStroke(sAcceleratorKey);
				putValue(Action.ACCELERATOR_KEY, keyStroke);
			}
			putValue(Action.MNEMONIC_KEY, (int) sName.charAt(0));
			java.net.URL tempURL = ClassLoader.getSystemResource(ICONPATH + sIcon + ".png");
			if (tempURL != null) {
				putValue(Action.SMALL_ICON, new ImageIcon(tempURL));
			} else {
				putValue(Action.SMALL_ICON, new ImageIcon(new BufferedImage(20,20, BufferedImage.TYPE_4BYTE_ABGR)));
				//System.err.println(ICONPATH + sIcon + ".png not found for weka.gui.graphvisualizer.Graph");
			}
		} // c'tor

		/* Place holder. Should be implemented by derived classes.
		 *  (non-Javadoc)
		 * @see java.awt.event.ActionListener#actionPerformed(java.awt.event.ActionEvent)
		 */
		public void actionPerformed(ActionEvent ae) {}
	} // class MyAction

	class ActionGenerateNetwork extends MyAction {
		/** for serialization */
		private static final long serialVersionUID = -2038911085935517L;

		public ActionGenerateNetwork() {
			super("Generate Network", "Generate Random Bayesian Network", "generate.network", "ctrl N");
		} // c'tor

		int m_nNrOfNodes = 10;

		int m_nNrOfArcs = 15;

		int m_nCardinality = 2;

		int m_nSeed = 123;

		JDialog dlg = null;

		public void actionPerformed(ActionEvent ae) {
			if (dlg == null) {
				dlg = new JDialog();
				dlg.setTitle("Generate Random Bayesian Network Options");

				final JLabel jLbNrOfNodes = new JLabel("Nr of nodes");
				final JTextField jTfNrOfNodes = new JTextField(3);
				jTfNrOfNodes.setHorizontalAlignment(JTextField.CENTER);
				jTfNrOfNodes.setText("" + m_nNrOfNodes);
				final JLabel jLbNrOfArcs = new JLabel("Nr of arcs");
				final JTextField jTfNrOfArcs = new JTextField(3);
				jTfNrOfArcs.setHorizontalAlignment(JTextField.CENTER);
				jTfNrOfArcs.setText("" + m_nNrOfArcs);
				final JLabel jLbCardinality = new JLabel("Cardinality");
				final JTextField jTfCardinality = new JTextField(3);
				jTfCardinality.setHorizontalAlignment(JTextField.CENTER);
				jTfCardinality.setText("" + m_nCardinality);
				final JLabel jLbSeed = new JLabel("Random seed");
				final JTextField jTfSeed = new JTextField(3);
				jTfSeed.setHorizontalAlignment(JTextField.CENTER);
				jTfSeed.setText("" + m_nSeed);

				JButton jBtGo;
				jBtGo = new JButton("Generate Network");

				jBtGo.addActionListener(new ActionListener() {
					public void actionPerformed(ActionEvent ae) {
						try {
							BayesNetGenerator generator = new BayesNetGenerator();
							m_BayesNet = generator;
							m_BayesNet.clearUndoStack();
							
							String[] options = new String[8];
							options[0] = "-N";
							options[1] = "" + jTfNrOfNodes.getText();
							options[2] = "-A";
							options[3] = "" + jTfNrOfArcs.getText();
							options[4] = "-C";
							options[5] = "" + jTfCardinality.getText();
							options[6] = "-S";
							options[7] = "" + jTfSeed.getText();
							generator.setOptions(options);
							generator.generateRandomNetwork();
							// Convert to EditableBayesNet
							// This ensures the getOptions() called by GenericObjectEditor to get the correct result.
							BIFReader bifReader = new BIFReader();
							bifReader.processString(m_BayesNet.toXMLBIF03());
							m_BayesNet = new EditableBayesNet(bifReader);

							updateStatus();
							layoutGraph();
							a_datagenerator.setEnabled(true);
							m_Instances = null;;
							a_learn.setEnabled(false);
							a_learnCPT.setEnabled(false);

							dlg.setVisible(false);
						} catch (Exception e) {
							e.printStackTrace();
						}
					}
				});

				JButton jBtCancel;
				jBtCancel = new JButton("Cancel");
				jBtCancel.setMnemonic('C');
				jBtCancel.addActionListener(new ActionListener() {
					public void actionPerformed(ActionEvent ae) {
						dlg.setVisible(false);
					}
				});
				GridBagConstraints gbc = new GridBagConstraints();
				dlg.setLayout(new GridBagLayout());

				Container c = new Container();
				c.setLayout(new GridBagLayout());
				gbc.gridwidth = 2;
				gbc.insets = new Insets(8, 0, 0, 0);
				gbc.anchor = GridBagConstraints.NORTHWEST;
				gbc.gridwidth = GridBagConstraints.RELATIVE;
				gbc.fill = GridBagConstraints.HORIZONTAL;
				c.add(jLbNrOfNodes, gbc);
				gbc.gridwidth = GridBagConstraints.REMAINDER;
				c.add(jTfNrOfNodes, gbc);
				gbc.gridwidth = GridBagConstraints.RELATIVE;
				c.add(jLbNrOfArcs, gbc);
				gbc.gridwidth = GridBagConstraints.REMAINDER;
				c.add(jTfNrOfArcs, gbc);
				gbc.gridwidth = GridBagConstraints.RELATIVE;
				c.add(jLbCardinality, gbc);
				gbc.gridwidth = GridBagConstraints.REMAINDER;
				c.add(jTfCardinality, gbc);
				gbc.gridwidth = GridBagConstraints.RELATIVE;
				c.add(jLbSeed, gbc);
				gbc.gridwidth = GridBagConstraints.REMAINDER;
				c.add(jTfSeed, gbc);

				gbc.fill = GridBagConstraints.HORIZONTAL;
				dlg.add(c, gbc);
				dlg.add(jBtGo);
				gbc.gridwidth = GridBagConstraints.REMAINDER;
				dlg.add(jBtCancel);
			}
			dlg.pack();
			dlg.setLocation(100, 100);
			dlg.setVisible(true);
			dlg.setSize(dlg.getPreferredSize());
			dlg.setVisible(false);
			dlg.setVisible(true);
			dlg.repaint();
		} // actionPerformed
	} // class ActionGenerate

	class ActionGenerateData extends MyAction {
		/** for serialization */
		private static final long serialVersionUID = -2038911085935516L;

		public ActionGenerateData() {
			super("Generate Data", "Generate Random Instances from Network", "generate.data", "ctrl D");
		} // c'tor

		int m_nNrOfInstances = 100;

		int m_nSeed = 1234;

		String m_sFile = "";

		JDialog dlg = null;

		public void actionPerformed(ActionEvent ae) {
			if (dlg == null) {
				dlg = new JDialog();
				dlg.setTitle("Generate Random Data Options");

				final JLabel jLbNrOfInstances = new JLabel("Nr of instances");
				final JTextField jTfNrOfInstances = new JTextField(3);
				jTfNrOfInstances.setHorizontalAlignment(JTextField.CENTER);
				jTfNrOfInstances.setText("" + m_nNrOfInstances);
				final JLabel jLbSeed = new JLabel("Random seed");
				final JTextField jTfSeed = new JTextField(3);
				jTfSeed.setHorizontalAlignment(JTextField.CENTER);
				jTfSeed.setText("" + m_nSeed);
				final JLabel jLbFile = new JLabel("Output file (optional)");
				final JTextField jTfFile = new JTextField(12);
				jTfFile.setHorizontalAlignment(JTextField.CENTER);
				jTfFile.setText(m_sFile);

				JButton jBtGo;
				jBtGo = new JButton("Generate Data");

				jBtGo.addActionListener(new ActionListener() {
					public void actionPerformed(ActionEvent ae) {
						try {
							String tmpfilename = "tmp.bif.file.xml";
							BayesNetGenerator generator = new BayesNetGenerator();
							String[] options = new String[4];
							options[0] = "-M";
							options[1] = "" + jTfNrOfInstances.getText();
							options[2] = "-F";
							options[3] = tmpfilename;
							FileWriter outfile = new FileWriter(tmpfilename);
							StringBuffer text = new StringBuffer();
							if (m_marginCalculator == null) {
								m_marginCalculator = new MarginCalculator();
								m_marginCalculator.calcMargins(m_BayesNet);
							}
							text.append(m_marginCalculator.toXMLBIF03());
							outfile.write(text.toString());
							outfile.close();

							generator.setOptions(options);
							generator.generateRandomNetwork();
							generator.generateInstances();
							m_Instances = generator.m_Instances;
							a_learn.setEnabled(true);
							a_learnCPT.setEnabled(true);

							m_sFile = jTfFile.getText();
							if (m_sFile != null && !m_sFile.equals("")) {
								AbstractFileSaver saver = ConverterUtils.getSaverForFile(m_sFile);
								// no idea what the format is, so let's save it as ARFF file
								if (saver == null)
									saver = new ArffSaver();
								saver.setFile(new File(m_sFile));
								saver.setInstances(m_Instances);
								saver.writeBatch();
							}

						} catch (Exception e) {
							e.printStackTrace();
						}
						dlg.setVisible(false);
					}
				});

				JButton jBtFile = new JButton("Browse");
				jBtFile.addActionListener(new ActionListener() {
					public void actionPerformed(ActionEvent ae) {
						ConverterFileChooser fc = new ConverterFileChooser(System.getProperty("user.dir"));
						fc.setDialogTitle("Save Instances As");
						int rval = fc.showSaveDialog(GUI.this);

						if (rval == JFileChooser.APPROVE_OPTION) {
							String filename = fc.getSelectedFile().toString();
							jTfFile.setText(filename);
						}
						dlg.setVisible(true);
					}
				});
				JButton jBtCancel;
				jBtCancel = new JButton("Cancel");
				jBtCancel.setMnemonic('C');
				jBtCancel.addActionListener(new ActionListener() {
					public void actionPerformed(ActionEvent ae) {
						dlg.setVisible(false);
					}
				});
				GridBagConstraints gbc = new GridBagConstraints();
				dlg.setLayout(new GridBagLayout());

				Container c = new Container();
				c.setLayout(new GridBagLayout());
				gbc.gridwidth = 2;
				gbc.insets = new Insets(8, 0, 0, 0);
				gbc.anchor = GridBagConstraints.NORTHWEST;
				gbc.gridwidth = GridBagConstraints.RELATIVE;
				gbc.fill = GridBagConstraints.HORIZONTAL;
				c.add(jLbNrOfInstances, gbc);
				gbc.gridwidth = GridBagConstraints.REMAINDER;
				c.add(jTfNrOfInstances, gbc);
				gbc.gridwidth = GridBagConstraints.RELATIVE;
				c.add(jLbSeed, gbc);
				gbc.gridwidth = GridBagConstraints.REMAINDER;
				c.add(jTfSeed, gbc);
				gbc.gridwidth = GridBagConstraints.RELATIVE;
				c.add(jLbFile, gbc);
				gbc.gridwidth = GridBagConstraints.REMAINDER;
				c.add(jTfFile, gbc);
				gbc.gridwidth = GridBagConstraints.REMAINDER;
				c.add(jBtFile, gbc);

				gbc.fill = GridBagConstraints.HORIZONTAL;
				dlg.add(c, gbc);
				dlg.add(jBtGo);
				gbc.gridwidth = GridBagConstraints.REMAINDER;
				dlg.add(jBtCancel);
			}
			dlg.setLocation(100, 100);
			dlg.setVisible(true);
			dlg.setSize(dlg.getPreferredSize());
			dlg.setVisible(false);
			dlg.setVisible(true);
			dlg.repaint();

		} // actionPerformed
	} // class ActionGenerateData

	class ActionLearn extends MyAction {
		/** for serialization */
		private static final long serialVersionUID = -2038911085935516L;

		public ActionLearn() {
			super("Learn Network", "Learn Bayesian Network", "learn", "ctrl L");
			setEnabled(false);
		} // c'tor

		JDialog dlg = null;

		public void actionPerformed(ActionEvent ae) {
			if (dlg == null) {
				dlg = new JDialog();
				dlg.setTitle("Learn Bayesian Network");

				final JButton jBtOptions = new JButton("Options");
				jBtOptions.addActionListener(new ActionListener() {
					public void actionPerformed(ActionEvent ae) {
						//m_BayesNet = new EditableBayesNet();
						try {
							GenericObjectEditor.registerEditors();
							GenericObjectEditor ce = new GenericObjectEditor(true);
							ce.setClassType(weka.classifiers.Classifier.class);
							ce.setValue(m_BayesNet);

							PropertyDialog pd;
							if (PropertyDialog.getParentDialog(GUI.this) != null)
								pd = new PropertyDialog(PropertyDialog.getParentDialog(GUI.this), ce, 100, 100);
							else
								pd = new PropertyDialog(PropertyDialog.getParentFrame(GUI.this), ce, 100, 100);
							pd.addWindowListener(new WindowAdapter() {
								public void windowClosing(WindowEvent e) {
									PropertyEditor pe = ((PropertyDialog) e.getSource()).getEditor();
									Object c = (Object) pe.getValue();
									String options = "";
									if (c instanceof OptionHandler) {
										options = Utils.joinOptions(((OptionHandler) c).getOptions());
										try {
											m_BayesNet.setOptions(((OptionHandler) c).getOptions());
										} catch (Exception e2) {
											e2.printStackTrace();
										}
									}
									System.out.println(c.getClass().getName() + " " + options);
									System.exit(0);
								}
							});
							pd.setVisible(true);
						} catch (Exception ex) {
							ex.printStackTrace();
							System.err.println(ex.getMessage());
						}
						m_BayesNet.clearUndoStack();
						a_undo.setEnabled(false);
						a_redo.setEnabled(false);
					}
				});

				final JTextField jTfOptions = new JTextField(40);
				jTfOptions.setHorizontalAlignment(JTextField.CENTER);
				jTfOptions.setText("" + Utils.joinOptions(m_BayesNet.getOptions()));

				JButton jBtGo;
				jBtGo = new JButton("Learn");

				jBtGo.addActionListener(new ActionListener() {
					public void actionPerformed(ActionEvent ae) {
						try {
							m_BayesNet.buildClassifier(m_Instances);
							layoutGraph();
							updateStatus();
							m_BayesNet.clearUndoStack();

							dlg.setVisible(false);
						} catch (Exception e) {
							e.printStackTrace();
						}
						dlg.setVisible(false);
					}
				});

				JButton jBtCancel;
				jBtCancel = new JButton("Cancel");
				jBtCancel.setMnemonic('C');
				jBtCancel.addActionListener(new ActionListener() {
					public void actionPerformed(ActionEvent ae) {
						dlg.setVisible(false);
					}
				});
				GridBagConstraints gbc = new GridBagConstraints();
				dlg.setLayout(new GridBagLayout());

				Container c = new Container();
				c.setLayout(new GridBagLayout());
				gbc.gridwidth = 2;
				gbc.insets = new Insets(8, 0, 0, 0);
				gbc.anchor = GridBagConstraints.NORTHWEST;
				gbc.gridwidth = GridBagConstraints.RELATIVE;
				gbc.fill = GridBagConstraints.HORIZONTAL;
				c.add(jBtOptions, gbc);
				gbc.gridwidth = GridBagConstraints.REMAINDER;
				c.add(jTfOptions, gbc);

				gbc.fill = GridBagConstraints.HORIZONTAL;
				dlg.add(c, gbc);
				dlg.add(jBtGo);
				gbc.gridwidth = GridBagConstraints.REMAINDER;
				dlg.add(jBtCancel);
			}
			dlg.setLocation(100, 100);
			dlg.setVisible(true);
			dlg.setSize(dlg.getPreferredSize());
			dlg.setVisible(false);
			dlg.setVisible(true);
			dlg.repaint();
		} // actionPerformed
	} // class ActionLearn

	class ActionLearnCPT extends MyAction {
		/** for serialization */
		private static final long serialVersionUID = -2022211085935516L;

		public ActionLearnCPT() {
			super("Learn CPT", "Learn conditional probability tables", "learncpt", "");
			setEnabled(false);
		} // c'tor

		public void actionPerformed(ActionEvent ae) {
			if (m_Instances == null) {
				JOptionPane.showMessageDialog(null, "Select instances to learn from first (menu Tools/Set Data)");
				return;
			}
			try {
				m_BayesNet.setData(m_Instances);
			} catch (Exception e) {
				JOptionPane.showMessageDialog(null, "Data set is not compatible with network.\n"+e.getMessage() + "\nChoose other instances (menu Tools/Set Data)");
				return;
			}
			try {
				m_BayesNet.estimateCPTs();
				m_BayesNet.clearUndoStack();
			} catch (Exception e) {
				e.printStackTrace();
			}
			updateStatus();
		} // actionPerformed
	} // class ActionLearnCPT

	class ActionSetData extends MyAction {
		/** for serialization */
		private static final long serialVersionUID = -2038911085935519L;

		public ActionSetData() {
			super("Set Data", "Set Data File", "setdata", "ctrl A");
		} // c'tor

		public void actionPerformed(ActionEvent ae) {
			ConverterFileChooser fc = new ConverterFileChooser(System.getProperty("user.dir"));
			fc.setDialogTitle("Set Data File");
			int rval = fc.showOpenDialog(GUI.this);

			if (rval == JFileChooser.APPROVE_OPTION) {
				AbstractFileLoader loader = fc.getLoader();
				try {
					if (loader != null)
					  m_Instances = loader.getDataSet();
					if (m_Instances.classIndex() == -1)
					  m_Instances.setClassIndex(m_Instances.numAttributes() - 1);
					a_learn.setEnabled(true);
					a_learnCPT.setEnabled(true);
					repaint();
				} catch (Exception e) {
					e.printStackTrace();
				}
			}
		}
	} // class ActionSetData

	class ActionUndo extends MyAction {
		/** for serialization */
		private static final long serialVersionUID = -3038910085935519L;

		public ActionUndo() {
			super("Undo", "Undo", "undo", "ctrl Z");
			setEnabled(false);
		} // c'tor

		public boolean isEnabled() {
			return m_BayesNet.canUndo();
		}

		public void actionPerformed(ActionEvent ae) {
			String sMsg = m_BayesNet.undo();
			m_jStatusBar.setText("Undo action performed: " + sMsg);
			//if (!sMsg.equals("")) {
			//	JOptionPane.showMessageDialog(null, sMsg, "Undo action successful", JOptionPane.INFORMATION_MESSAGE);
			//}
			a_redo.setEnabled(m_BayesNet.canRedo());
			a_undo.setEnabled(m_BayesNet.canUndo());
			m_Selection.clear();
			updateStatus();
			repaint();
		}
	} // ActionUndo
	
	class ActionRedo extends MyAction {
		/** for serialization */
		private static final long serialVersionUID = -4038910085935519L;

		public ActionRedo() {
			super("Redo", "Redo", "redo", "ctrl Y");
			setEnabled(false);
		} // c'tor

		public boolean isEnabled() {
			return m_BayesNet.canRedo();
		}

		public void actionPerformed(ActionEvent ae) {
			String sMsg = m_BayesNet.redo();
			m_jStatusBar.setText("Redo action performed: " + sMsg);
			//if (!sMsg.equals("")) {
			//	JOptionPane.showMessageDialog(null, sMsg, "Redo action successful", JOptionPane.INFORMATION_MESSAGE);
			//}
			m_Selection.clear();
			updateStatus();
			repaint();
		}
	} // ActionRedo

	class ActionAddNode extends MyAction {
		/** for serialization */
		private static final long serialVersionUID = -2038910085935519L;

		public ActionAddNode() {
			super("Add Node", "Add Node", "addnode", "");
		} // c'tor

		JDialog dlg = null;

		JTextField jTfName = new JTextField(20);

		JTextField jTfCard = new JTextField(3);

		int m_X = Integer.MAX_VALUE;
		int m_Y;
		public void addNode(int nX, int nY) {
			m_X = nX;
			m_Y = nY;
			addNode();
		} // addNode
		
		void addNode() {
			if (dlg == null) {
				dlg = new JDialog();
				dlg.setTitle("Add node");
				JLabel jLbName = new JLabel("Name");
				jTfName.setHorizontalAlignment(JTextField.CENTER);
				JLabel jLbCard = new JLabel("Cardinality");
				jTfCard.setHorizontalAlignment(JTextField.CENTER);
				jTfCard.setText("2");

				JButton jBtCancel;
				jBtCancel = new JButton("Cancel");
				jBtCancel.setMnemonic('C');
				jBtCancel.addActionListener(new ActionListener() {
					public void actionPerformed(ActionEvent ae) {
						dlg.setVisible(false);
					}
				});
				JButton jBtOk = new JButton("Ok");
				jBtOk.setMnemonic('O');
				jBtOk.addActionListener(new ActionListener() {
					public void actionPerformed(ActionEvent ae) {
						String sName = jTfName.getText();
						if (sName.length() <= 0) {
							JOptionPane.showMessageDialog(null, "Name should have at least one character");
							return;
						}
						int nCard = new Integer(jTfCard.getText()).intValue();
						if (nCard <= 1) {
							JOptionPane.showMessageDialog(null, "Cardinality should be larger than 1");
							return;
						}
						try {
							if (m_X < Integer.MAX_VALUE) {
								m_BayesNet.addNode(sName, nCard, m_X, m_Y);
							} else {
								m_BayesNet.addNode(sName, nCard);
							}
						    m_jStatusBar.setText(m_BayesNet.lastActionMsg());
							a_undo.setEnabled(true);
							a_redo.setEnabled(false);
							//GraphNode n = new GraphNode("id" + m_nodes.size(), sName);
							//n.probs = m_BayesNet.getDistribution(sName);
							//n.outcomes = m_BayesNet.getValues(sName);
							//n.x = 100 + m_nodes.size() * 10;
							//n.y = 100 + m_nodes.size() * 10;
							//m_nodes.addElement(n);
						} catch (Exception e) {
							e.printStackTrace();
						}
						repaint();
						dlg.setVisible(false);
					}
				});
				dlg.setLayout(new GridLayout(3, 2, 10, 10));
				dlg.add(jLbName);
				dlg.add(jTfName);
				dlg.add(jLbCard);
				dlg.add(jTfCard);
				dlg.add(jBtOk);
				dlg.add(jBtCancel);
				dlg.setSize(dlg.getPreferredSize());
			}
			jTfName.setText("Node" + (m_BayesNet.getNrOfNodes() + 1));
			dlg.setVisible(true);
		} // addNode
		
		public void actionPerformed(ActionEvent ae) {
			m_X = Integer.MAX_VALUE;
			addNode();
		}
	} // class ActionAddNode

	class ActionDeleteNode extends MyAction {
		/** for serialization */
		private static final long serialVersionUID = -2038912085935519L;

		public ActionDeleteNode() {
			super("Delete Node", "Delete Node", "delnode", "DELETE");
		} // c'tor

		public void actionPerformed(ActionEvent ae) {
			if (m_Selection.getSelected().size() > 0) {
				m_BayesNet.deleteSelection(m_Selection.getSelected());
			    m_jStatusBar.setText(m_BayesNet.lastActionMsg());
				m_Selection.clear();
				updateStatus();
				repaint();
			} else {
				String[] options = new String[m_BayesNet.getNrOfNodes()];
				for (int i = 0; i < options.length; i++) {
					options[i] = m_BayesNet.getNodeName(i);
				}
				String sResult = (String) JOptionPane.showInputDialog(null, "Select node to delete", "Nodes", 0, null,
						options, options[0]);
				if (sResult != null && !sResult.equals("")) {
					int iNode = m_BayesNet.getNode2(sResult);
					deleteNode(iNode);
				}
			}
		}
	} // class ActionDeleteNode

	class ActionCopyNode extends MyAction {
		/** for serialization */
		private static final long serialVersionUID = -2038732085935519L;

		public ActionCopyNode() {
			super("Copy", "Copy Nodes", "copy", "ctrl C");
		} // c'tor
		
		public ActionCopyNode(String sName, String sToolTipText, String sIcon, String sAcceleratorKey) {
			super(sName, sToolTipText, sIcon, sAcceleratorKey);
		} // c'rot
		
		public void actionPerformed(ActionEvent ae) {
			copy();
		}
		
		public void copy() {
			String sXML = m_BayesNet.toXMLBIF03(m_Selection.getSelected());
			m_clipboard.setText(sXML);
		} // copy
	} // class ActionCopyNode

	class ActionCutNode extends ActionCopyNode {
		/** for serialization */
		private static final long serialVersionUID = -2038822085935519L;

		public ActionCutNode() {
			super("Cut", "Cut Nodes", "cut", "ctrl X");
		} // c'tor

		public void actionPerformed(ActionEvent ae) {
			copy();
			m_BayesNet.deleteSelection(m_Selection.getSelected());
		    m_jStatusBar.setText(m_BayesNet.lastActionMsg());
			m_Selection.clear();
			a_undo.setEnabled(true);
			a_redo.setEnabled(false);
			repaint();
		}
	} // class ActionCutNode

	class ActionPasteNode extends MyAction {
		/** for serialization */
		private static final long serialVersionUID = -2038732085935519L;

		public ActionPasteNode() {
			super("Paste", "Paste Nodes", "paste", "ctrl V");
		} // c'tor

		public void actionPerformed(ActionEvent ae) {
			try {
				m_BayesNet.paste(m_clipboard.getText());
				updateStatus();
			    m_jStatusBar.setText(m_BayesNet.lastActionMsg());
			} catch (Exception e) {
				e.printStackTrace();
			}
		}
		public boolean isEnabled() {
			return m_clipboard.hasText();
		}
	} // class ActionPasteNode

	class ActionSelectAll extends MyAction {
		/** for serialization */
		private static final long serialVersionUID = -2038642085935519L;

		public ActionSelectAll() {
			super("Select All", "Select All Nodes", "selectall", "ctrl A");
		} // c'tor

		public void actionPerformed(ActionEvent ae) {
			m_Selection.selectAll();
			repaint();
		}
	} // class ActionSelectAll

	class ActionExport extends MyAction {
		boolean m_bIsExporting = false;		
		/** for serialization */
		private static final long serialVersionUID = -3027642085935519L;

		public ActionExport() {
			super("Export", "Export to graphics file", "export", "");
		} // c'tor

		public void actionPerformed(ActionEvent ae) {
			m_bIsExporting = true;
			m_GraphPanel.saveComponent();
			m_bIsExporting = false;
			repaint();
		}
		public boolean isExporting() {return m_bIsExporting;}
	} // class ActionExport

	class ActionAlignLeft extends MyAction {
		/** for serialization */
		private static final long serialVersionUID = -3138642085935519L;

		public ActionAlignLeft() {
			super("Align Left", "Align Left", "alignleft", "");
		} // c'tor

		public void actionPerformed(ActionEvent ae) {
			m_BayesNet.alignLeft(m_Selection.getSelected());
		    m_jStatusBar.setText(m_BayesNet.lastActionMsg());
			a_undo.setEnabled(true);
			a_redo.setEnabled(false);
			repaint();
		}
	} // class ActionAlignLeft
	
	class ActionAlignRight extends MyAction {
		/** for serialization */
		private static final long serialVersionUID = -4238642085935519L;

		public ActionAlignRight() {
			super("Align Right", "Align Right", "alignright", "");
		} // c'tor

		public void actionPerformed(ActionEvent ae) {
			m_BayesNet.alignRight(m_Selection.getSelected());
		    m_jStatusBar.setText(m_BayesNet.lastActionMsg());
			a_undo.setEnabled(true);
			a_redo.setEnabled(false);
			repaint();
		}
	} // class ActionAlignRight

	class ActionAlignTop extends MyAction {
		/** for serialization */
		private static final long serialVersionUID = -5338642085935519L;

		public ActionAlignTop() {
			super("Align Top", "Align Top", "aligntop", "");
		} // c'tor

		public void actionPerformed(ActionEvent ae) {
			m_BayesNet.alignTop(m_Selection.getSelected());
		    m_jStatusBar.setText(m_BayesNet.lastActionMsg());
			a_undo.setEnabled(true);
			a_redo.setEnabled(false);
			repaint();
		}
	} // class ActionAlignTop

	class ActionAlignBottom extends MyAction {
		/** for serialization */
		private static final long serialVersionUID = -6438642085935519L;

		public ActionAlignBottom() {
			super("Align Bottom", "Align Bottom", "alignbottom", "");
		} // c'tor

		public void actionPerformed(ActionEvent ae) {
			m_BayesNet.alignBottom(m_Selection.getSelected());
		    m_jStatusBar.setText(m_BayesNet.lastActionMsg());
			a_undo.setEnabled(true);
			a_redo.setEnabled(false);
			repaint();
		}
	} // class ActionAlignBottom

	class ActionCenterHorizontal extends MyAction {
		/** for serialization */
		private static final long serialVersionUID = -7538642085935519L;

		public ActionCenterHorizontal() {
			super("Center Horizontal", "Center Horizontal", "centerhorizontal", "");
		} // c'tor

		public void actionPerformed(ActionEvent ae) {
			m_BayesNet.centerHorizontal(m_Selection.getSelected());
		    m_jStatusBar.setText(m_BayesNet.lastActionMsg());
			a_undo.setEnabled(true);
			a_redo.setEnabled(false);
			repaint();
		}
	} // class ActionCenterHorizontal

	class ActionCenterVertical extends MyAction {
		/** for serialization */
		private static final long serialVersionUID = -8638642085935519L;

		public ActionCenterVertical() {
			super("Center Vertical", "Center Vertical", "centervertical", "");
		} // c'tor

		public void actionPerformed(ActionEvent ae) {
			m_BayesNet.centerVertical(m_Selection.getSelected());
		    m_jStatusBar.setText(m_BayesNet.lastActionMsg());
			a_undo.setEnabled(true);
			a_redo.setEnabled(false);
			repaint();
		}
	} // class ActionCenterVertical

	class ActionSpaceHorizontal extends MyAction {
		/** for serialization */
		private static final long serialVersionUID = -9738642085935519L;

		public ActionSpaceHorizontal() {
			super("Space Horizontal", "Space Horizontal", "spacehorizontal", "");
		} // c'tor

		public void actionPerformed(ActionEvent ae) {
			m_BayesNet.spaceHorizontal(m_Selection.getSelected());
		    m_jStatusBar.setText(m_BayesNet.lastActionMsg());
			a_undo.setEnabled(true);
			a_redo.setEnabled(false);
			repaint();
		}
	} // class ActionSpaceHorizontal

	class ActionSpaceVertical extends MyAction {
		/** for serialization */
		private static final long serialVersionUID = -838642085935519L;

		public ActionSpaceVertical() {
			super("Space Vertical", "Space Vertical", "spacevertical", "");
		} // c'tor

		public void actionPerformed(ActionEvent ae) {
			m_BayesNet.spaceVertical(m_Selection.getSelected());
		    m_jStatusBar.setText(m_BayesNet.lastActionMsg());
			a_undo.setEnabled(true);
			a_redo.setEnabled(false);
			repaint();
		}
	} // class ActionSpaceVertical

	class ActionAddArc extends MyAction {
		/** for serialization */
		private static final long serialVersionUID = -2038913085935519L;

		public ActionAddArc() {
			super("Add Arc", "Add Arc", "addarc", "");
		} // c'tor

		public void actionPerformed(ActionEvent ae) {
			try {
				String[] options = new String[m_BayesNet.getNrOfNodes()];
				for (int i = 0; i < options.length; i++) {
					options[i] = m_BayesNet.getNodeName(i);
				}
				String sChild = (String) JOptionPane.showInputDialog(null, "Select child node", "Nodes", 0, null,
						options, options[0]);
				if (sChild == null || sChild.equals("")) {
					return;
				}
				int iChild = m_BayesNet.getNode(sChild);
				addArcInto(iChild);
			} catch (Exception e) {
				e.printStackTrace();
			}
		}
	} // class ActionAddArc

	class ActionDeleteArc extends MyAction {
		/** for serialization */
		private static final long serialVersionUID = -2038914085935519L;

		public ActionDeleteArc() {
			super("Delete Arc", "Delete Arc", "delarc", "");
		} // c'tor

		public void actionPerformed(ActionEvent ae) {
			int nEdges = 0;
			for (int iNode = 0; iNode < m_BayesNet.getNrOfNodes(); iNode++) {
				nEdges += m_BayesNet.getNrOfParents(iNode);
			}
			String[] options = new String[nEdges];
			int i = 0;
			for (int iNode = 0; iNode < m_BayesNet.getNrOfNodes(); iNode++) {
				for (int iParent = 0; iParent < m_BayesNet.getNrOfParents(iNode); iParent++) {
					int nParent = m_BayesNet.getParent(iNode, iParent);
					String sEdge = m_BayesNet.getNodeName(nParent);
					sEdge += " -> ";
					sEdge += m_BayesNet.getNodeName(iNode);
					options[i++] = sEdge;
				}
				
			}
			deleteArc(options);
		}
	} // class ActionDeleteArc

	class ActionNew extends MyAction {
		/** for serialization */
		private static final long serialVersionUID = -2038911085935515L;

		public ActionNew() {
			super("New", "New Network", "new", "");
		} // c'tor

		public void actionPerformed(ActionEvent ae) {
			m_sFileName = "";
			m_BayesNet = new EditableBayesNet(true);
			updateStatus();
			layoutGraph();
			a_datagenerator.setEnabled(false);
			m_BayesNet.clearUndoStack();
		    m_jStatusBar.setText("New Network");
			m_Selection = new Selection();
			repaint();
		}
	} // class ActionNew

	class ActionLoad extends MyAction {
		/** for serialization */
		private static final long serialVersionUID = -2038911085935515L;

		public ActionLoad() {
			super("Load", "Load Graph", "open", "ctrl O");
		} // c'tor

		public void actionPerformed(ActionEvent ae) {
			JFileChooser fc = new JFileChooser(System.getProperty("user.dir"));
			ExtensionFileFilter ef1 = new ExtensionFileFilter(".arff", "ARFF files");
			ExtensionFileFilter ef2 = new ExtensionFileFilter(".xml", "XML BIF files");
			fc.addChoosableFileFilter(ef1);
			fc.addChoosableFileFilter(ef2);
			fc.setDialogTitle("Load Graph");
			int rval = fc.showOpenDialog(GUI.this);

			if (rval == JFileChooser.APPROVE_OPTION) {
				String sFileName = fc.getSelectedFile().toString();
				if (sFileName.endsWith(ef1.getExtensions()[0])) {
					initFromArffFile(sFileName);
				} else {
					try {
						readBIFFromFile(sFileName);
					} catch (Exception e) {
						e.printStackTrace();
					}
				}
			    m_jStatusBar.setText("Loaded " + sFileName);
			    updateStatus();
			}
		}
	} // class ActionLoad
	
	
	class ActionViewStatusbar extends MyAction {
		/** for serialization */
		private static final long serialVersionUID = -20389330812354L;

		public ActionViewStatusbar() {
			super("View statusbar", "View statusbar", "statusbar", "");
		} // c'tor

		public void actionPerformed(ActionEvent ae) {
			m_jStatusBar.setVisible(!m_jStatusBar.isVisible());
		} // actionPerformed
	} // class ActionViewStatusbar

	class ActionViewToolbar extends MyAction {
		/** for serialization */
		private static final long serialVersionUID = -20389110812354L;

		public ActionViewToolbar() {
			super("View toolbar", "View toolbar", "toolbar", "");
		} // c'tor

		public void actionPerformed(ActionEvent ae) {
			m_jTbTools.setVisible(!m_jTbTools.isVisible());
		} // actionPerformed
	} // class ActionViewToolbar

	class ActionSave extends MyAction {
		/** for serialization */
		private static final long serialVersionUID = -20389110859355156L;

		public ActionSave() {
			super("Save", "Save Graph", "save", "ctrl S");
		} // c'tor

		public ActionSave(String sName, String sToolTipText, String sIcon, String sAcceleratorKey) {
			super(sName, sToolTipText, sIcon, sAcceleratorKey);
		} // c'tor

		public void actionPerformed(ActionEvent ae) {
			if (!m_sFileName.equals("")) {
				saveFile(m_sFileName);
				m_BayesNet.isSaved();
			    m_jStatusBar.setText("Saved as " + m_sFileName);
			} else {
				if (saveAs()) {
					m_BayesNet.isSaved();
				    m_jStatusBar.setText("Saved as " + m_sFileName);					
				}
			}
		} // actionPerformed


		ExtensionFileFilter ef1 = new ExtensionFileFilter(".xml", "XML BIF files");

		boolean saveAs() {
			JFileChooser fc = new JFileChooser(System.getProperty("user.dir"));
			fc.addChoosableFileFilter(ef1);
			fc.setDialogTitle("Save Graph As");
			if (!m_sFileName.equals("")) {
				// can happen on actionQuit
				fc.setSelectedFile(new File(m_sFileName));
			}
			int rval = fc.showSaveDialog(GUI.this);

			if (rval == JFileChooser.APPROVE_OPTION) {
				// System.out.println("Saving to file \""+
				// f.getAbsoluteFile().toString()+"\"");
				String sFileName = fc.getSelectedFile().toString();
				if (!sFileName.endsWith(".xml"))
					sFileName = sFileName.concat(".xml");
				saveFile(sFileName);
				return true;
			}
			return false;
		} // saveAs

		protected void saveFile(String sFileName) {
		    try {
		        FileWriter outfile = new FileWriter(sFileName);
		        outfile.write(m_BayesNet.toXMLBIF03());
		        outfile.close();
				m_sFileName = sFileName;
			    m_jStatusBar.setText("Saved as " + m_sFileName);
		      }
		      catch(IOException e) { 
		    	  e.printStackTrace(); 
		      }
		  } // saveFile
	} // class ActionSave
	
	class ActionSaveAs extends ActionSave {
		/** for serialization */
		private static final long serialVersionUID = -20389110859354L;

		public ActionSaveAs() {
			super("Save As", "Save Graph As", "saveas", "");
		} // c'tor

		public void actionPerformed(ActionEvent ae) {
			saveAs();
		} // actionPerformed
	} // class ActionSaveAs

	class ActionPrint extends ActionSave {
		/** for serialization */
		private static final long serialVersionUID = -20389001859354L;
		boolean m_bIsPrinting = false;
		public ActionPrint() {
			super("Print", "Print Graph", "print", "ctrl P");
		} // c'tor

		public void actionPerformed(ActionEvent ae) {
		    PrinterJob printJob = PrinterJob.getPrinterJob();
		    printJob.setPrintable(m_GraphPanel);
		    if (printJob.printDialog())
		      try { 
		  		m_bIsPrinting = true;
		        printJob.print();
				m_bIsPrinting = false;
		      } catch(PrinterException pe) {
		        m_jStatusBar.setText("Error printing: " + pe);
				m_bIsPrinting = false;
		      }
			    m_jStatusBar.setText("Print");
		} // actionPerformed
		public boolean isPrinting() {return m_bIsPrinting;}

	} // class ActionPrint

	class ActionQuit extends ActionSave {
		/** for serialization */
		private static final long serialVersionUID = -2038911085935515L;

		public ActionQuit() {
			super("Exit", "Exit Program", "exit", "");
		} // c'tor

		public void actionPerformed(ActionEvent ae) {
			if (m_BayesNet.isChanged()) {
				int result = JOptionPane.showConfirmDialog(null, "Network changed. Do you want to save it?", "Save before closing?", JOptionPane.YES_NO_CANCEL_OPTION);
				if (result == JOptionPane.CANCEL_OPTION) {
					return;
				}
				if (result == JOptionPane.YES_OPTION) {
					if (!saveAs()) {
						return;
					}
				}
			}
			System.exit(0);
		}
	} // class ActionQuit

	class ActionHelp extends MyAction {
		/** for serialization */
		private static final long serialVersionUID = -20389110859354L;

		public ActionHelp() {
			super("Help", "Bayesian Network Workbench Help", "help", "");
		} // c'tor

		public void actionPerformed(ActionEvent ae) {
			JOptionPane.showMessageDialog(null, "See Weka Homepage\nhttp://www.cs.waikato.ac.nz/ml", "Help Message",
					JOptionPane.PLAIN_MESSAGE);
		}
	} // class ActionHelp

	class ActionAbout extends MyAction {
		/** for serialization */
		private static final long serialVersionUID = -20389110859353L;

		public ActionAbout() {
			super("About", "Help about", "about", "");
		} // c'tor

		public void actionPerformed(ActionEvent ae) {
			JOptionPane.showMessageDialog(null, "Bayesian Network Workbench\nPart of Weka\n2007", "About Message",
					JOptionPane.PLAIN_MESSAGE);
		}
	} // class ActionAbout

	class ActionZoomIn extends MyAction {
		/** for serialization */
		private static final long serialVersionUID = -2038911085935515L;

		public ActionZoomIn() {
			super("Zoom in", "Zoom in", "zoomin", "+");
		} // c'tor

		public void actionPerformed(ActionEvent ae) {
			int i = 0, s = (int) (m_fScale * 100);
			if (s < 300)
				i = s / 25;
			else if (s < 700)
				i = 6 + s / 50;
			else
				i = 13 + s / 100;

			if (s >= 999) {
				setEnabled(false);
				return;
			} else if (s >= 10) {
				if (i >= 22) {
					setEnabled(false);
				}
				if (s == 10 && !a_zoomout.isEnabled()) {
					a_zoomout.setEnabled(true);
				}
				m_jTfZoom.setText(m_nZoomPercents[i + 1] + "%");
				m_fScale = m_nZoomPercents[i + 1] / 100D;
			} else {
				if (!a_zoomout.isEnabled())
					a_zoomout.setEnabled(true);
				m_jTfZoom.setText(m_nZoomPercents[0] + "%");
				m_fScale = m_nZoomPercents[0] / 100D;
			}
			setAppropriateSize();
			m_GraphPanel.repaint();
			m_GraphPanel.invalidate();
			m_jScrollPane.revalidate();
		    m_jStatusBar.setText("Zooming in");
		}
	} // class ActionZoomIn

	class ActionZoomOut extends MyAction {
		/** for serialization */
		private static final long serialVersionUID = -203891108593551L;

		public ActionZoomOut() {
			super("Zoom out", "Zoom out", "zoomout", "-");
		} // c'tor

		public void actionPerformed(ActionEvent ae) {
			int i = 0, s = (int) (m_fScale * 100);
			if (s < 300)
				i = (int) Math.ceil(s / 25D);
			else if (s < 700)
				i = 6 + (int) Math.ceil(s / 50D);
			else
				i = 13 + (int) Math.ceil(s / 100D);

			if (s <= 10) {
				setEnabled(false);
			} else if (s < 999) {
				if (i <= 1) {
					setEnabled(false);
				}
				m_jTfZoom.setText(m_nZoomPercents[i - 1] + "%");
				m_fScale = m_nZoomPercents[i - 1] / 100D;
			} else {
				if (!a_zoomin.isEnabled())
					a_zoomin.setEnabled(true);
				m_jTfZoom.setText(m_nZoomPercents[22] + "%");
				m_fScale = m_nZoomPercents[22] / 100D;
			}
			setAppropriateSize();
			m_GraphPanel.repaint();
			m_GraphPanel.invalidate();
			m_jScrollPane.revalidate();
		    m_jStatusBar.setText("Zooming out");
		}
	} // class ActionZoomOut

	class ActionLayout extends MyAction {
		/** for serialization */
		private static final long serialVersionUID = -203891108593551L;

		public ActionLayout() {
			super("Layout", "Layout Graph", "layout", "ctrl L");
		} // c'tor

		JDialog dlg = null;

		public void actionPerformed(ActionEvent ae) {
			if (dlg == null) {
				dlg = new JDialog();
				dlg.setTitle("Graph Layout Options");
				final JCheckBox jCbCustomNodeSize = new JCheckBox("Custom Node Size");
				final JLabel jLbNodeWidth = new JLabel("Width");
				final JLabel jLbNodeHeight = new JLabel("Height");

				m_jTfNodeWidth.setHorizontalAlignment(JTextField.CENTER);
				m_jTfNodeWidth.setText("" + m_nNodeWidth);
				m_jTfNodeHeight.setHorizontalAlignment(JTextField.CENTER);
				m_jTfNodeHeight.setText("" + m_nNodeHeight);
				jLbNodeWidth.setEnabled(false);
				m_jTfNodeWidth.setEnabled(false);
				jLbNodeHeight.setEnabled(false);
				m_jTfNodeHeight.setEnabled(false);

				jCbCustomNodeSize.addActionListener(new ActionListener() {
					public void actionPerformed(ActionEvent ae) {
						if (((JCheckBox) ae.getSource()).isSelected()) {
							jLbNodeWidth.setEnabled(true);
							m_jTfNodeWidth.setEnabled(true);
							jLbNodeHeight.setEnabled(true);
							m_jTfNodeHeight.setEnabled(true);
						} else {
							jLbNodeWidth.setEnabled(false);
							m_jTfNodeWidth.setEnabled(false);
							jLbNodeHeight.setEnabled(false);
							m_jTfNodeHeight.setEnabled(false);
							setAppropriateSize();
							setAppropriateNodeSize();
						}
					}
				});
				JButton jBtLayout;
				jBtLayout = new JButton("Layout Graph");
				jBtLayout.setMnemonic('L');

				jBtLayout.addActionListener(new ActionListener() {
					public void actionPerformed(ActionEvent ae) {
						int tmpW, tmpH;

						if (jCbCustomNodeSize.isSelected()) {
							try {
								tmpW = Integer.parseInt(m_jTfNodeWidth.getText());
							} catch (NumberFormatException ne) {
								JOptionPane.showMessageDialog(GUI.this.getParent(),
										"Invalid integer entered for node width.", "Error", JOptionPane.ERROR_MESSAGE);
								tmpW = m_nNodeWidth;
								m_jTfNodeWidth.setText("" + m_nNodeWidth);

							}
							try {
								tmpH = Integer.parseInt(m_jTfNodeHeight.getText());
							} catch (NumberFormatException ne) {
								JOptionPane.showMessageDialog(GUI.this.getParent(),
										"Invalid integer entered for node height.", "Error", JOptionPane.ERROR_MESSAGE);
								tmpH = m_nNodeHeight;
								m_jTfNodeWidth.setText("" + m_nNodeHeight);
							}

							if (tmpW != m_nNodeWidth || tmpH != m_nNodeHeight) {
								m_nNodeWidth = tmpW;
								m_nPaddedNodeWidth = m_nNodeWidth + PADDING;
								m_nNodeHeight = tmpH;
							}
						}
						// JButton bt = (JButton) ae.getSource();
						// bt.setEnabled(false);
						dlg.setVisible(false);
						updateStatus();
						layoutGraph();
					    m_jStatusBar.setText("Laying out Bayes net");
					}
				});
				JButton jBtCancel;
				jBtCancel = new JButton("Cancel");
				jBtCancel.setMnemonic('C');
				jBtCancel.addActionListener(new ActionListener() {
					public void actionPerformed(ActionEvent ae) {
						dlg.setVisible(false);
					}
				});
				GridBagConstraints gbc = new GridBagConstraints();
				dlg.setLayout(new GridBagLayout());
				//dlg.add(m_le.getControlPanel());

				Container c = new Container();
				c.setLayout(new GridBagLayout());

				gbc.gridwidth = 1;
				gbc.insets = new Insets(8, 0, 0, 0);
				gbc.anchor = GridBagConstraints.NORTHWEST;
				gbc.gridwidth = GridBagConstraints.REMAINDER;
				c.add(jCbCustomNodeSize, gbc);
				gbc.gridwidth = GridBagConstraints.RELATIVE;
				c.add(jLbNodeWidth, gbc);
				gbc.gridwidth = GridBagConstraints.REMAINDER;
				c.add(m_jTfNodeWidth, gbc);
				gbc.gridwidth = GridBagConstraints.RELATIVE;
				c.add(jLbNodeHeight, gbc);
				gbc.gridwidth = GridBagConstraints.REMAINDER;
				c.add(m_jTfNodeHeight, gbc);
				gbc.fill = GridBagConstraints.HORIZONTAL;
				dlg.add(c, gbc);
				dlg.add(jBtLayout);
				gbc.gridwidth = GridBagConstraints.REMAINDER;
				dlg.add(jBtCancel);
			}
			dlg.setLocation(100, 100);
			dlg.setVisible(true);
			dlg.setSize(dlg.getPreferredSize());
			dlg.setVisible(false);
			dlg.setVisible(true);
			dlg.repaint();
		}
	} // class ActionLayout

	/**
	 * Constructor
* Sets up the gui and initializes all the other previously uninitialized * variables. */ public GUI() { m_GraphPanel = new GraphPanel(); m_jScrollPane = new JScrollPane(m_GraphPanel); // creating a new layout engine and adding this class as its listener // to receive layoutComplete events m_jTfZoom = new JTextField("100%"); m_jTfZoom.setMinimumSize(m_jTfZoom.getPreferredSize()); m_jTfZoom.setHorizontalAlignment(JTextField.CENTER); m_jTfZoom.setToolTipText("Zoom"); m_jTfZoom.addActionListener(new ActionListener() { public void actionPerformed(ActionEvent ae) { JTextField jt = (JTextField) ae.getSource(); try { int i = -1; i = jt.getText().indexOf('%'); if (i == -1) i = Integer.parseInt(jt.getText()); else i = Integer.parseInt(jt.getText().substring(0, i)); if (i <= 999) m_fScale = i / 100D; jt.setText((int) (m_fScale * 100) + "%"); if (m_fScale > 0.1) { if (!a_zoomout.isEnabled()) a_zoomout.setEnabled(true); } else a_zoomout.setEnabled(false); if (m_fScale < 9.99) { if (!a_zoomin.isEnabled()) a_zoomin.setEnabled(true); } else a_zoomin.setEnabled(false); setAppropriateSize(); // m_GraphPanel.clearBuffer(); m_GraphPanel.repaint(); m_GraphPanel.invalidate(); m_jScrollPane.revalidate(); } catch (NumberFormatException ne) { JOptionPane.showMessageDialog(GUI.this.getParent(), "Invalid integer entered for zoom.", "Error", JOptionPane.ERROR_MESSAGE); jt.setText((m_fScale * 100) + "%"); } } }); GridBagConstraints gbc = new GridBagConstraints(); final JPanel p = new JPanel(new GridBagLayout()); p.setBorder(BorderFactory.createCompoundBorder(BorderFactory.createTitledBorder("ExtraControls"), BorderFactory .createEmptyBorder(4, 4, 4, 4))); p.setPreferredSize(new Dimension(0, 0)); m_jTbTools = new JToolBar(); m_jTbTools.setFloatable(false); m_jTbTools.setLayout(new GridBagLayout()); gbc.anchor = GridBagConstraints.NORTHWEST; gbc.gridwidth = GridBagConstraints.REMAINDER; gbc.insets = new Insets(0, 0, 0, 0); m_jTbTools.add(p, gbc); gbc.gridwidth = 1; m_jTbTools.add(a_new); m_jTbTools.add(a_save); m_jTbTools.add(a_load); m_jTbTools.addSeparator(new Dimension(2, 2)); m_jTbTools.add(a_cutnode); m_jTbTools.add(a_copynode); m_jTbTools.add(a_pastenode); m_jTbTools.addSeparator(new Dimension(2, 2)); m_jTbTools.add(a_undo); m_jTbTools.add(a_redo); m_jTbTools.addSeparator(new Dimension(2, 2)); m_jTbTools.add(a_alignleft); m_jTbTools.add(a_alignright); m_jTbTools.add(a_aligntop); m_jTbTools.add(a_alignbottom); m_jTbTools.add(a_centerhorizontal); m_jTbTools.add(a_centervertical); m_jTbTools.add(a_spacehorizontal); m_jTbTools.add(a_spacevertical); m_jTbTools.addSeparator(new Dimension(2, 2)); m_jTbTools.add(a_zoomin); gbc.fill = GridBagConstraints.VERTICAL; gbc.weighty = 1; JPanel p2 = new JPanel(new BorderLayout()); p2.setPreferredSize(m_jTfZoom.getPreferredSize()); p2.setMinimumSize(m_jTfZoom.getPreferredSize()); p2.add(m_jTfZoom, BorderLayout.CENTER); m_jTbTools.add(p2, gbc); gbc.weighty = 0; gbc.fill = GridBagConstraints.NONE; m_jTbTools.add(a_zoomout); m_jTbTools.addSeparator(new Dimension(2, 2)); // jTbTools.add(jBtExtraControls, gbc); m_jTbTools.add(a_layout); m_jTbTools.addSeparator(new Dimension(4, 2)); gbc.weightx = 1; gbc.fill = GridBagConstraints.BOTH; //jTbTools.add(m_layoutEngine.getProgressBar(), gbc); m_jStatusBar = new JLabel("Status bar"); this.setLayout(new BorderLayout()); this.add(m_jTbTools, BorderLayout.NORTH); this.add(m_jScrollPane, BorderLayout.CENTER); this.add(m_jStatusBar, BorderLayout.SOUTH); updateStatus(); a_datagenerator.setEnabled(false); makeMenuBar(); } /** * Get the menu bar for this application. * * @return the menu bar */ public JMenuBar getMenuBar() { return m_menuBar; } private void makeMenuBar() { m_menuBar = new JMenuBar(); JMenu fileMenu = new JMenu("File"); fileMenu.setMnemonic('F'); m_menuBar.add(fileMenu); fileMenu.add(a_new); fileMenu.add(a_load); fileMenu.add(a_save); fileMenu.add(a_saveas); fileMenu.addSeparator(); fileMenu.add(a_print); fileMenu.add(a_export); fileMenu.addSeparator(); fileMenu.add(a_quit); JMenu editMenu = new JMenu("Edit"); editMenu.setMnemonic('E'); m_menuBar.add(editMenu); editMenu.add(a_undo); editMenu.add(a_redo); editMenu.addSeparator(); editMenu.add(a_selectall); editMenu.add(a_delnode); editMenu.add(a_cutnode); editMenu.add(a_copynode); editMenu.add(a_pastenode); editMenu.addSeparator(); editMenu.add(a_addnode); editMenu.add(a_addarc); editMenu.add(a_delarc); editMenu.addSeparator(); editMenu.add(a_alignleft); editMenu.add(a_alignright); editMenu.add(a_aligntop); editMenu.add(a_alignbottom); editMenu.add(a_centerhorizontal); editMenu.add(a_centervertical); editMenu.add(a_spacehorizontal); editMenu.add(a_spacevertical); JMenu toolMenu = new JMenu("Tools"); toolMenu.setMnemonic('T'); toolMenu.add(a_networkgenerator); toolMenu.add(a_datagenerator); toolMenu.add(a_datasetter); toolMenu.add(a_learn); toolMenu.add(a_learnCPT); toolMenu.addSeparator(); toolMenu.add(a_layout); toolMenu.addSeparator(); final JCheckBoxMenuItem viewMargins = new JCheckBoxMenuItem("Show Margins", false); viewMargins.addActionListener(new ActionListener() { public void actionPerformed(ActionEvent ae) { boolean bPrev = m_bViewMargins; m_bViewMargins = viewMargins.getState(); if (bPrev == false && viewMargins.getState() == true) { updateStatus(); } repaint(); } }); toolMenu.add(viewMargins); final JCheckBoxMenuItem viewCliques = new JCheckBoxMenuItem("Show Cliques", false); viewCliques.addActionListener(new ActionListener() { public void actionPerformed(ActionEvent ae) { boolean bPrev = m_bViewCliques; m_bViewCliques = viewCliques.getState(); if (bPrev == false && viewCliques.getState() == true) { updateStatus(); } repaint(); } }); toolMenu.add(viewCliques); m_menuBar.add(toolMenu); JMenu viewMenu = new JMenu("View"); viewMenu.setMnemonic('V'); m_menuBar.add(viewMenu); viewMenu.add(a_zoomin); viewMenu.add(a_zoomout); viewMenu.addSeparator(); viewMenu.add(a_viewtoolbar); viewMenu.add(a_viewstatusbar); JMenu helpMenu = new JMenu("Help"); helpMenu.setMnemonic('H'); m_menuBar.add(helpMenu); helpMenu.add(a_help); helpMenu.add(a_about); } /** * This method sets the node size that is appropriate considering the * maximum label size that is present. It is used internally when custom * node size checkbox is unchecked. */ protected void setAppropriateNodeSize() { int strWidth; FontMetrics fm = this.getFontMetrics(this.getFont()); int nMaxStringWidth = DEFAULT_NODE_WIDTH; if (nMaxStringWidth == 0) for (int iNode = 0; iNode < m_BayesNet.getNrOfNodes(); iNode++) { strWidth = fm.stringWidth(m_BayesNet.getNodeName(iNode)); if (strWidth > nMaxStringWidth) nMaxStringWidth = strWidth; } m_nNodeWidth = nMaxStringWidth + 4; m_nPaddedNodeWidth = m_nNodeWidth + PADDING; m_jTfNodeWidth.setText("" + m_nNodeWidth); m_nNodeHeight = 2 * fm.getHeight(); m_jTfNodeHeight.setText("" + m_nNodeHeight); } /** * Sets the preferred size for m_GraphPanel GraphPanel to the minimum size that is * neccessary to display the graph. */ public void setAppropriateSize() { int maxX = 0, maxY = 0; m_GraphPanel.setScale(m_fScale, m_fScale); for (int iNode = 0; iNode < m_BayesNet.getNrOfNodes(); iNode++) { int nPosX = m_BayesNet.getPositionX(iNode); int nPosY = m_BayesNet.getPositionY(iNode); if (maxX < nPosX) maxX = nPosX + 100; if (maxY < nPosY) maxY = nPosY; } m_GraphPanel.setPreferredSize(new Dimension((int) ((maxX + m_nPaddedNodeWidth + 2) * m_fScale), (int) ((maxY + m_nNodeHeight + 2) * m_fScale))); m_GraphPanel.revalidate(); } // setAppropriateSize /** * This method is an implementation for LayoutCompleteEventListener class. * It sets the size appropriate for m_GraphPanel GraphPanel and and revalidates it's * container JScrollPane once a LayoutCompleteEvent is received from the * LayoutEngine. Also, it updates positions of the Bayesian network stored * in m_BayesNet. */ public void layoutCompleted(LayoutCompleteEvent le) { LayoutEngine layoutEngine = m_layoutEngine; // (LayoutEngine) le.getSource(); FastVector nPosX = new FastVector(m_BayesNet.getNrOfNodes()); FastVector nPosY = new FastVector(m_BayesNet.getNrOfNodes()); for (int iNode = 0; iNode < layoutEngine.getNodes().size(); iNode++) { GraphNode gNode = (GraphNode) layoutEngine.getNodes().elementAt(iNode); if (gNode.nodeType == GraphNode.NORMAL) { nPosX.addElement(gNode.x); nPosY.addElement(gNode.y); } } m_BayesNet.layoutGraph(nPosX, nPosY); m_jStatusBar.setText("Graph layed out"); a_undo.setEnabled(true); a_redo.setEnabled(false); setAppropriateSize(); m_GraphPanel.invalidate(); m_jScrollPane.revalidate(); m_GraphPanel.repaint(); } // layoutCompleted /** * BIF reader
* Reads a graph description in XMLBIF03 from an file * with name sFileName */ public void readBIFFromFile(String sFileName) throws BIFFormatException, IOException { m_sFileName = sFileName; try { BIFReader bayesNet = new BIFReader(); bayesNet.processFile(sFileName); m_BayesNet = new EditableBayesNet(bayesNet); updateStatus(); a_datagenerator.setEnabled(m_BayesNet.getNrOfNodes() > 0); m_BayesNet.clearUndoStack(); } catch (Exception ex) { ex.printStackTrace(); return; } setAppropriateNodeSize(); setAppropriateSize(); } // readBIFFromFile /* read arff file from file sFileName * and start new Bayesian network with nodes * representing attributes in data set. */ void initFromArffFile(String sFileName) { try { Instances instances = new Instances(new FileReader(sFileName)); m_BayesNet = new EditableBayesNet(instances); m_Instances = instances; a_learn.setEnabled(true); a_learnCPT.setEnabled(true); setAppropriateNodeSize(); setAppropriateSize(); } catch (Exception ex) { ex.printStackTrace(); return; } } // initFromArffFile /** * The panel which contains the actual Bayeian network. */ private class GraphPanel extends PrintablePanel implements Printable { /** for serialization */ private static final long serialVersionUID = -3562813603236753173L; /** node drawing modes */ final static int HIGHLIGHTED = 1; final static int NORMAL = 0; public GraphPanel() { super(); this.addMouseListener(new GraphVisualizerMouseListener()); this.addMouseMotionListener(new GraphVisualizerMouseMotionListener()); this.setToolTipText(""); } // c'tor /* For showing instructions when hovering over a node * (non-Javadoc) * @see javax.swing.JComponent#getToolTipText(java.awt.event.MouseEvent) */ public String getToolTipText(MouseEvent me) { int x, y; Rectangle r; x = y = 0; r = new Rectangle(0, 0, (int) (m_nPaddedNodeWidth * m_fScale), (int) (m_nNodeHeight * m_fScale)); x += me.getX(); y += me.getY(); for (int iNode = 0; iNode < m_BayesNet.getNrOfNodes(); iNode++) { r.x = (int) (m_BayesNet.getPositionX(iNode) * m_fScale); r.y = (int) (m_BayesNet.getPositionY(iNode) * m_fScale); if (r.contains(x, y)) { return m_BayesNet.getNodeName(iNode) + " (right click to manipulate this node)"; } } return null; } // getToolTipText /* Code for showing the graph in the panel. * (non-Javadoc) * @see javax.swing.JComponent#paintComponent(java.awt.Graphics) */ public void paintComponent(Graphics gr) { Graphics2D g = (Graphics2D) gr; RenderingHints rh = new RenderingHints(RenderingHints.KEY_ANTIALIASING, RenderingHints.VALUE_ANTIALIAS_ON); rh.put(RenderingHints.KEY_RENDERING, RenderingHints.VALUE_RENDER_SPEED); g.setRenderingHints(rh); g.scale(m_fScale, m_fScale); Rectangle r = g.getClipBounds(); g.clearRect(r.x, r.y, r.width, r.height); if (m_bViewCliques) { m_nClique = 1; viewCliques(g, m_marginCalculator.m_root); } for (int iNode = 0; iNode < m_BayesNet.getNrOfNodes(); iNode++) { drawNode(g, iNode, NORMAL); } if (!a_export.isExporting() && !a_print.isPrinting()) { m_Selection.draw(g); } if (m_nSelectedRect != null) { g.drawRect((int)(m_nSelectedRect.x/ m_fScale), (int)(m_nSelectedRect.y/ m_fScale), (int)(m_nSelectedRect.width/ m_fScale), (int)(m_nSelectedRect.height/ m_fScale)); } } // paintComponent /** number of the clique being drawn. Used for selecting the color of the clique */ int m_nClique = 1; /* draws cliques in junction tree. * */ void viewCliques(Graphics g, JunctionTreeNode node) { int [] nodes = node.m_nNodes; g.setColor( new Color(m_nClique % 7 * 256 /7, (m_nClique % 2 * 256 / 2), (m_nClique % 3 * 256 / 3)) ); int dX = m_nPaddedNodeWidth / 2 + m_nClique; int dY = m_nNodeHeight / 2; int nPosX = 0; int nPosY = 0; String sStr = ""; for (int j = 0; j < nodes.length; j++) { nPosX += m_BayesNet.getPositionX(nodes[j]); nPosY += m_BayesNet.getPositionY(nodes[j]); sStr += " " + nodes[j]; for (int k = j+1; k < nodes.length; k++) { g.drawLine( m_BayesNet.getPositionX(nodes[j]) + dX, m_BayesNet.getPositionY(nodes[j]) + dY, m_BayesNet.getPositionX(nodes[k]) + dX, m_BayesNet.getPositionY(nodes[k]) + dY ); } } m_nClique++; nPosX /= nodes.length; nPosY /= nodes.length; g.drawString("Clique " + m_nClique + "("+sStr+")", nPosX, nPosY); for (int iChild = 0; iChild < node.m_children.size(); iChild++) { viewCliques(g, (JunctionTreeNode) node.m_children.elementAt(iChild)); } } // viewCliques /* Draw a node with index iNode on Graphics g at position * Drawing mode can be NORMAL or HIGHLIGHTED. */ protected void drawNode(Graphics g, int iNode, int mode) { int nPosX = m_BayesNet.getPositionX(iNode); int nPosY = m_BayesNet.getPositionY(iNode); g.setColor(this.getBackground().darker().darker()); FontMetrics fm = getFontMetrics(getFont()); if (mode == HIGHLIGHTED) { g.setXORMode(Color.green); // g.setColor(Color.green); } g.fillOval(nPosX + m_nPaddedNodeWidth - m_nNodeWidth - (m_nPaddedNodeWidth - m_nNodeWidth) / 2, nPosY, m_nNodeWidth, m_nNodeHeight); g.setColor(Color.white); if (mode == HIGHLIGHTED) { g.setXORMode(Color.red); } // Draw the node's label if it can fit inside the node's // current width otherwise just display its node nr // if it can fit in node's current width if (fm.stringWidth(m_BayesNet.getNodeName(iNode)) <= m_nNodeWidth) { g.drawString(m_BayesNet.getNodeName(iNode), nPosX + m_nPaddedNodeWidth / 2 - fm.stringWidth(m_BayesNet.getNodeName(iNode)) / 2, nPosY + m_nNodeHeight / 2 + fm.getHeight() / 2 - 2); } else if (fm.stringWidth("" + iNode) <= m_nNodeWidth) { g.drawString("" + iNode, nPosX + m_nPaddedNodeWidth / 2 - fm.stringWidth("" + iNode) / 2, nPosY + m_nNodeHeight / 2 + fm.getHeight() / 2 - 2); } if (mode == HIGHLIGHTED) { g.setXORMode(Color.green); } if (m_bViewMargins) { if (m_BayesNet.getEvidence(iNode) < 0) { g.setColor(new Color(0, 128, 0)); } else { g.setColor(new Color(128, 0, 0)); } double[] P = m_BayesNet.getMargin(iNode); for (int iValue = 0; iValue < P.length; iValue++) { String sP = P[iValue] + ""; if (sP.charAt(0) == '0') { sP = sP.substring(1); } if (sP.length() > 5) { sP = sP.substring(1, 5); } g.fillRect(nPosX + m_nPaddedNodeWidth, nPosY + iValue * 10 + 2, (int) (P[iValue] * 100), 8); g.drawString(m_BayesNet.getNodeValue(iNode, iValue) + " " + sP, nPosX + m_nPaddedNodeWidth + (int) (P[iValue] * 100), nPosY + iValue * 10 + 10); } } if (m_bViewCliques) { return; } g.setColor(Color.black); // Drawing all incoming edges into the node, for (int iParent = 0; iParent < m_BayesNet.getNrOfParents(iNode); iParent++) { int nParent = m_BayesNet.getParent(iNode, iParent); int nPosX1 = nPosX + m_nPaddedNodeWidth / 2; int nPosY1 = nPosY + m_nNodeHeight; int nPosX2 = m_BayesNet.getPositionX(nParent); int nPosY2 = m_BayesNet.getPositionY(nParent); int nPosX2b = nPosX2 + m_nPaddedNodeWidth / 2; int nPosY2b = nPosY2; double phi = Math.atan2((nPosX2b - nPosX1 + 0.0) * m_nNodeHeight, (nPosY2b - nPosY1 + 0.0) * m_nNodeWidth); nPosX1 = (int) (nPosX + m_nPaddedNodeWidth / 2 + Math.sin(phi) * m_nNodeWidth / 2); nPosY1 = (int) (nPosY + m_nNodeHeight / 2 + Math.cos(phi) * m_nNodeHeight / 2); nPosX2b = (int) (nPosX2 + m_nPaddedNodeWidth / 2 - Math.sin(phi) * m_nNodeWidth / 2); nPosY2b = (int) (nPosY2 + m_nNodeHeight / 2 - Math.cos(phi) * m_nNodeHeight / 2); drawArrow(g, nPosX2b, nPosY2b, nPosX1, nPosY1); } if (mode == HIGHLIGHTED) { FastVector children = m_BayesNet.getChildren(iNode); for (int iChild = 0; iChild < children.size(); iChild++) { int nChild = (Integer) children.elementAt(iChild); int nPosX1 = nPosX + m_nPaddedNodeWidth / 2; int nPosY1 = nPosY; int nPosX2 = m_BayesNet.getPositionX(nChild); int nPosY2 = m_BayesNet.getPositionY(nChild); int nPosX2b = nPosX2 + m_nPaddedNodeWidth / 2; int nPosY2b = nPosY2 + m_nNodeHeight; double phi = Math.atan2((nPosX2b - nPosX1 + 0.0) * m_nNodeHeight, (nPosY2b - nPosY1 + 0.0) * m_nNodeWidth); nPosX1 = (int) (nPosX + m_nPaddedNodeWidth / 2 + Math.sin(phi) * m_nNodeWidth / 2); nPosY1 = (int) (nPosY + m_nNodeHeight / 2 + Math.cos(phi) * m_nNodeHeight / 2); nPosX2b = (int) (nPosX2 + m_nPaddedNodeWidth / 2 - Math.sin(phi) * m_nNodeWidth / 2); nPosY2b = (int) (nPosY2 + m_nNodeHeight / 2 - Math.cos(phi) * m_nNodeHeight / 2); drawArrow(g, nPosX1, nPosY1, nPosX2b, nPosY2b); } } } // drawNode /** * This method draws an arrow on a line from (x1,y1) to (x2,y2). The * arrow head is seated on (x2,y2) and is in the direction of the line. * If the arrow is needed to be drawn in the opposite direction then * simply swap the order of (x1, y1) and (x2, y2) when calling this * function. */ protected void drawArrow(Graphics g, int nPosX1, int nPosY1, int nPosX2, int nPosY2) { g.drawLine(nPosX1, nPosY1, nPosX2, nPosY2); if (nPosX1 == nPosX2) { if (nPosY1 < nPosY2) { g.drawLine(nPosX2, nPosY2, nPosX2 + 4, nPosY2 - 8); g.drawLine(nPosX2, nPosY2, nPosX2 - 4, nPosY2 - 8); } else { g.drawLine(nPosX2, nPosY2, nPosX2 + 4, nPosY2 + 8); g.drawLine(nPosX2, nPosY2, nPosX2 - 4, nPosY2 + 8); } } else { // theta=line's angle from base, beta=angle of arrow's side from // line double hyp = 0, base = 0, perp = 0, theta, beta; int nPosX3 = 0, nPosY3 = 0; if (nPosX2 < nPosX1) { base = nPosX1 - nPosX2; hyp = Math.sqrt((nPosX2 - nPosX1) * (nPosX2 - nPosX1) + (nPosY2 - nPosY1) * (nPosY2 - nPosY1)); theta = Math.acos(base / hyp); } else { // x1>x2 as we already checked x1==x2 before base = nPosX1 - nPosX2; hyp = Math.sqrt((nPosX2 - nPosX1) * (nPosX2 - nPosX1) + (nPosY2 - nPosY1) * (nPosY2 - nPosY1)); theta = Math.acos(base / hyp); } beta = 30 * Math.PI / 180; hyp = 8; base = Math.cos(theta - beta) * hyp; perp = Math.sin(theta - beta) * hyp; nPosX3 = (int) (nPosX2 + base); if (nPosY1 < nPosY2) nPosY3 = (int) (nPosY2 - perp); else nPosY3 = (int) (nPosY2 + perp); g.drawLine(nPosX2, nPosY2, nPosX3, nPosY3); base = Math.cos(theta + beta) * hyp; perp = Math.sin(theta + beta) * hyp; nPosX3 = (int) (nPosX2 + base); if (nPosY1 < nPosY2) nPosY3 = (int) (nPosY2 - perp); else nPosY3 = (int) (nPosY2 + perp); g.drawLine(nPosX2, nPosY2, nPosX3, nPosY3); } } // drawArrow /** * This method highlights a given node and all its incoming and outgoing arcs */ public void highLight(int iNode) { Graphics2D g = (Graphics2D) this.getGraphics(); RenderingHints rh = new RenderingHints(RenderingHints.KEY_ANTIALIASING, RenderingHints.VALUE_ANTIALIAS_ON); rh.put(RenderingHints.KEY_RENDERING, RenderingHints.VALUE_RENDER_SPEED); g.setRenderingHints(rh); g.setPaintMode(); g.scale(m_fScale, m_fScale); drawNode(g, iNode, HIGHLIGHTED); } // highlight /** implementation of Printable, used for printing * @see Printable */ public int print(Graphics g, PageFormat pageFormat, int pageIndex) { if (pageIndex > 0) { return(NO_SUCH_PAGE); } else { Graphics2D g2d = (Graphics2D)g; g2d.translate(pageFormat.getImageableX(), pageFormat.getImageableY()); double fHeight = pageFormat.getImageableHeight(); double fWidth = pageFormat.getImageableWidth(); int xMax = 1; int yMax = 1; for (int iNode = 0; iNode < m_BayesNet.getNrOfNodes(); iNode++) { if (xMax < m_BayesNet.getPositionX(iNode)) { xMax = m_BayesNet.getPositionX(iNode); } if (yMax < m_BayesNet.getPositionY(iNode)) { yMax = m_BayesNet.getPositionY(iNode); } } double fCurrentScale = m_fScale; xMax += m_nPaddedNodeWidth + 100; if (fWidth/xMax < fHeight/yMax) { m_fScale = fWidth/xMax; } else { m_fScale = fHeight/yMax; } // Turn off double buffering paint(g2d); m_fScale = fCurrentScale; // Turn double buffering back on return(PAGE_EXISTS); } } // print } // class GraphPanel /** * Table Model for the Table for editing CPTs */ private class GraphVisualizerTableModel extends AbstractTableModel { /** for serialization */ private static final long serialVersionUID = -4789813491347366596L; /** labels for the columns */ final String [] m_sColumnNames; /** probability table data **/ final double [][] m_fProbs; /** nr of node for currently editted CPT */ int m_iNode; public GraphVisualizerTableModel(int iNode) { m_iNode = iNode; double [][] probs = m_BayesNet.getDistribution(iNode); m_fProbs = new double[probs.length][probs[0].length]; for (int i = 0; i < probs.length; i++) { for (int j = 0; j < probs[0].length; j++) { m_fProbs[i][j] = probs[i][j]; } } m_sColumnNames = m_BayesNet.getValues(iNode); } // c'tor /** method that generates random CPTs */ public void randomize() { int nProbs = m_fProbs[0].length; Random random = new Random(); for (int i = 0; i < m_fProbs.length; i++) { // get random nrs for (int j = 0; j < nProbs-1; j++) { m_fProbs[i][j] = random.nextDouble(); } // sort for (int j = 0; j < nProbs-1; j++) { for (int k = j+1; k < nProbs-1; k++) { if (m_fProbs[i][j] > m_fProbs[i][k]) { double h = m_fProbs[i][j]; m_fProbs[i][j] = m_fProbs[i][k]; m_fProbs[i][k] = h; } } } double sum = m_fProbs[i][0]; for (int j = 1; j < nProbs-1; j++) { m_fProbs[i][j] = m_fProbs[i][j] - sum; sum += m_fProbs[i][j]; } m_fProbs[i][nProbs - 1] = 1.0 - sum; } } // randomize public void setData() {} /** return nr of colums */ public int getColumnCount() { return m_sColumnNames.length; } /** return nr of rows */ public int getRowCount() { return m_fProbs.length; } /** return name of specified colum * @param iCol index of the column */ public String getColumnName(int iCol) { return m_sColumnNames[iCol]; } /** return data point * @param iRow index of row in table * @param iCol index of column in table */ public Object getValueAt(int iRow, int iCol) { return new Double(m_fProbs[iRow][iCol]); } /** Set data point, assigns value to CPT entry * specified by row and column. The remainder of the * CPT is normalized so that the values add up to 1. * IF a value below zero of over 1 is given, no changes * take place. * @param oProb data point * @param iRow index of row in table * @param iCol index of column in table */ public void setValueAt(Object oProb, int iRow, int iCol) { Double fProb = (Double) oProb; if (fProb < 0 || fProb > 1) { return; } m_fProbs[iRow][iCol] = (double) fProb; double sum = 0; for (int i = 0; i < m_fProbs[iRow].length; i++) { sum += m_fProbs[iRow][i]; } if (sum > 1) { // handle overflow int i = m_fProbs[iRow].length - 1; while (sum > 1) { if (i != iCol) { if (m_fProbs[iRow][i] > sum - 1) { m_fProbs[iRow][i] -= sum - 1; sum = 1; } else { sum -= m_fProbs[iRow][i]; m_fProbs[iRow][i] = 0; } } i--; } } else { // handle underflow int i = m_fProbs[iRow].length - 1; while (sum < 1) { if (i != iCol) { m_fProbs[iRow][i] += 1 - sum; sum = 1; } i--; } } validate(); } // setData /* * JTable uses this method to determine the default renderer/ editor for * each cell. */ public Class getColumnClass(int c) { return getValueAt(0, c).getClass(); } /* * Implemented this to make sure the table is uneditable. */ public boolean isCellEditable(int row, int col) { return true; } } // class GraphVisualizerTableModel /** * Listener class for processing mouseClicked */ private class GraphVisualizerMouseListener extends MouseAdapter { /** A left mouseclick on a node adds node to selection (depending * on shift and ctrl keys). * A right mouseclick on a node pops up menu with actions to be * performed on the node. * A right mouseclick outside another node pops up menu. */ public void mouseClicked(MouseEvent me) { int x, y; Rectangle r = new Rectangle(0, 0, (int) (m_nPaddedNodeWidth * m_fScale), (int) (m_nNodeHeight * m_fScale)); x = me.getX(); y = me.getY(); for (int iNode = 0; iNode < m_BayesNet.getNrOfNodes(); iNode++) { r.x = (int) (m_BayesNet.getPositionX(iNode) * m_fScale); r.y = (int) (m_BayesNet.getPositionY(iNode) * m_fScale); if (r.contains(x, y)) { m_nCurrentNode = iNode; if (me.getButton() == MouseEvent.BUTTON3) { handleRightNodeClick(me); } if (me.getButton() == MouseEvent.BUTTON1) { if ((me.getModifiersEx() & MouseEvent.CTRL_DOWN_MASK) != 0) { m_Selection.toggleSelection(m_nCurrentNode); } else if ((me.getModifiersEx() & MouseEvent.SHIFT_DOWN_MASK) != 0) { m_Selection.addToSelection(m_nCurrentNode); } else { m_Selection.clear(); m_Selection.addToSelection(m_nCurrentNode); } repaint(); } return; } } if (me.getButton() == MouseEvent.BUTTON3) { handleRightClick(me, (int)(x/m_fScale), (int)(y/m_fScale)); } } // mouseClicked /* update selection * (non-Javadoc) * @see java.awt.event.MouseListener#mouseReleased(java.awt.event.MouseEvent) */ public void mouseReleased(MouseEvent me) { if (m_nSelectedRect != null) { if ((me.getModifiersEx() & MouseEvent.CTRL_DOWN_MASK) != 0) { m_Selection.toggleSelection(m_nSelectedRect); } else if ((me.getModifiersEx() & MouseEvent.SHIFT_DOWN_MASK) != 0) { m_Selection.addToSelection(m_nSelectedRect); } else { m_Selection.clear(); m_Selection.addToSelection(m_nSelectedRect); } m_nSelectedRect = null; repaint(); } } // mouseReleased /** position clicked on */ int m_nPosX = 0, m_nPosY = 0; /* pop up menu with actions that apply in general or to selection (if any exists) */ void handleRightClick(MouseEvent me, int nPosX, int nPosY) { ActionListener act = new ActionListener() { public void actionPerformed(ActionEvent ae) { if (ae.getActionCommand().equals("Add node")) { a_addnode.addNode(m_nPosX, m_nPosY); return; } repaint(); } }; JPopupMenu popupMenu = new JPopupMenu("Choose a value"); JMenuItem addNodeItem = new JMenuItem("Add node"); addNodeItem.addActionListener(act); popupMenu.add(addNodeItem); FastVector selected = m_Selection.getSelected(); JMenu addArcMenu = new JMenu("Add parent"); popupMenu.add(addArcMenu); if (selected.size() == 0) { addArcMenu.setEnabled(false); } else { int nNodes = m_BayesNet.getNrOfNodes(); boolean[] isNotAllowedAsParent = new boolean[nNodes]; // prevent it being a parent of itself for (int iNode = 0; iNode < selected.size(); iNode++) { isNotAllowedAsParent[(Integer) selected.elementAt(iNode)] = true; } // prevent a descendant being a parent, since it introduces cycles for (int i = 0; i < nNodes; i++) { for (int iNode = 0; iNode < nNodes; iNode++) { for (int iParent = 0; iParent < m_BayesNet.getNrOfParents(iNode); iParent++) { if (isNotAllowedAsParent[m_BayesNet.getParent(iNode, iParent)]) { isNotAllowedAsParent[iNode] = true; } } } } // prevent nodes that are already a parent for (int iNode = 0; iNode < selected.size(); iNode++) { int nNode = (Integer) selected.elementAt(iNode); for (int iParent = 0; iParent < m_BayesNet.getNrOfParents(nNode); iParent++) { isNotAllowedAsParent[m_BayesNet.getParent(nNode, iParent)] = true; } } ActionListener addParentAction = new ActionListener() { public void actionPerformed(ActionEvent ae) { try { m_BayesNet.addArc(ae.getActionCommand(), m_Selection.getSelected()); m_jStatusBar.setText(m_BayesNet.lastActionMsg()); updateStatus(); } catch (Exception e) { e.printStackTrace(); } } }; // count nr of remaining candidates int nCandidates = 0; for (int i = 0; i < nNodes; i++) { if (!isNotAllowedAsParent[i]) { JMenuItem item = new JMenuItem(m_BayesNet.getNodeName(i)); item.addActionListener(addParentAction); addArcMenu.add(item); nCandidates++; } } if (nCandidates == 0) { addArcMenu.setEnabled(false); } } m_nPosX = nPosX; m_nPosY = nPosY; popupMenu.setLocation(me.getX(), me.getY()); popupMenu.show(m_GraphPanel, me.getX(), me.getY()); } // handleRightClick /* pop up menu with actions that apply to node that was clicked on */ void handleRightNodeClick(MouseEvent me) { m_Selection.clear(); repaint(); ActionListener renameValueAction = new ActionListener() { public void actionPerformed(ActionEvent ae) { renameValue(m_nCurrentNode, ae.getActionCommand()); } }; ActionListener delValueAction = new ActionListener() { public void actionPerformed(ActionEvent ae) { delValue(m_nCurrentNode, ae.getActionCommand()); } }; ActionListener addParentAction = new ActionListener() { public void actionPerformed(ActionEvent ae) { try { m_BayesNet.addArc(ae.getActionCommand(), m_BayesNet.getNodeName(m_nCurrentNode)); m_jStatusBar.setText(m_BayesNet.lastActionMsg()); updateStatus(); } catch (Exception e) { e.printStackTrace(); } } }; ActionListener delParentAction = new ActionListener() { public void actionPerformed(ActionEvent ae) { deleteArc(m_nCurrentNode, ae.getActionCommand()); } }; ActionListener delChildAction = new ActionListener() { public void actionPerformed(ActionEvent ae) { deleteArc(ae.getActionCommand(), m_nCurrentNode); } }; ActionListener setAvidenceAction = new ActionListener() { public void actionPerformed(ActionEvent ae) { try { String [] outcomes = m_BayesNet.getValues(m_nCurrentNode); int iValue = 0; while (iValue < outcomes.length && !outcomes[iValue].equals(ae.getActionCommand())) { iValue++; } if (iValue == outcomes.length) { iValue = -1; } if (iValue < outcomes.length) { m_jStatusBar.setText("Set evidence for " + m_BayesNet.getNodeName(m_nCurrentNode)); if (m_BayesNet.getEvidence(m_nCurrentNode) < 0 && iValue >= 0) { m_BayesNet.setEvidence(m_nCurrentNode, iValue); m_marginCalculatorWithEvidence.setEvidence(m_nCurrentNode, iValue); } else { m_BayesNet.setEvidence(m_nCurrentNode, iValue); SerializedObject so = new SerializedObject(m_marginCalculator); m_marginCalculatorWithEvidence = (MarginCalculator) so.getObject(); for (int iNode = 0; iNode < m_BayesNet.getNrOfNodes(); iNode++) { if (m_BayesNet.getEvidence(iNode) >= 0) { m_marginCalculatorWithEvidence.setEvidence(iNode, m_BayesNet.getEvidence(iNode)); } } } for (int iNode = 0; iNode < m_BayesNet.getNrOfNodes(); iNode++) { m_BayesNet.setMargin(iNode, m_marginCalculatorWithEvidence.getMargin(iNode)); } } } catch (Exception e) { e.printStackTrace(); } repaint(); } }; ActionListener act = new ActionListener() { public void actionPerformed(ActionEvent ae) { if (ae.getActionCommand().equals("Rename")) { renameNode(m_nCurrentNode); return; } if (ae.getActionCommand().equals("Add parent")) { addArcInto(m_nCurrentNode); return; } if (ae.getActionCommand().equals("Add value")) { addValue(); return; } if (ae.getActionCommand().equals("Delete node")) { deleteNode(m_nCurrentNode); return; } if (ae.getActionCommand().equals("Edit CPT")) { editCPT(m_nCurrentNode); return; } repaint(); } }; try { JPopupMenu popupMenu = new JPopupMenu("Choose a value"); JMenu setEvidenceMenu = new JMenu("Set evidence"); String [] outcomes = m_BayesNet.getValues(m_nCurrentNode); for (int iValue = 0; iValue < outcomes.length; iValue++) { JMenuItem item = new JMenuItem(outcomes[iValue]); item.addActionListener(setAvidenceAction); setEvidenceMenu.add(item); } setEvidenceMenu.addSeparator(); JMenuItem item = new JMenuItem("Clear"); item.addActionListener(setAvidenceAction); setEvidenceMenu.add(item); popupMenu.add(setEvidenceMenu); setEvidenceMenu.setEnabled(m_bViewMargins); popupMenu.addSeparator(); JMenuItem renameItem = new JMenuItem("Rename"); renameItem.addActionListener(act); popupMenu.add(renameItem); JMenuItem delNodeItem = new JMenuItem("Delete node"); delNodeItem.addActionListener(act); popupMenu.add(delNodeItem); JMenuItem editCPTItem = new JMenuItem("Edit CPT"); editCPTItem.addActionListener(act); popupMenu.add(editCPTItem); popupMenu.addSeparator(); JMenu addArcMenu = new JMenu("Add parent"); popupMenu.add(addArcMenu); int nNodes = m_BayesNet.getNrOfNodes(); boolean[] isNotAllowedAsParent = new boolean[nNodes]; // prevent it being a parent of itself isNotAllowedAsParent[m_nCurrentNode] = true; // prevent a descendant being a parent, since it introduces cycles for (int i = 0; i < nNodes; i++) { for (int iNode = 0; iNode < nNodes; iNode++) { for (int iParent = 0; iParent < m_BayesNet.getNrOfParents(iNode); iParent++) { if (isNotAllowedAsParent[m_BayesNet.getParent(iNode, iParent)]) { isNotAllowedAsParent[iNode] = true; } } } } // prevent nodes that are already a parent for (int iParent = 0; iParent < m_BayesNet.getNrOfParents(m_nCurrentNode); iParent++) { isNotAllowedAsParent[m_BayesNet.getParent(m_nCurrentNode, iParent)] = true; } // count nr of remaining candidates int nCandidates = 0; for (int i = 0; i < nNodes; i++) { if (!isNotAllowedAsParent[i]) { item = new JMenuItem(m_BayesNet.getNodeName(i)); item.addActionListener(addParentAction); addArcMenu.add(item); nCandidates++; } } if (nCandidates == 0) { addArcMenu.setEnabled(false); } JMenu delArcMenu = new JMenu("Delete parent"); popupMenu.add(delArcMenu); if (m_BayesNet.getNrOfParents(m_nCurrentNode) == 0) { delArcMenu.setEnabled(false); } for (int iParent = 0; iParent < m_BayesNet.getNrOfParents(m_nCurrentNode); iParent++) { item = new JMenuItem(m_BayesNet.getNodeName(m_BayesNet.getParent(m_nCurrentNode, iParent))); item.addActionListener(delParentAction); delArcMenu.add(item); } JMenu delChildMenu = new JMenu("Delete child"); popupMenu.add(delChildMenu); FastVector nChildren = m_BayesNet.getChildren(m_nCurrentNode); if (nChildren.size() == 0) { delChildMenu.setEnabled(false); } for (int iChild = 0; iChild < nChildren.size(); iChild++) { item = new JMenuItem(m_BayesNet.getNodeName((Integer) nChildren.elementAt(iChild))); item.addActionListener(delChildAction); delChildMenu.add(item); } popupMenu.addSeparator(); JMenuItem addValueItem = new JMenuItem("Add value"); addValueItem.addActionListener(act); popupMenu.add(addValueItem); JMenu renameValue = new JMenu("Rename value"); popupMenu.add(renameValue); for (int iValue = 0; iValue < outcomes.length; iValue++) { item = new JMenuItem(outcomes[iValue]); item.addActionListener(renameValueAction); renameValue.add(item); } JMenu delValue = new JMenu("Delete value"); popupMenu.add(delValue); if (m_BayesNet.getCardinality(m_nCurrentNode) <= 2) { delValue.setEnabled(false); } for (int iValue = 0; iValue < outcomes.length; iValue++) { JMenuItem delValueItem = new JMenuItem(outcomes[iValue]); delValueItem.addActionListener(delValueAction); delValue.add(delValueItem); } popupMenu.setLocation(me.getX(), me.getY()); popupMenu.show(m_GraphPanel, me.getX(), me.getY()); } catch (Exception e) { e.printStackTrace(); } } // handleRightNodeClick } // class GraphVisualizerMouseListener /** * private class for handling mouseMoved events to highlight nodes if the * the mouse is moved on one, move it around or move selection around */ private class GraphVisualizerMouseMotionListener extends MouseMotionAdapter { /* last node moved over. Used for turning highlight on and off */ int m_nLastNode = -1; /* current mouse position clicked */ int m_nPosX, m_nPosY; /* identify the node under the mouse * @returns node index of node under mouse, or -1 if there is no such node */ int getGraphNode(MouseEvent me) { m_nPosX = m_nPosY = 0; Rectangle r = new Rectangle(0, 0, (int) (m_nPaddedNodeWidth * m_fScale), (int) (m_nNodeHeight * m_fScale)); m_nPosX += me.getX(); m_nPosY += me.getY(); for (int iNode = 0; iNode < m_BayesNet.getNrOfNodes(); iNode++) { r.x = (int) (m_BayesNet.getPositionX(iNode) * m_fScale); r.y = (int) (m_BayesNet.getPositionY(iNode) * m_fScale); if (r.contains(m_nPosX, m_nPosY)) { return iNode; } } return -1; } // getGraphNode /* handle mouse dragging event * (non-Javadoc) * @see java.awt.event.MouseMotionListener#mouseDragged(java.awt.event.MouseEvent) */ public void mouseDragged(MouseEvent me) { if (m_nSelectedRect != null) { m_nSelectedRect.width = me.getPoint().x - m_nSelectedRect.x; m_nSelectedRect.height = me.getPoint().y - m_nSelectedRect.y; repaint(); return; } int iNode = getGraphNode(me); if (iNode >= 0) { if (m_Selection.getSelected().size() > 0) { if (m_Selection.getSelected().contains(iNode)) { m_BayesNet.setPosition(iNode, (int) ((m_nPosX / m_fScale - m_nPaddedNodeWidth / 2)), (int) ((m_nPosY / m_fScale - m_nNodeHeight / 2)), m_Selection.getSelected()); } else { m_Selection.clear(); m_BayesNet.setPosition(iNode, (int) ((m_nPosX / m_fScale - m_nPaddedNodeWidth / 2)), (int) ((m_nPosY / m_fScale - m_nNodeHeight / 2))); } repaint(); } else { m_BayesNet.setPosition(iNode, (int) ((m_nPosX / m_fScale - m_nPaddedNodeWidth / 2)), (int) ((m_nPosY / m_fScale - m_nNodeHeight / 2))); } m_jStatusBar.setText(m_BayesNet.lastActionMsg()); a_undo.setEnabled(true); a_redo.setEnabled(false); m_GraphPanel.highLight(iNode); } if (iNode < 0) { if (m_nLastNode >= 0) { m_GraphPanel.repaint(); m_nLastNode = -1; } else { m_nSelectedRect = new Rectangle(me.getPoint().x, me.getPoint().y, 1, 1); m_GraphPanel.repaint(); } } } // mouseDragged /* handles mouse move event * (non-Javadoc) * @see java.awt.event.MouseMotionListener#mouseMoved(java.awt.event.MouseEvent) */ public void mouseMoved(MouseEvent me) { int iNode = getGraphNode(me); if (iNode >= 0) { if (iNode != m_nLastNode) { m_GraphPanel.highLight(iNode); if (m_nLastNode >= 0) { m_GraphPanel.highLight(m_nLastNode); } m_nLastNode = iNode; } } if (iNode < 0 && m_nLastNode >= 0) { m_GraphPanel.repaint(); m_nLastNode = -1; } } // mouseMoved } // class GraphVisualizerMouseMotionListener /* apply graph layout algorithm to Bayesian network */ void layoutGraph() { if (m_BayesNet.getNrOfNodes() == 0) { return; } try { FastVector m_nodes = new FastVector(); FastVector m_edges = new FastVector(); BIFParser bp = new BIFParser(m_BayesNet.toXMLBIF03(), m_nodes, m_edges); bp.parse(); updateStatus(); m_layoutEngine = new HierarchicalBCEngine(m_nodes, m_edges, m_nPaddedNodeWidth, m_nNodeHeight); m_layoutEngine.addLayoutCompleteEventListener(this); m_layoutEngine.layoutGraph(); } catch (Exception e) { e.printStackTrace(); } } // layoutGraph /* Update status of various items that need regular updating * such as enabled status of some menu items, marginal distributions * if shown, repainting of graph. */ void updateStatus() { a_undo.setEnabled(m_BayesNet.canUndo()); a_redo.setEnabled(m_BayesNet.canRedo()); a_datagenerator.setEnabled(m_BayesNet.getNrOfNodes() > 0); if (!m_bViewMargins && !m_bViewCliques) { repaint(); return; } try { m_marginCalculator = new MarginCalculator(); m_marginCalculator.calcMargins(m_BayesNet); SerializedObject so = new SerializedObject(m_marginCalculator); m_marginCalculatorWithEvidence = (MarginCalculator) so.getObject(); for (int iNode = 0; iNode < m_BayesNet.getNrOfNodes(); iNode++) { if (m_BayesNet.getEvidence(iNode) >= 0) { m_marginCalculatorWithEvidence.setEvidence(iNode, m_BayesNet.getEvidence(iNode)); } } for (int iNode = 0; iNode < m_BayesNet.getNrOfNodes(); iNode++) { m_BayesNet.setMargin(iNode, m_marginCalculatorWithEvidence.getMargin(iNode)); } } catch (Exception e) { e.printStackTrace(); } repaint(); } // updateStatus /* add arc with node iChild as child. * This pops up a selection list with potential parents for the child. * All decendants and current parents are excluded from the list as is * the child node itself. * @param iChild index of the node for which to add an arc */ void addArcInto(int iChild) { String sChild = m_BayesNet.getNodeName(iChild); try { int nNodes = m_BayesNet.getNrOfNodes(); boolean[] isNotAllowedAsParent = new boolean[nNodes]; // prevent it being a parent of itself isNotAllowedAsParent[iChild] = true; // prevent a descendant being a parent, since it introduces cycles for (int i = 0; i < nNodes; i++) { for (int iNode = 0; iNode < nNodes; iNode++) { for (int iParent = 0; iParent < m_BayesNet.getNrOfParents(iNode); iParent++) { if (isNotAllowedAsParent[m_BayesNet.getParent(iNode, iParent)]) { isNotAllowedAsParent[iNode] = true; } } } } // prevent nodes that are already a parent for (int iParent = 0; iParent < m_BayesNet.getNrOfParents(iChild); iParent++) { isNotAllowedAsParent[m_BayesNet.getParent(iChild, iParent)] = true; } // count nr of remaining candidates int nCandidates = 0; for (int i = 0; i < nNodes; i++) { if (!isNotAllowedAsParent[i]) { nCandidates++; } } if (nCandidates == 0) { JOptionPane.showMessageDialog(null, "No potential parents available for this node (" + sChild + "). Choose another node as child node."); return; } String[] options = new String[nCandidates]; int k = 0; for (int i = 0; i < nNodes; i++) { if (!isNotAllowedAsParent[i]) { options[k++] = m_BayesNet.getNodeName(i); } } String sParent = (String) JOptionPane.showInputDialog(null, "Select parent node for " + sChild, "Nodes", 0, null, options, options[0]); if (sParent == null || sParent.equals("")) { return; } // update all data structures m_BayesNet.addArc(sParent, sChild); m_jStatusBar.setText(m_BayesNet.lastActionMsg()); updateStatus(); } catch (Exception e) { e.printStackTrace(); } } // addArcInto /* deletes arc from node with name sParent into child with index iChild * */ void deleteArc(int iChild, String sParent) { try { m_BayesNet.deleteArc(m_BayesNet.getNode(sParent), iChild); m_jStatusBar.setText(m_BayesNet.lastActionMsg()); } catch (Exception e) { e.printStackTrace(); } updateStatus(); } // deleteArc /* deletes arc from node with index iParent into child with name sChild * */ void deleteArc(String sChild, int iParent) { try { m_BayesNet.deleteArc(iParent, m_BayesNet.getNode(sChild)); m_jStatusBar.setText(m_BayesNet.lastActionMsg()); } catch (Exception e) { e.printStackTrace(); } updateStatus(); } // deleteArc /* deletes arc. Pops up list of arcs listed in 'options' as * " -> ". */ void deleteArc(String[] options) { String sResult = (String) JOptionPane.showInputDialog(null, "Select arc to delete", "Arcs", 0, null, options, options[0]); if (sResult != null && !sResult.equals("")) { int nPos = sResult.indexOf(" -> "); String sParent = sResult.substring(0, nPos); String sChild = sResult.substring(nPos + 4); try { m_BayesNet.deleteArc(sParent, sChild); m_jStatusBar.setText(m_BayesNet.lastActionMsg()); } catch (Exception e) { e.printStackTrace(); } updateStatus(); } } // deleteArc /* Rename node with index nTargetNode. * Pops up window that allwos for entering a new name. */ void renameNode(int nTargetNode) { String sName = (String) JOptionPane.showInputDialog(null, m_BayesNet.getNodeName(nTargetNode), "New name for node", JOptionPane.OK_CANCEL_OPTION); if (sName == null || sName.equals("")) { return; } try { while (m_BayesNet.getNode2(sName) >= 0) { sName = (String) JOptionPane.showInputDialog(null, "Cannot rename to " + sName + ".\nNode with that name already exists."); if (sName == null || sName.equals("")) { return; } } m_BayesNet.setNodeName(nTargetNode, sName); m_jStatusBar.setText(m_BayesNet.lastActionMsg()); } catch (Exception e) { e.printStackTrace(); } repaint(); } // renameNode /* Rename value with name sValeu of a node with index nTargetNode. * Pops up window that allows entering a new name. */ void renameValue(int nTargetNode, String sValue) { String sNewValue = (String) JOptionPane.showInputDialog(null, "New name for value " + sValue, "Node " + m_BayesNet.getNodeName(nTargetNode), JOptionPane.OK_CANCEL_OPTION); if (sNewValue == null || sNewValue.equals("")) { return; } m_BayesNet.renameNodeValue(nTargetNode, sValue, sNewValue); m_jStatusBar.setText(m_BayesNet.lastActionMsg()); a_undo.setEnabled(true); a_redo.setEnabled(false); repaint(); } // renameValue /* delete a single node with index iNode */ void deleteNode(int iNode) { try { m_BayesNet.deleteNode(iNode); m_jStatusBar.setText(m_BayesNet.lastActionMsg()); } catch (Exception e) { e.printStackTrace(); } updateStatus(); } // deleteNode /* Add a value to currently selected node. * Shows window that allows to enter the name of the value. */ void addValue() { //GraphNode n = (GraphNode) m_nodes.elementAt(m_nCurrentNode); String sValue = "Value" + (m_BayesNet.getCardinality(m_nCurrentNode) + 1); String sNewValue = (String) JOptionPane.showInputDialog(null, "New value " + sValue, "Node " + m_BayesNet.getNodeName(m_nCurrentNode), JOptionPane.OK_CANCEL_OPTION); if (sNewValue == null || sNewValue.equals("")) { return; } try { m_BayesNet.addNodeValue(m_nCurrentNode, sNewValue); m_jStatusBar.setText(m_BayesNet.lastActionMsg()); //n.outcomes = m_BayesNet.getValues(m_nCurrentNode); //for (int iNode = 0; iNode < m_BayesNet.getNrOfNodes(); iNode++) { // n = (GraphNode) m_nodes.elementAt(iNode); // n.probs = m_BayesNet.getDistribution(iNode); //} } catch (Exception e) { e.printStackTrace(); } updateStatus(); } // addValue /* remove value with name sValue from the node with index nTargetNode */ void delValue(int nTargetNode, String sValue) { try { m_BayesNet.delNodeValue(nTargetNode, sValue); m_jStatusBar.setText(m_BayesNet.lastActionMsg()); } catch (Exception e) { e.printStackTrace(); } updateStatus(); } // delValue /* Edits CPT of node with index nTargetNode. * Pops up table with probability table that the user can change or just view. */ void editCPT(int nTargetNode) { m_nCurrentNode = nTargetNode; final GraphVisualizerTableModel tm = new GraphVisualizerTableModel(nTargetNode); JTable jTblProbs = new JTable(tm); JScrollPane js = new JScrollPane(jTblProbs); int nParents = m_BayesNet.getNrOfParents(nTargetNode); if (nParents > 0) { GridBagConstraints gbc = new GridBagConstraints(); JPanel jPlRowHeader = new JPanel(new GridBagLayout()); // indices of the parent nodes in the Vector int[] idx = new int[nParents]; // max length of values of each parent int[] lengths = new int[nParents]; // Adding labels for rows gbc.anchor = GridBagConstraints.NORTHWEST; gbc.fill = GridBagConstraints.HORIZONTAL; gbc.insets = new Insets(0, 1, 0, 0); int addNum = 0, temp = 0; boolean dark = false; while (true) { gbc.gridwidth = 1; for (int k = 0; k < nParents; k++) { int iParent2 = m_BayesNet.getParent(nTargetNode, k); JLabel lb = new JLabel(m_BayesNet.getValueName(iParent2,idx[k])); lb.setFont(new Font("Dialog", Font.PLAIN, 12)); lb.setOpaque(true); lb.setBorder(BorderFactory.createEmptyBorder(1, 2, 1, 1)); lb.setHorizontalAlignment(JLabel.CENTER); if (dark) { lb.setBackground(lb.getBackground().darker()); lb.setForeground(Color.white); } else lb.setForeground(Color.black); temp = lb.getPreferredSize().width; lb.setPreferredSize(new Dimension(temp, jTblProbs.getRowHeight())); if (lengths[k] < temp) lengths[k] = temp; temp = 0; if (k == nParents - 1) { gbc.gridwidth = GridBagConstraints.REMAINDER; dark = (dark == true) ? false : true; } jPlRowHeader.add(lb, gbc); addNum++; } for (int k = nParents - 1; k >= 0; k--) { int iParent2 = m_BayesNet.getParent(m_nCurrentNode, k); if (idx[k] == m_BayesNet.getCardinality(iParent2) - 1 && k != 0) { idx[k] = 0; continue; } else { idx[k]++; break; } } int iParent2 = m_BayesNet.getParent(m_nCurrentNode, 0); if (idx[0] == m_BayesNet.getCardinality(iParent2)) { JLabel lb = (JLabel) jPlRowHeader.getComponent(addNum - 1); jPlRowHeader.remove(addNum - 1); lb.setPreferredSize(new Dimension(lb.getPreferredSize().width, jTblProbs .getRowHeight())); gbc.gridwidth = GridBagConstraints.REMAINDER; gbc.weighty = 1; jPlRowHeader.add(lb, gbc); gbc.weighty = 0; break; } } gbc.gridwidth = 1; // The following panel contains the names of the // parents // and is displayed above the row names to identify // which value belongs to which parent JPanel jPlRowNames = new JPanel(new GridBagLayout()); for (int j = 0; j < nParents; j++) { JLabel lb2; JLabel lb1 = new JLabel(m_BayesNet.getNodeName(m_BayesNet.getParent(nTargetNode, j))); lb1.setBorder(BorderFactory.createEmptyBorder(1, 2, 1, 1)); Dimension tempd = lb1.getPreferredSize(); if (tempd.width < lengths[j]) { lb1.setPreferredSize(new Dimension(lengths[j], tempd.height)); lb1.setHorizontalAlignment(JLabel.CENTER); lb1.setMinimumSize(new Dimension(lengths[j], tempd.height)); } else if (tempd.width > lengths[j]) { lb2 = (JLabel) jPlRowHeader.getComponent(j); lb2.setPreferredSize(new Dimension(tempd.width, lb2.getPreferredSize().height)); } jPlRowNames.add(lb1, gbc); } js.setRowHeaderView(jPlRowHeader); js.setCorner(JScrollPane.UPPER_LEFT_CORNER, jPlRowNames); } final JDialog dlg = new JDialog((Frame) GUI.this.getTopLevelAncestor(), "Probability Distribution Table For " + m_BayesNet.getNodeName(nTargetNode), true); dlg.setSize(500, 400); dlg.setLocation(GUI.this.getLocation().x + GUI.this.getWidth() / 2 - 250, GUI.this.getLocation().y + GUI.this.getHeight() / 2 - 200); dlg.getContentPane().setLayout(new BorderLayout()); dlg.getContentPane().add(js, BorderLayout.CENTER); JButton jBtRandomize = new JButton("Randomize"); jBtRandomize.setMnemonic('R'); jBtRandomize.addActionListener(new ActionListener() { public void actionPerformed(ActionEvent ae) { tm.randomize(); dlg.repaint(); } }); JButton jBtOk = new JButton("Ok"); jBtOk.setMnemonic('O'); jBtOk.addActionListener(new ActionListener() { public void actionPerformed(ActionEvent ae) { tm.setData(); try { m_BayesNet.setDistribution(m_nCurrentNode, tm.m_fProbs); m_jStatusBar.setText(m_BayesNet.lastActionMsg()); updateStatus(); } catch (Exception e) { e.printStackTrace(); } dlg.setVisible(false); } }); JButton jBtCancel = new JButton("Cancel"); jBtCancel.setMnemonic('C'); jBtCancel.addActionListener(new ActionListener() { public void actionPerformed(ActionEvent ae) { dlg.setVisible(false); } }); Container c = new Container(); c.setLayout(new GridBagLayout()); c.add(jBtRandomize); c.add(jBtOk); c.add(jBtCancel); dlg.getContentPane().add(c, BorderLayout.SOUTH); dlg.setVisible(true); } // editCPT /** * Main method. Builds up menus and reads from file if one is specified. */ public static void main(String[] args) { weka.core.logging.Logger.log(weka.core.logging.Logger.Level.INFO, "Logging started"); LookAndFeel.setLookAndFeel(); JFrame jf = new JFrame("Bayes Network Editor"); final GUI g = new GUI(); JMenuBar menuBar = g.getMenuBar(); if (args.length>0) { try { g.readBIFFromFile(args[0]); } catch (IOException ex) { ex.printStackTrace(); } catch (BIFFormatException bf) { bf.printStackTrace(); System.exit(-1); } } jf.setJMenuBar(menuBar); jf.getContentPane().add(g); jf.setDefaultCloseOperation(JFrame.EXIT_ON_CLOSE); jf.setSize(800, 600); jf.setVisible(true); g.m_Selection.updateGUI(); GenericObjectEditor.registerEditors(); } // main } // end of class




© 2015 - 2025 Weber Informatics LLC | Privacy Policy