
weka.gui.explorer.ClustererPanel Maven / Gradle / Ivy
Go to download
The Waikato Environment for Knowledge Analysis (WEKA), a machine
learning workbench. This is the stable version. Apart from bugfixes, this version
does not receive any other updates.
/*
* 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.
*/
/*
* ClustererPanel.java
* Copyright (C) 1999 University of Waikato, Hamilton, New Zealand
*
*/
package weka.gui.explorer;
import java.awt.BorderLayout;
import java.awt.Dimension;
import java.awt.Font;
import java.awt.GridBagConstraints;
import java.awt.GridBagLayout;
import java.awt.GridLayout;
import java.awt.Insets;
import java.awt.Point;
import java.awt.event.ActionEvent;
import java.awt.event.ActionListener;
import java.awt.event.InputEvent;
import java.awt.event.MouseAdapter;
import java.awt.event.MouseEvent;
import java.beans.PropertyChangeEvent;
import java.beans.PropertyChangeListener;
import java.io.File;
import java.io.FileInputStream;
import java.io.FileOutputStream;
import java.io.InputStream;
import java.io.ObjectInputStream;
import java.io.ObjectOutputStream;
import java.io.OutputStream;
import java.text.SimpleDateFormat;
import java.util.Date;
import java.util.Random;
import java.util.zip.GZIPInputStream;
import java.util.zip.GZIPOutputStream;
import javax.swing.BorderFactory;
import javax.swing.ButtonGroup;
import javax.swing.DefaultComboBoxModel;
import javax.swing.DefaultListModel;
import javax.swing.JButton;
import javax.swing.JCheckBox;
import javax.swing.JComboBox;
import javax.swing.JFileChooser;
import javax.swing.JFrame;
import javax.swing.JLabel;
import javax.swing.JList;
import javax.swing.JMenuItem;
import javax.swing.JOptionPane;
import javax.swing.JPanel;
import javax.swing.JPopupMenu;
import javax.swing.JRadioButton;
import javax.swing.JScrollPane;
import javax.swing.JTextArea;
import javax.swing.JTextField;
import javax.swing.JViewport;
import javax.swing.SwingConstants;
import javax.swing.event.ChangeEvent;
import javax.swing.event.ChangeListener;
import javax.swing.filechooser.FileFilter;
import weka.clusterers.ClusterEvaluation;
import weka.clusterers.Clusterer;
import weka.core.Attribute;
import weka.core.Capabilities;
import weka.core.CapabilitiesHandler;
import weka.core.Drawable;
import weka.core.FastVector;
import weka.core.Instance;
import weka.core.Instances;
import weka.core.OptionHandler;
import weka.core.SerializedObject;
import weka.core.Utils;
import weka.filters.Filter;
import weka.filters.unsupervised.attribute.Remove;
import weka.gui.ExtensionFileFilter;
import weka.gui.GenericObjectEditor;
import weka.gui.InstancesSummaryPanel;
import weka.gui.ListSelectorDialog;
import weka.gui.Logger;
import weka.gui.PropertyPanel;
import weka.gui.ResultHistoryPanel;
import weka.gui.SaveBuffer;
import weka.gui.SetInstancesPanel;
import weka.gui.SysErrLog;
import weka.gui.TaskLogger;
import weka.gui.explorer.Explorer.CapabilitiesFilterChangeEvent;
import weka.gui.explorer.Explorer.CapabilitiesFilterChangeListener;
import weka.gui.explorer.Explorer.ExplorerPanel;
import weka.gui.explorer.Explorer.LogHandler;
import weka.gui.hierarchyvisualizer.HierarchyVisualizer;
import weka.gui.treevisualizer.PlaceNode2;
import weka.gui.treevisualizer.TreeVisualizer;
import weka.gui.visualize.Plot2D;
import weka.gui.visualize.PlotData2D;
import weka.gui.visualize.VisualizePanel;
/**
* This panel allows the user to select and configure a clusterer, and evaluate
* the clusterer using a number of testing modes (test on the training data,
* train/test on a percentage split, test on a separate split). The results of
* clustering runs are stored in a result history so that previous results are
* accessible.
*
* @author Mark Hall ([email protected])
* @author Richard Kirkby ([email protected])
* @version $Revision: 9728 $
*/
public class ClustererPanel extends JPanel implements
CapabilitiesFilterChangeListener, ExplorerPanel, LogHandler {
/** for serialization */
static final long serialVersionUID = -2474932792950820990L;
/** the parent frame */
protected Explorer m_Explorer = null;
/** The filename extension that should be used for model files */
public static String MODEL_FILE_EXTENSION = ".model";
/** Lets the user configure the clusterer */
protected GenericObjectEditor m_ClustererEditor = new GenericObjectEditor();
/** The panel showing the current clusterer selection */
protected PropertyPanel m_CLPanel = new PropertyPanel(m_ClustererEditor);
/** The output area for classification results */
protected JTextArea m_OutText = new JTextArea(20, 40);
/** The destination for log/status messages */
protected Logger m_Log = new SysErrLog();
/** The buffer saving object for saving output */
SaveBuffer m_SaveOut = new SaveBuffer(m_Log, this);
/** A panel controlling results viewing */
protected ResultHistoryPanel m_History = new ResultHistoryPanel(m_OutText);
/** Click to set test mode to generate a % split */
protected JRadioButton m_PercentBut = new JRadioButton(Messages.getInstance()
.getString("ClustererPanel_PercentBut_JRadioButton_Text"));
/** Click to set test mode to test on training data */
protected JRadioButton m_TrainBut = new JRadioButton(Messages.getInstance()
.getString("ClustererPanel_TrainBut_JRadioButton_Text"));
/** Click to set test mode to a user-specified test set */
protected JRadioButton m_TestSplitBut = new JRadioButton(Messages
.getInstance().getString("ClustererPanel_TestSplitBut_JRadioButton_Text"));
/** Click to set test mode to classes to clusters based evaluation */
protected JRadioButton m_ClassesToClustersBut = new JRadioButton(Messages
.getInstance().getString(
"ClustererPanel_ClassesToClustersBut_JRadioButton_Text"));
/**
* Lets the user select the class column for classes to clusters based
* evaluation
*/
protected JComboBox m_ClassCombo = new JComboBox();
/** Label by where the % split is entered */
protected JLabel m_PercentLab = new JLabel("%", SwingConstants.RIGHT);
/** The field where the % split is entered */
protected JTextField m_PercentText = new JTextField("66");
/** The button used to open a separate test dataset */
protected JButton m_SetTestBut = new JButton(Messages.getInstance()
.getString("ClustererPanel_SetTestBut_JButton_Text"));
/** The frame used to show the test set selection panel */
protected JFrame m_SetTestFrame;
/**
* The button used to popup a list for choosing attributes to ignore while
* clustering
*/
protected JButton m_ignoreBut = new JButton(Messages.getInstance().getString(
"ClustererPanel_IgnoreBut_JButton_Text"));
protected DefaultListModel m_ignoreKeyModel = new DefaultListModel();
protected JList m_ignoreKeyList = new JList(m_ignoreKeyModel);
// protected Remove m_ignoreFilter = null;
/**
* Alters the enabled/disabled status of elements associated with each radio
* button
*/
ActionListener m_RadioListener = new ActionListener() {
public void actionPerformed(ActionEvent e) {
updateRadioLinks();
}
};
/** Click to start running the clusterer */
protected JButton m_StartBut = new JButton(Messages.getInstance().getString(
"ClustererPanel_StartBut_JButton_Text"));
/** Stop the class combo from taking up to much space */
private final Dimension COMBO_SIZE = new Dimension(250,
m_StartBut.getPreferredSize().height);
/** Click to stop a running clusterer */
protected JButton m_StopBut = new JButton(Messages.getInstance().getString(
"ClustererPanel_StopBut_JButton_Text"));
/** The main set of instances we're playing with */
protected Instances m_Instances;
/** The user-supplied test set (if any) */
protected Instances m_TestInstances;
/** The current visualization object */
protected VisualizePanel m_CurrentVis = null;
/**
* Check to save the predictions in the results list for visualizing later on
*/
protected JCheckBox m_StorePredictionsBut = new JCheckBox(Messages
.getInstance().getString("ClustererPanel_StopBut_JCheckBox_Text"));
/** A thread that clustering runs in */
protected Thread m_RunThread;
/** The instances summary panel displayed by m_SetTestFrame */
protected InstancesSummaryPanel m_Summary;
/** Filter to ensure only model files are selected */
protected FileFilter m_ModelFilter = new ExtensionFileFilter(
MODEL_FILE_EXTENSION, Messages.getInstance().getString(
"ClustererPanel_ModelFilter_ExtensionFileFilter_Text"));
/** The file chooser for selecting model files */
protected JFileChooser m_FileChooser = new JFileChooser(new File(
System.getProperty("user.dir")));
/* Register the property editors we need */
static {
GenericObjectEditor.registerEditors();
}
/**
* Creates the clusterer panel
*/
public ClustererPanel() {
// Connect / configure the components
m_OutText.setEditable(false);
m_OutText.setFont(new Font("Monospaced", Font.PLAIN, 12));
m_OutText.setBorder(BorderFactory.createEmptyBorder(5, 5, 5, 5));
m_OutText.addMouseListener(new MouseAdapter() {
@Override
public void mouseClicked(MouseEvent e) {
if ((e.getModifiers() & InputEvent.BUTTON1_MASK) != InputEvent.BUTTON1_MASK) {
m_OutText.selectAll();
}
}
});
m_History.setBorder(BorderFactory.createTitledBorder(Messages.getInstance()
.getString(
"ClustererPanel_History_BorderFactoryCreateTitledBorder_Text")));
m_ClustererEditor.setClassType(Clusterer.class);
m_ClustererEditor.setValue(ExplorerDefaults.getClusterer());
m_ClustererEditor.addPropertyChangeListener(new PropertyChangeListener() {
public void propertyChange(PropertyChangeEvent e) {
m_StartBut.setEnabled(true);
Capabilities currentFilter = m_ClustererEditor.getCapabilitiesFilter();
Clusterer clusterer = (Clusterer) m_ClustererEditor.getValue();
Capabilities currentSchemeCapabilities = null;
if (clusterer != null && currentFilter != null
&& (clusterer instanceof CapabilitiesHandler)) {
currentSchemeCapabilities = ((CapabilitiesHandler) clusterer)
.getCapabilities();
if (!currentSchemeCapabilities.supportsMaybe(currentFilter)
&& !currentSchemeCapabilities.supports(currentFilter)) {
m_StartBut.setEnabled(false);
}
}
repaint();
}
});
m_TrainBut.setToolTipText(Messages.getInstance().getString(
"ClustererPanel_TrainBut_SetToolTipText_Text"));
m_PercentBut.setToolTipText(Messages.getInstance().getString(
"ClustererPanel_PercentBut_SetToolTipText_Text"));
m_TestSplitBut.setToolTipText(Messages.getInstance().getString(
"ClustererPanel_TestSplitBut_SetToolTipText_Text"));
m_ClassesToClustersBut.setToolTipText(Messages.getInstance().getString(
"ClustererPanel_ClassesToClustersBut_SetToolTipText_Text"));
m_ClassCombo.setToolTipText(Messages.getInstance().getString(
"ClustererPanel_ClassCombo_SetToolTipText_Text"));
m_StartBut.setToolTipText(Messages.getInstance().getString(
"ClustererPanel_StartBut_SetToolTipText_Text"));
m_StopBut.setToolTipText(Messages.getInstance().getString(
"ClustererPanel_StartBut_StopBut_Text"));
m_StorePredictionsBut.setToolTipText(Messages.getInstance().getString(
"ClustererPanel_StorePredictionsBut_SetToolTipText_Text"));
m_ignoreBut.setToolTipText(Messages.getInstance().getString(
"ClustererPanel_IgnoreBut_SetToolTipText_Text"));
m_FileChooser.setFileFilter(m_ModelFilter);
m_FileChooser.setFileSelectionMode(JFileChooser.FILES_ONLY);
m_ClassCombo.setPreferredSize(COMBO_SIZE);
m_ClassCombo.setMaximumSize(COMBO_SIZE);
m_ClassCombo.setMinimumSize(COMBO_SIZE);
m_ClassCombo.setEnabled(false);
m_PercentBut.setSelected(ExplorerDefaults.getClustererTestMode() == 2);
m_TrainBut.setSelected(ExplorerDefaults.getClustererTestMode() == 3);
m_TestSplitBut.setSelected(ExplorerDefaults.getClustererTestMode() == 4);
m_ClassesToClustersBut
.setSelected(ExplorerDefaults.getClustererTestMode() == 5);
m_StorePredictionsBut.setSelected(ExplorerDefaults
.getClustererStoreClustersForVis());
updateRadioLinks();
ButtonGroup bg = new ButtonGroup();
bg.add(m_TrainBut);
bg.add(m_PercentBut);
bg.add(m_TestSplitBut);
bg.add(m_ClassesToClustersBut);
m_TrainBut.addActionListener(m_RadioListener);
m_PercentBut.addActionListener(m_RadioListener);
m_TestSplitBut.addActionListener(m_RadioListener);
m_ClassesToClustersBut.addActionListener(m_RadioListener);
m_SetTestBut.addActionListener(new ActionListener() {
public void actionPerformed(ActionEvent e) {
setTestSet();
}
});
m_StartBut.setEnabled(false);
m_StopBut.setEnabled(false);
m_ignoreBut.setEnabled(false);
m_StartBut.addActionListener(new ActionListener() {
public void actionPerformed(ActionEvent e) {
boolean proceed = true;
if (Explorer.m_Memory.memoryIsLow()) {
proceed = Explorer.m_Memory.showMemoryIsLow();
}
if (proceed) {
startClusterer();
}
}
});
m_StopBut.addActionListener(new ActionListener() {
public void actionPerformed(ActionEvent e) {
stopClusterer();
}
});
m_ignoreBut.addActionListener(new ActionListener() {
public void actionPerformed(ActionEvent e) {
setIgnoreColumns();
}
});
m_History.setHandleRightClicks(false);
// see if we can popup a menu for the selected result
m_History.getList().addMouseListener(new MouseAdapter() {
@Override
public void mouseClicked(MouseEvent e) {
if (((e.getModifiers() & InputEvent.BUTTON1_MASK) != InputEvent.BUTTON1_MASK)
|| e.isAltDown()) {
int index = m_History.getList().locationToIndex(e.getPoint());
if (index != -1) {
String name = m_History.getNameAtIndex(index);
visualizeClusterer(name, e.getX(), e.getY());
} else {
visualizeClusterer(null, e.getX(), e.getY());
}
}
}
});
m_ClassCombo.addActionListener(new ActionListener() {
public void actionPerformed(ActionEvent e) {
updateCapabilitiesFilter(m_ClustererEditor.getCapabilitiesFilter());
}
});
// Layout the GUI
JPanel p1 = new JPanel();
p1.setBorder(BorderFactory.createCompoundBorder(
BorderFactory.createTitledBorder(Messages.getInstance().getString(
"ClustererPanel_P1_BorderFactoryCreateTitledBorder_Text")),
BorderFactory.createEmptyBorder(0, 5, 5, 5)));
p1.setLayout(new BorderLayout());
p1.add(m_CLPanel, BorderLayout.NORTH);
JPanel p2 = new JPanel();
GridBagLayout gbL = new GridBagLayout();
p2.setLayout(gbL);
p2.setBorder(BorderFactory.createCompoundBorder(
BorderFactory.createTitledBorder(Messages.getInstance().getString(
"ClustererPanel_P2_BorderFactoryCreateTitledBorder_Text")),
BorderFactory.createEmptyBorder(0, 5, 5, 5)));
GridBagConstraints gbC = new GridBagConstraints();
gbC.anchor = GridBagConstraints.WEST;
gbC.gridy = 0;
gbC.gridx = 0;
gbL.setConstraints(m_TrainBut, gbC);
p2.add(m_TrainBut);
gbC = new GridBagConstraints();
gbC.anchor = GridBagConstraints.WEST;
gbC.gridy = 1;
gbC.gridx = 0;
gbL.setConstraints(m_TestSplitBut, gbC);
p2.add(m_TestSplitBut);
gbC = new GridBagConstraints();
gbC.anchor = GridBagConstraints.EAST;
gbC.fill = GridBagConstraints.HORIZONTAL;
gbC.gridy = 1;
gbC.gridx = 1;
gbC.gridwidth = 2;
gbC.insets = new Insets(2, 10, 2, 0);
gbL.setConstraints(m_SetTestBut, gbC);
p2.add(m_SetTestBut);
gbC = new GridBagConstraints();
gbC.anchor = GridBagConstraints.WEST;
gbC.gridy = 2;
gbC.gridx = 0;
gbL.setConstraints(m_PercentBut, gbC);
p2.add(m_PercentBut);
gbC = new GridBagConstraints();
gbC.anchor = GridBagConstraints.EAST;
gbC.fill = GridBagConstraints.HORIZONTAL;
gbC.gridy = 2;
gbC.gridx = 1;
gbC.insets = new Insets(2, 10, 2, 10);
gbL.setConstraints(m_PercentLab, gbC);
p2.add(m_PercentLab);
gbC = new GridBagConstraints();
gbC.anchor = GridBagConstraints.EAST;
gbC.fill = GridBagConstraints.HORIZONTAL;
gbC.gridy = 2;
gbC.gridx = 2;
gbC.weightx = 100;
gbC.ipadx = 20;
gbL.setConstraints(m_PercentText, gbC);
p2.add(m_PercentText);
gbC = new GridBagConstraints();
gbC.anchor = GridBagConstraints.WEST;
gbC.gridy = 3;
gbC.gridx = 0;
gbC.gridwidth = 2;
gbL.setConstraints(m_ClassesToClustersBut, gbC);
p2.add(m_ClassesToClustersBut);
m_ClassCombo.setBorder(BorderFactory.createEmptyBorder(0, 20, 0, 0));
gbC = new GridBagConstraints();
gbC.anchor = GridBagConstraints.WEST;
gbC.gridy = 4;
gbC.gridx = 0;
gbC.gridwidth = 2;
gbL.setConstraints(m_ClassCombo, gbC);
p2.add(m_ClassCombo);
gbC = new GridBagConstraints();
gbC.anchor = GridBagConstraints.WEST;
gbC.gridy = 5;
gbC.gridx = 0;
gbC.gridwidth = 2;
gbL.setConstraints(m_StorePredictionsBut, gbC);
p2.add(m_StorePredictionsBut);
JPanel buttons = new JPanel();
buttons.setLayout(new GridLayout(2, 1));
JPanel ssButs = new JPanel();
ssButs.setBorder(BorderFactory.createEmptyBorder(5, 5, 5, 5));
ssButs.setLayout(new GridLayout(1, 2, 5, 5));
ssButs.add(m_StartBut);
ssButs.add(m_StopBut);
JPanel ib = new JPanel();
ib.setBorder(BorderFactory.createEmptyBorder(5, 5, 5, 5));
ib.setLayout(new GridLayout(1, 1, 5, 5));
ib.add(m_ignoreBut);
buttons.add(ib);
buttons.add(ssButs);
JPanel p3 = new JPanel();
p3.setBorder(BorderFactory.createTitledBorder(Messages.getInstance()
.getString("ClustererPanel_P3_BorderFactoryCreateTitledBorder_Text")));
p3.setLayout(new BorderLayout());
final JScrollPane js = new JScrollPane(m_OutText);
p3.add(js, BorderLayout.CENTER);
js.getViewport().addChangeListener(new ChangeListener() {
private int lastHeight;
public void stateChanged(ChangeEvent e) {
JViewport vp = (JViewport) e.getSource();
int h = vp.getViewSize().height;
if (h != lastHeight) { // i.e. an addition not just a user scrolling
lastHeight = h;
int x = h - vp.getExtentSize().height;
vp.setViewPosition(new Point(0, x));
}
}
});
JPanel mondo = new JPanel();
gbL = new GridBagLayout();
mondo.setLayout(gbL);
gbC = new GridBagConstraints();
// gbC.anchor = GridBagConstraints.WEST;
gbC.fill = GridBagConstraints.HORIZONTAL;
gbC.gridy = 0;
gbC.gridx = 0;
gbL.setConstraints(p2, gbC);
mondo.add(p2);
gbC = new GridBagConstraints();
gbC.anchor = GridBagConstraints.NORTH;
gbC.fill = GridBagConstraints.HORIZONTAL;
gbC.gridy = 1;
gbC.gridx = 0;
gbL.setConstraints(buttons, gbC);
mondo.add(buttons);
gbC = new GridBagConstraints();
// gbC.anchor = GridBagConstraints.NORTH;
gbC.fill = GridBagConstraints.BOTH;
gbC.gridy = 2;
gbC.gridx = 0;
gbC.weightx = 0;
gbL.setConstraints(m_History, gbC);
mondo.add(m_History);
gbC = new GridBagConstraints();
gbC.fill = GridBagConstraints.BOTH;
gbC.gridy = 0;
gbC.gridx = 1;
gbC.gridheight = 3;
gbC.weightx = 100;
gbC.weighty = 100;
gbL.setConstraints(p3, gbC);
mondo.add(p3);
setLayout(new BorderLayout());
add(p1, BorderLayout.NORTH);
add(mondo, BorderLayout.CENTER);
}
/**
* Updates the enabled status of the input fields and labels.
*/
protected void updateRadioLinks() {
m_SetTestBut.setEnabled(m_TestSplitBut.isSelected());
if ((m_SetTestFrame != null) && (!m_TestSplitBut.isSelected())) {
m_SetTestFrame.setVisible(false);
}
m_PercentText.setEnabled(m_PercentBut.isSelected());
m_PercentLab.setEnabled(m_PercentBut.isSelected());
m_ClassCombo.setEnabled(m_ClassesToClustersBut.isSelected());
updateCapabilitiesFilter(m_ClustererEditor.getCapabilitiesFilter());
}
/**
* Sets the Logger to receive informational messages
*
* @param newLog the Logger that will now get info messages
*/
public void setLog(Logger newLog) {
m_Log = newLog;
}
/**
* Tells the panel to use a new set of instances.
*
* @param inst a set of Instances
*/
public void setInstances(Instances inst) {
m_Instances = inst;
m_ignoreKeyModel.removeAllElements();
String[] attribNames = new String[m_Instances.numAttributes()];
for (int i = 0; i < m_Instances.numAttributes(); i++) {
String name = m_Instances.attribute(i).name();
m_ignoreKeyModel.addElement(name);
String type = "";
switch (m_Instances.attribute(i).type()) {
case Attribute.NOMINAL:
type = Messages.getInstance().getString(
"ClustererPanel_SetInstances_Type_AttributeNOMINAL_Text");
break;
case Attribute.NUMERIC:
type = Messages.getInstance().getString(
"ClustererPanel_SetInstances_Type_AttributeNUMERIC_Text");
break;
case Attribute.STRING:
type = Messages.getInstance().getString(
"ClustererPanel_SetInstances_Type_AttributeSTRING_Text");
break;
case Attribute.DATE:
type = Messages.getInstance().getString(
"ClustererPanel_SetInstances_Type_AttributeDATE_Text");
break;
case Attribute.RELATIONAL:
type = Messages.getInstance().getString(
"ClustererPanel_SetInstances_Type_AttributeRELATIONAL_Text");
break;
default:
type = Messages.getInstance().getString(
"ClustererPanel_SetInstances_Type_AttributeDEFAULT_Text");
}
String attnm = m_Instances.attribute(i).name();
attribNames[i] = type + attnm;
}
m_StartBut.setEnabled(m_RunThread == null);
m_StopBut.setEnabled(m_RunThread != null);
m_ignoreBut.setEnabled(true);
m_ClassCombo.setModel(new DefaultComboBoxModel(attribNames));
if (inst.classIndex() == -1)
m_ClassCombo.setSelectedIndex(attribNames.length - 1);
else
m_ClassCombo.setSelectedIndex(inst.classIndex());
updateRadioLinks();
}
/**
* Sets the user test set. Information about the current test set is displayed
* in an InstanceSummaryPanel and the user is given the ability to load
* another set from a file or url.
*
*/
protected void setTestSet() {
if (m_SetTestFrame == null) {
final SetInstancesPanel sp = new SetInstancesPanel();
sp.setReadIncrementally(false);
m_Summary = sp.getSummary();
if (m_TestInstances != null) {
sp.setInstances(m_TestInstances);
}
sp.addPropertyChangeListener(new PropertyChangeListener() {
public void propertyChange(PropertyChangeEvent e) {
m_TestInstances = sp.getInstances();
m_TestInstances.setClassIndex(-1); // make sure that no class
// attribute is set!
}
});
// Add propertychangelistener to update m_TestInstances whenever
// it changes in the settestframe
m_SetTestFrame = new JFrame(Messages.getInstance().getString(
"ClustererPanel_SetUpVisualizableInstances_JFrame_Text"));
sp.setParentFrame(m_SetTestFrame); // enable Close-Button
m_SetTestFrame.getContentPane().setLayout(new BorderLayout());
m_SetTestFrame.getContentPane().add(sp, BorderLayout.CENTER);
m_SetTestFrame.pack();
}
m_SetTestFrame.setVisible(true);
}
/**
* Sets up the structure for the visualizable instances. This dataset contains
* the original attributes plus the clusterer's cluster assignments
*
* @param testInstances the instances that the clusterer has clustered
* @param eval the evaluation to use
* @return a PlotData2D object encapsulating the visualizable instances. The
* instances contain one more attribute (predicted cluster) than the
* testInstances
*/
public static PlotData2D setUpVisualizableInstances(Instances testInstances,
ClusterEvaluation eval) throws Exception {
int numClusters = eval.getNumClusters();
double[] clusterAssignments = eval.getClusterAssignments();
FastVector hv = new FastVector();
Instances newInsts;
Attribute predictedCluster;
FastVector clustVals = new FastVector();
for (int i = 0; i < numClusters; i++) {
clustVals.addElement(Messages.getInstance().getString(
"ClustererPanel_SetUpVisualizableInstances_ClustVals_Text")
+ i);
}
predictedCluster = new Attribute(Messages.getInstance().getString(
"ClustererPanel_SetUpVisualizableInstances_PredictedCluster_Text"),
clustVals);
for (int i = 0; i < testInstances.numAttributes(); i++) {
hv.addElement(testInstances.attribute(i).copy());
}
hv.addElement(predictedCluster);
newInsts = new Instances(testInstances.relationName() + "_clustered", hv,
testInstances.numInstances());
double[] values;
int j;
int[] pointShapes = null;
int[] classAssignments = null;
if (testInstances.classIndex() >= 0) {
classAssignments = eval.getClassesToClusters();
pointShapes = new int[testInstances.numInstances()];
for (int i = 0; i < testInstances.numInstances(); i++) {
pointShapes[i] = Plot2D.CONST_AUTOMATIC_SHAPE;
}
}
for (int i = 0; i < testInstances.numInstances(); i++) {
values = new double[newInsts.numAttributes()];
for (j = 0; j < testInstances.numAttributes(); j++) {
values[j] = testInstances.instance(i).value(j);
}
if (clusterAssignments[i] < 0) {
values[j] = Instance.missingValue();
} else {
values[j] = clusterAssignments[i];
}
newInsts.add(new Instance(1.0, values));
if (pointShapes != null) {
if (clusterAssignments[i] >= 0) {
if ((int) testInstances.instance(i).classValue() != classAssignments[(int) clusterAssignments[i]]) {
pointShapes[i] = Plot2D.ERROR_SHAPE;
}
} else {
pointShapes[i] = Plot2D.MISSING_SHAPE;
}
}
}
PlotData2D plotData = new PlotData2D(newInsts);
if (pointShapes != null) {
plotData.setShapeType(pointShapes);
}
plotData.addInstanceNumberAttribute();
return plotData;
}
/**
* Starts running the currently configured clusterer with the current
* settings. This is run in a separate thread, and will only start if there is
* no clusterer already running. The clusterer output is sent to the results
* history panel.
*/
protected void startClusterer() {
if (m_RunThread == null) {
m_StartBut.setEnabled(false);
m_StopBut.setEnabled(true);
m_ignoreBut.setEnabled(false);
m_RunThread = new Thread() {
@Override
public void run() {
// for timing
long trainTimeStart = 0, trainTimeElapsed = 0;
// Copy the current state of things
m_Log
.statusMessage(Messages
.getInstance()
.getString(
"ClustererPanel_StartClusterer_Run_Log_StatusMessage_Text_First"));
Instances inst = new Instances(m_Instances);
inst.setClassIndex(-1);
Instances userTest = null;
PlotData2D predData = null;
if (m_TestInstances != null) {
userTest = new Instances(m_TestInstances);
}
boolean saveVis = m_StorePredictionsBut.isSelected();
String grph = null;
int[] ignoredAtts = null;
int testMode = 0;
int percent = 66;
Clusterer clusterer = (Clusterer) m_ClustererEditor.getValue();
Clusterer fullClusterer = null;
StringBuffer outBuff = new StringBuffer();
String name = (new SimpleDateFormat("HH:mm:ss - "))
.format(new Date());
String cname = clusterer.getClass().getName();
if (cname.startsWith("weka.clusterers.")) {
name += cname.substring("weka.clusterers.".length());
} else {
name += cname;
}
String cmd = m_ClustererEditor.getValue().getClass().getName();
if (m_ClustererEditor.getValue() instanceof OptionHandler)
cmd += " "
+ Utils.joinOptions(((OptionHandler) m_ClustererEditor
.getValue()).getOptions());
try {
m_Log.logMessage(Messages.getInstance().getString(
"ClustererPanel_StartClusterer_Run_Log_LogMessage_Text_First")
+ cname);
m_Log.logMessage(Messages.getInstance().getString(
"ClustererPanel_StartClusterer_Run_Log_LogMessage_Text_Second")
+ cmd);
if (m_Log instanceof TaskLogger) {
((TaskLogger) m_Log).taskStarted();
}
if (m_PercentBut.isSelected()) {
testMode = 2;
percent = Integer.parseInt(m_PercentText.getText());
if ((percent <= 0) || (percent >= 100)) {
throw new Exception(Messages.getInstance().getString(
"ClustererPanel_StartClusterer_Run_Exception_Text_First"));
}
} else if (m_TrainBut.isSelected()) {
testMode = 3;
} else if (m_TestSplitBut.isSelected()) {
testMode = 4;
// Check the test instance compatibility
if (userTest == null) {
throw new Exception(Messages.getInstance().getString(
"ClustererPanel_StartClusterer_Run_Exception_Text_Second"));
}
if (!inst.equalHeaders(userTest)) {
throw new Exception(Messages.getInstance().getString(
"ClustererPanel_StartClusterer_Run_Exception_Text_Third"));
}
} else if (m_ClassesToClustersBut.isSelected()) {
testMode = 5;
} else {
throw new Exception(Messages.getInstance().getString(
"ClustererPanel_StartClusterer_Run_Exception_Text_Fourth"));
}
Instances trainInst = new Instances(inst);
if (m_ClassesToClustersBut.isSelected()) {
trainInst.setClassIndex(m_ClassCombo.getSelectedIndex());
inst.setClassIndex(m_ClassCombo.getSelectedIndex());
if (inst.classAttribute().isNumeric()) {
throw new Exception(Messages.getInstance().getString(
"ClustererPanel_StartClusterer_Run_Exception_Text_Fifth"));
}
}
if (!m_ignoreKeyList.isSelectionEmpty()) {
trainInst = removeIgnoreCols(trainInst);
}
// Output some header information
outBuff.append(Messages.getInstance().getString(
"ClustererPanel_StartClusterer_Run_OutBuffer_Text_First"));
outBuff.append(Messages.getInstance().getString(
"ClustererPanel_StartClusterer_Run_OutBuffer_Text_Second")
+ cname);
if (clusterer instanceof OptionHandler) {
String[] o = ((OptionHandler) clusterer).getOptions();
outBuff.append(" " + Utils.joinOptions(o));
}
outBuff.append(Messages.getInstance().getString(
"ClustererPanel_StartClusterer_Run_OutBuffer_Text_Third"));
outBuff.append(Messages.getInstance().getString(
"ClustererPanel_StartClusterer_Run_OutBuffer_Text_Fourth")
+ inst.relationName() + '\n');
outBuff.append(Messages.getInstance().getString(
"ClustererPanel_StartClusterer_Run_OutBuffer_Text_Sixth")
+ inst.numInstances() + '\n');
outBuff.append(Messages.getInstance().getString(
"ClustererPanel_StartClusterer_Run_OutBuffer_Text_Eighth")
+ inst.numAttributes() + '\n');
if (inst.numAttributes() < 100) {
boolean[] selected = new boolean[inst.numAttributes()];
for (int i = 0; i < inst.numAttributes(); i++) {
selected[i] = true;
}
if (!m_ignoreKeyList.isSelectionEmpty()) {
int[] indices = m_ignoreKeyList.getSelectedIndices();
for (int i = 0; i < indices.length; i++) {
selected[indices[i]] = false;
}
}
if (m_ClassesToClustersBut.isSelected()) {
selected[m_ClassCombo.getSelectedIndex()] = false;
}
for (int i = 0; i < inst.numAttributes(); i++) {
if (selected[i]) {
outBuff.append(" " + inst.attribute(i).name()
+ '\n');
}
}
if (!m_ignoreKeyList.isSelectionEmpty()
|| m_ClassesToClustersBut.isSelected()) {
outBuff
.append(Messages
.getInstance()
.getString(
"ClustererPanel_StartClusterer_Run_OutBuffer_Text_Eleventh"));
for (int i = 0; i < inst.numAttributes(); i++) {
if (!selected[i]) {
outBuff.append(" " + inst.attribute(i).name()
+ '\n');
}
}
}
} else {
outBuff
.append(Messages
.getInstance()
.getString(
"ClustererPanel_StartClusterer_Run_OutBuffer_Text_Thirteenth"));
}
if (!m_ignoreKeyList.isSelectionEmpty()) {
ignoredAtts = m_ignoreKeyList.getSelectedIndices();
}
if (m_ClassesToClustersBut.isSelected()) {
// add class to ignored list
if (ignoredAtts == null) {
ignoredAtts = new int[1];
ignoredAtts[0] = m_ClassCombo.getSelectedIndex();
} else {
int[] newIgnoredAtts = new int[ignoredAtts.length + 1];
System.arraycopy(ignoredAtts, 0, newIgnoredAtts, 0,
ignoredAtts.length);
newIgnoredAtts[ignoredAtts.length] = m_ClassCombo
.getSelectedIndex();
ignoredAtts = newIgnoredAtts;
}
}
outBuff.append(Messages.getInstance().getString(
"ClustererPanel_StartClusterer_Run_OutBuffer_Text_Fourteenth"));
switch (testMode) {
case 3: // Test on training
outBuff
.append(Messages
.getInstance()
.getString(
"ClustererPanel_StartClusterer_Run_OutBuffer_Text_Fifteenth"));
break;
case 2: // Percent split
outBuff
.append(Messages
.getInstance()
.getString(
"ClustererPanel_StartClusterer_Run_OutBuffer_Text_Sixteenth")
+ percent
+ Messages
.getInstance()
.getString(
"ClustererPanel_StartClusterer_Run_OutBuffer_Text_Seventeenth"));
break;
case 4: // Test on user split
outBuff
.append(Messages
.getInstance()
.getString(
"ClustererPanel_StartClusterer_Run_OutBuffer_Text_Eighteenth")
+ userTest.numInstances()
+ Messages
.getInstance()
.getString(
"ClustererPanel_StartClusterer_Run_OutBuffer_Text_Nineteenth"));
break;
case 5: // Classes to clusters evaluation on training
outBuff.append(Messages.getInstance().getString(
"ClustererPanel_StartClusterer_Run_OutBuffer_Text_Twentyth"));
break;
}
outBuff.append("\n");
m_History.addResult(name, outBuff);
m_History.setSingle(name);
// Build the model and output it.
m_Log.statusMessage(Messages.getInstance().getString(
"ClustererPanel_StartClusterer_Run_Log_StatusMessage_Second"));
trainTimeStart = System.currentTimeMillis();
// remove the class attribute (if set) and build the clusterer
clusterer.buildClusterer(removeClass(trainInst));
trainTimeElapsed = System.currentTimeMillis() - trainTimeStart;
// if (testMode == 2) {
outBuff
.append(Messages
.getInstance()
.getString(
"ClustererPanel_StartClusterer_Run_OutBuffer_Text_TwentySecond"));
outBuff.append(clusterer.toString() + '\n');
outBuff
.append(Messages
.getInstance()
.getString(
"ClustererPanel_StartClusterer_Run_OutBuffer_Text_TimeTakenFull")
+ Utils.doubleToString(trainTimeElapsed / 1000.0, 2)
+ " "
+ Messages
.getInstance()
.getString(
"ClassifierPanel_StartClassifier_OutBuffer_Text_TwentyNineth"));
// }
m_History.updateResult(name);
if (clusterer instanceof Drawable) {
try {
grph = ((Drawable) clusterer).graph();
} catch (Exception ex) {
}
}
// copy full model for output
SerializedObject so = new SerializedObject(clusterer);
fullClusterer = (Clusterer) so.getObject();
ClusterEvaluation eval = new ClusterEvaluation();
eval.setClusterer(clusterer);
switch (testMode) {
case 3:
case 5: // Test on training
m_Log.statusMessage(Messages.getInstance().getString(
"ClustererPanel_StartClusterer_Run_Log_StatusMessage_Third"));
eval.evaluateClusterer(trainInst, "", false);
predData = setUpVisualizableInstances(inst, eval);
outBuff
.append(Messages
.getInstance()
.getString(
"ClustererPanel_StartClusterer_Run_OutBuffer_Text_TwentySecond"));
break;
case 2: // Percent split
m_Log
.statusMessage(Messages
.getInstance()
.getString(
"ClustererPanel_StartClusterer_Run_Log_StatusMessage_Fourth"));
inst.randomize(new Random(1));
trainInst.randomize(new Random(1));
int trainSize = trainInst.numInstances() * percent / 100;
int testSize = trainInst.numInstances() - trainSize;
Instances train = new Instances(trainInst, 0, trainSize);
Instances test = new Instances(trainInst, trainSize, testSize);
Instances testVis = new Instances(inst, trainSize, testSize);
m_Log.statusMessage(Messages.getInstance().getString(
"ClustererPanel_StartClusterer_Run_Log_StatusMessage_Fifth"));
trainTimeStart = System.currentTimeMillis();
clusterer.buildClusterer(train);
trainTimeElapsed = System.currentTimeMillis() - trainTimeStart;
m_Log.statusMessage(Messages.getInstance().getString(
"ClustererPanel_StartClusterer_Run_Log_StatusMessage_Sixth"));
eval.evaluateClusterer(test, "", false);
predData = setUpVisualizableInstances(testVis, eval);
outBuff
.append(Messages
.getInstance()
.getString(
"ClustererPanel_StartClusterer_Run_OutBuffer_Text_TwentyThird"));
outBuff.append(clusterer.toString() + '\n');
outBuff
.append(Messages
.getInstance()
.getString(
"ClustererPanel_StartClusterer_Run_OutBuffer_Text_TimeTakenPercentage")
+ Utils.doubleToString(trainTimeElapsed / 1000.0, 2)
+ " "
+ Messages
.getInstance()
.getString(
"ClassifierPanel_StartClassifier_OutBuffer_Text_TwentyNineth"));
break;
case 4: // Test on user split
m_Log
.statusMessage(Messages
.getInstance()
.getString(
"ClustererPanel_StartClusterer_Run_Log_StatusMessage_Seventh"));
Instances userTestT = new Instances(userTest);
if (!m_ignoreKeyList.isSelectionEmpty()) {
userTestT = removeIgnoreCols(userTestT);
}
eval.evaluateClusterer(userTestT, "", false);
predData = setUpVisualizableInstances(userTest, eval);
outBuff
.append(Messages
.getInstance()
.getString(
"ClustererPanel_StartClusterer_Run_OutBuffer_Text_TwentyFourth"));
break;
default:
throw new Exception(Messages.getInstance().getString(
"ClustererPanel_StartClusterer_Run_Exception_Text_Sixth"));
}
outBuff.append(eval.clusterResultsToString());
outBuff.append("\n");
m_History.updateResult(name);
m_Log.logMessage(Messages.getInstance().getString(
"ClustererPanel_StartClusterer_Run_Log_LogMessage_Text_Third")
+ cname);
m_Log.statusMessage(Messages.getInstance().getString(
"ClustererPanel_StartClusterer_Run_Log_StatusMessage_Eighth"));
} catch (Exception ex) {
ex.printStackTrace();
m_Log.logMessage(ex.getMessage());
JOptionPane
.showMessageDialog(
ClustererPanel.this,
Messages
.getInstance()
.getString(
"ClustererPanel_StartClusterer_Run_JOptionPaneShowMessageDialog_Text_First")
+ ex.getMessage(),
Messages
.getInstance()
.getString(
"ClustererPanel_StartClusterer_Run_JOptionPaneShowMessageDialog_Text_Second"),
JOptionPane.ERROR_MESSAGE);
m_Log.statusMessage(Messages.getInstance().getString(
"ClustererPanel_StartClusterer_Run_Log_StatusMessage_Nineth"));
} finally {
if (predData != null) {
m_CurrentVis = new VisualizePanel();
m_CurrentVis.setName(name + " (" + inst.relationName() + ")");
m_CurrentVis.setLog(m_Log);
predData.setPlotName(name + " (" + inst.relationName() + ")");
try {
m_CurrentVis.addPlot(predData);
} catch (Exception ex) {
System.err.println(ex);
}
FastVector vv = new FastVector();
vv.addElement(fullClusterer);
Instances trainHeader = new Instances(m_Instances, 0);
vv.addElement(trainHeader);
if (ignoredAtts != null)
vv.addElement(ignoredAtts);
if (saveVis) {
vv.addElement(m_CurrentVis);
if (grph != null) {
vv.addElement(grph);
}
}
m_History.addObject(name, vv);
}
if (isInterrupted()) {
m_Log
.logMessage(Messages
.getInstance()
.getString(
"ClustererPanel_StartClusterer_Run_Log_LogMessage_Text_Fourth")
+ cname);
m_Log.statusMessage(Messages.getInstance().getString(
"ClustererPanel_StartClusterer_Run_Log_StatusMessage_Tenth"));
}
m_RunThread = null;
m_StartBut.setEnabled(true);
m_StopBut.setEnabled(false);
m_ignoreBut.setEnabled(true);
if (m_Log instanceof TaskLogger) {
((TaskLogger) m_Log).taskFinished();
}
}
}
};
m_RunThread.setPriority(Thread.MIN_PRIORITY);
m_RunThread.start();
}
}
private Instances removeClass(Instances inst) {
Remove af = new Remove();
Instances retI = null;
try {
if (inst.classIndex() < 0) {
retI = inst;
} else {
af.setAttributeIndices("" + (inst.classIndex() + 1));
af.setInvertSelection(false);
af.setInputFormat(inst);
retI = Filter.useFilter(inst, af);
}
} catch (Exception e) {
e.printStackTrace();
}
return retI;
}
private Instances removeIgnoreCols(Instances inst) {
// If the user is doing classes to clusters evaluation and
// they have opted to ignore the class, then unselect the class in
// the ignore list
if (m_ClassesToClustersBut.isSelected()) {
int classIndex = m_ClassCombo.getSelectedIndex();
if (m_ignoreKeyList.isSelectedIndex(classIndex)) {
m_ignoreKeyList.removeSelectionInterval(classIndex, classIndex);
}
}
int[] selected = m_ignoreKeyList.getSelectedIndices();
Remove af = new Remove();
Instances retI = null;
try {
af.setAttributeIndicesArray(selected);
af.setInvertSelection(false);
af.setInputFormat(inst);
retI = Filter.useFilter(inst, af);
} catch (Exception e) {
e.printStackTrace();
}
return retI;
}
private Instances removeIgnoreCols(Instances inst, int[] toIgnore) {
Remove af = new Remove();
Instances retI = null;
try {
af.setAttributeIndicesArray(toIgnore);
af.setInvertSelection(false);
af.setInputFormat(inst);
retI = Filter.useFilter(inst, af);
} catch (Exception e) {
e.printStackTrace();
}
return retI;
}
/**
* Stops the currently running clusterer (if any).
*/
protected void stopClusterer() {
if (m_RunThread != null) {
m_RunThread.interrupt();
// This is deprecated (and theoretically the interrupt should do).
m_RunThread.stop();
}
}
/**
* Pops up a TreeVisualizer for the clusterer from the currently selected item
* in the results list
*
* @param graphString the description of the tree in dotty format
* @param treeName the title to assign to the display
*/
protected void visualizeTree(String graphString, String treeName) {
final javax.swing.JFrame jf = new javax.swing.JFrame(Messages.getInstance()
.getString("ClustererPanel_VisualizeTree_JFrame_Text") + treeName);
jf.setSize(500, 400);
jf.getContentPane().setLayout(new BorderLayout());
if (graphString.contains("digraph")) {
TreeVisualizer tv = new TreeVisualizer(null, graphString,
new PlaceNode2());
jf.getContentPane().add(tv, BorderLayout.CENTER);
jf.addWindowListener(new java.awt.event.WindowAdapter() {
@Override
public void windowClosing(java.awt.event.WindowEvent e) {
jf.dispose();
}
});
jf.setVisible(true);
tv.fitToScreen();
} else if (graphString.startsWith(Messages.getInstance().getString(
"ClustererPanel_VisualizeTree_GraphStringStartsWith_Text"))) {
HierarchyVisualizer tv = new HierarchyVisualizer(graphString.substring(7));
jf.getContentPane().add(tv, BorderLayout.CENTER);
jf.addWindowListener(new java.awt.event.WindowAdapter() {
@Override
public void windowClosing(java.awt.event.WindowEvent e) {
jf.dispose();
}
});
jf.setVisible(true);
tv.fitToScreen();
}
}
/**
* Pops up a visualize panel to display cluster assignments
*
* @param sp the visualize panel to display
*/
protected void visualizeClusterAssignments(VisualizePanel sp) {
if (sp != null) {
String plotName = sp.getName();
final javax.swing.JFrame jf = new javax.swing.JFrame(Messages
.getInstance().getString(
"ClustererPanel_VisualizeClusterAssignments_JFrame_Text")
+ plotName);
jf.setSize(500, 400);
jf.getContentPane().setLayout(new BorderLayout());
jf.getContentPane().add(sp, BorderLayout.CENTER);
jf.addWindowListener(new java.awt.event.WindowAdapter() {
@Override
public void windowClosing(java.awt.event.WindowEvent e) {
jf.dispose();
}
});
jf.setVisible(true);
}
}
/**
* Handles constructing a popup menu with visualization options
*
* @param name the name of the result history list entry clicked on by the
* user
* @param x the x coordinate for popping up the menu
* @param y the y coordinate for popping up the menu
*/
protected void visualizeClusterer(String name, int x, int y) {
final String selectedName = name;
JPopupMenu resultListMenu = new JPopupMenu();
JMenuItem visMainBuffer = new JMenuItem(Messages.getInstance().getString(
"ClustererPanel_VisualizeClusterer_VisMainBuffer_JMenuItem_Text"));
if (selectedName != null) {
visMainBuffer.addActionListener(new ActionListener() {
public void actionPerformed(ActionEvent e) {
m_History.setSingle(selectedName);
}
});
} else {
visMainBuffer.setEnabled(false);
}
resultListMenu.add(visMainBuffer);
JMenuItem visSepBuffer = new JMenuItem(Messages.getInstance().getString(
"ClustererPanel_VisualizeClusterer_VisSepBuffer_JMenuItem_Text"));
if (selectedName != null) {
visSepBuffer.addActionListener(new ActionListener() {
public void actionPerformed(ActionEvent e) {
m_History.openFrame(selectedName);
}
});
} else {
visSepBuffer.setEnabled(false);
}
resultListMenu.add(visSepBuffer);
JMenuItem saveOutput = new JMenuItem(Messages.getInstance().getString(
"ClustererPanel_VisualizeClusterer_SaveOutput_JMenuItem_Text"));
if (selectedName != null) {
saveOutput.addActionListener(new ActionListener() {
public void actionPerformed(ActionEvent e) {
saveBuffer(selectedName);
}
});
} else {
saveOutput.setEnabled(false);
}
resultListMenu.add(saveOutput);
JMenuItem deleteOutput = new JMenuItem(Messages.getInstance().getString(
"ClustererPanel_VisualizeClusterer_DeleteOutput_JMenuItem_Text"));
if (selectedName != null) {
deleteOutput.addActionListener(new ActionListener() {
public void actionPerformed(ActionEvent e) {
m_History.removeResult(selectedName);
}
});
} else {
deleteOutput.setEnabled(false);
}
resultListMenu.add(deleteOutput);
resultListMenu.addSeparator();
JMenuItem loadModel = new JMenuItem(Messages.getInstance().getString(
"ClustererPanel_VisualizeClusterer_LoadModel_JMenuItem_Text"));
loadModel.addActionListener(new ActionListener() {
public void actionPerformed(ActionEvent e) {
loadClusterer();
}
});
resultListMenu.add(loadModel);
FastVector o = null;
if (selectedName != null) {
o = (FastVector) m_History.getNamedObject(selectedName);
}
VisualizePanel temp_vp = null;
String temp_grph = null;
Clusterer temp_clusterer = null;
Instances temp_trainHeader = null;
int[] temp_ignoreAtts = null;
if (o != null) {
for (int i = 0; i < o.size(); i++) {
Object temp = o.elementAt(i);
if (temp instanceof Clusterer) {
temp_clusterer = (Clusterer) temp;
} else if (temp instanceof Instances) { // training header
temp_trainHeader = (Instances) temp;
} else if (temp instanceof int[]) { // ignored attributes
temp_ignoreAtts = (int[]) temp;
} else if (temp instanceof VisualizePanel) { // normal errors
temp_vp = (VisualizePanel) temp;
} else if (temp instanceof String) { // graphable output
temp_grph = (String) temp;
}
}
}
final VisualizePanel vp = temp_vp;
final String grph = temp_grph;
final Clusterer clusterer = temp_clusterer;
final Instances trainHeader = temp_trainHeader;
final int[] ignoreAtts = temp_ignoreAtts;
JMenuItem saveModel = new JMenuItem(Messages.getInstance().getString(
"ClustererPanel_VisualizeClusterer_SaveModel_JMenuItem_Text"));
if (clusterer != null) {
saveModel.addActionListener(new ActionListener() {
public void actionPerformed(ActionEvent e) {
saveClusterer(selectedName, clusterer, trainHeader, ignoreAtts);
}
});
} else {
saveModel.setEnabled(false);
}
resultListMenu.add(saveModel);
JMenuItem reEvaluate = new JMenuItem(Messages.getInstance().getString(
"ClustererPanel_VisualizeClusterer_ReEvaluate_JMenuItem_Text"));
if (clusterer != null && m_TestInstances != null) {
reEvaluate.addActionListener(new ActionListener() {
public void actionPerformed(ActionEvent e) {
reevaluateModel(selectedName, clusterer, trainHeader, ignoreAtts);
}
});
} else {
reEvaluate.setEnabled(false);
}
resultListMenu.add(reEvaluate);
resultListMenu.addSeparator();
JMenuItem visClusts = new JMenuItem(Messages.getInstance().getString(
"ClustererPanel_VisualizeClusterer_VisClusts_JMenuItem_Text"));
if (vp != null) {
visClusts.addActionListener(new ActionListener() {
public void actionPerformed(ActionEvent e) {
visualizeClusterAssignments(vp);
}
});
} else {
visClusts.setEnabled(false);
}
resultListMenu.add(visClusts);
JMenuItem visTree = new JMenuItem(Messages.getInstance().getString(
"ClustererPanel_VisualizeClusterer_VisTree_JMenuItem_Text"));
if (grph != null) {
visTree.addActionListener(new ActionListener() {
public void actionPerformed(ActionEvent e) {
String title;
if (vp != null)
title = vp.getName();
else
title = selectedName;
visualizeTree(grph, title);
}
});
} else {
visTree.setEnabled(false);
}
resultListMenu.add(visTree);
resultListMenu.show(m_History.getList(), x, y);
}
/**
* Save the currently selected clusterer output to a file.
*
* @param name the name of the buffer to save
*/
protected void saveBuffer(String name) {
StringBuffer sb = m_History.getNamedBuffer(name);
if (sb != null) {
if (m_SaveOut.save(sb)) {
m_Log
.logMessage(Messages
.getInstance()
.getString(
"ClustererPanel_VisualizeClusterer_SaveBuffer_Log_LohMessage_Text"));
}
}
}
private void setIgnoreColumns() {
ListSelectorDialog jd = new ListSelectorDialog(null, m_ignoreKeyList);
// Open the dialog
int result = jd.showDialog();
if (result != ListSelectorDialog.APPROVE_OPTION) {
// clear selected indices
m_ignoreKeyList.clearSelection();
}
updateCapabilitiesFilter(m_ClustererEditor.getCapabilitiesFilter());
}
/**
* Saves the currently selected clusterer
*/
protected void saveClusterer(String name, Clusterer clusterer,
Instances trainHeader, int[] ignoredAtts) {
File sFile = null;
boolean saveOK = true;
int returnVal = m_FileChooser.showSaveDialog(this);
if (returnVal == JFileChooser.APPROVE_OPTION) {
sFile = m_FileChooser.getSelectedFile();
if (!sFile.getName().toLowerCase().endsWith(MODEL_FILE_EXTENSION)) {
sFile = new File(sFile.getParent(), sFile.getName()
+ MODEL_FILE_EXTENSION);
}
m_Log
.statusMessage(Messages
.getInstance()
.getString(
"ClustererPanel_VisualizeClusterer_SaveBuffer_Log_LohMessage_Text_Alpha"));
try {
OutputStream os = new FileOutputStream(sFile);
if (sFile.getName().endsWith(".gz")) {
os = new GZIPOutputStream(os);
}
ObjectOutputStream objectOutputStream = new ObjectOutputStream(os);
objectOutputStream.writeObject(clusterer);
if (trainHeader != null)
objectOutputStream.writeObject(trainHeader);
if (ignoredAtts != null)
objectOutputStream.writeObject(ignoredAtts);
objectOutputStream.flush();
objectOutputStream.close();
} catch (Exception e) {
JOptionPane
.showMessageDialog(
null,
e,
Messages
.getInstance()
.getString(
"ClustererPanel_VisualizeClusterer_SaveCluster_JOptionPaneShowMessageDialog_Text"),
JOptionPane.ERROR_MESSAGE);
saveOK = false;
}
if (saveOK)
m_Log
.logMessage(Messages
.getInstance()
.getString(
"ClustererPanel_VisualizeClusterer_SaveCluster_Log_LogMessage_Text")
+ name
+ Messages
.getInstance()
.getString(
"ClustererPanel_VisualizeClusterer_SaveCluster_Log_LogMessage_Text_Alpha")
+ sFile.getName() + "'");
m_Log
.statusMessage(Messages
.getInstance()
.getString(
"ClustererPanel_VisualizeClusterer_SaveCluster_Log_StatusMessage_Text"));
}
}
/**
* Loads a clusterer
*/
protected void loadClusterer() {
int returnVal = m_FileChooser.showOpenDialog(this);
if (returnVal == JFileChooser.APPROVE_OPTION) {
File selected = m_FileChooser.getSelectedFile();
Clusterer clusterer = null;
Instances trainHeader = null;
int[] ignoredAtts = null;
m_Log
.statusMessage(Messages
.getInstance()
.getString(
"ClustererPanel_VisualizeClusterer_LoadClusterer_Log_StatusSessage_Text_First"));
try {
InputStream is = new FileInputStream(selected);
if (selected.getName().endsWith(".gz")) {
is = new GZIPInputStream(is);
}
ObjectInputStream objectInputStream = new ObjectInputStream(is);
clusterer = (Clusterer) objectInputStream.readObject();
try { // see if we can load the header & ignored attribute info
trainHeader = (Instances) objectInputStream.readObject();
ignoredAtts = (int[]) objectInputStream.readObject();
} catch (Exception e) {
} // don't fuss if we can't
objectInputStream.close();
} catch (Exception e) {
JOptionPane
.showMessageDialog(
null,
e,
Messages
.getInstance()
.getString(
"ClustererPanel_VisualizeClusterer_LoadClusterer_JOptionPaneShowMessageDialog_Text"),
JOptionPane.ERROR_MESSAGE);
}
m_Log
.statusMessage(Messages
.getInstance()
.getString(
"ClustererPanel_VisualizeClusterer_LoadClusterer_Log_StatusMessage_Text_Second"));
if (clusterer != null) {
m_Log
.logMessage(Messages
.getInstance()
.getString(
"ClustererPanel_VisualizeClusterer_LoadClusterer_Log_LogMessage_Text_First")
+ selected.getName() + "'");
String name = (new SimpleDateFormat("HH:mm:ss - ")).format(new Date());
String cname = clusterer.getClass().getName();
if (cname.startsWith("weka.clusterers."))
cname = cname.substring("weka.clusterers.".length());
name += cname
+ Messages
.getInstance()
.getString(
"ClustererPanel_VisualizeClusterer_LoadClusterer_CNAme_Text_First")
+ selected.getName() + "'";
StringBuffer outBuff = new StringBuffer();
outBuff
.append(Messages
.getInstance()
.getString(
"ClustererPanel_VisualizeClusterer_LoadClusterer_OutBuffer_Text_First"));
outBuff
.append(Messages
.getInstance()
.getString(
"ClustererPanel_VisualizeClusterer_LoadClusterer_OutBuffer_Text_Second")
+ selected.getName() + "\n");
outBuff
.append(Messages
.getInstance()
.getString(
"ClustererPanel_VisualizeClusterer_LoadClusterer_OutBuffer_Text_Fourth")
+ clusterer.getClass().getName());
if (clusterer instanceof OptionHandler) {
String[] o = ((OptionHandler) clusterer).getOptions();
outBuff.append(" " + Utils.joinOptions(o));
}
outBuff.append("\n");
if (trainHeader != null) {
outBuff
.append(Messages
.getInstance()
.getString(
"ClustererPanel_VisualizeClusterer_LoadClusterer_OutBuffer_Text_Sixth")
+ trainHeader.relationName() + '\n');
outBuff
.append(Messages
.getInstance()
.getString(
"ClustererPanel_VisualizeClusterer_LoadClusterer_OutBuffer_Text_Eighth")
+ trainHeader.numAttributes() + '\n');
if (trainHeader.numAttributes() < 100) {
boolean[] selectedAtts = new boolean[trainHeader.numAttributes()];
for (int i = 0; i < trainHeader.numAttributes(); i++) {
selectedAtts[i] = true;
}
if (ignoredAtts != null)
for (int i = 0; i < ignoredAtts.length; i++)
selectedAtts[ignoredAtts[i]] = false;
for (int i = 0; i < trainHeader.numAttributes(); i++) {
if (selectedAtts[i]) {
outBuff.append(" "
+ trainHeader.attribute(i).name() + '\n');
}
}
if (ignoredAtts != null) {
outBuff
.append(Messages
.getInstance()
.getString(
"ClustererPanel_VisualizeClusterer_LoadClusterer_OutBuffer_Text_Eleventh"));
for (int i = 0; i < ignoredAtts.length; i++)
outBuff.append(" "
+ trainHeader.attribute(ignoredAtts[i]).name() + '\n');
}
} else {
outBuff
.append(Messages
.getInstance()
.getString(
"ClustererPanel_VisualizeClusterer_LoadClusterer_OutBuffer_Text_Twelveth"));
}
} else {
outBuff
.append(Messages
.getInstance()
.getString(
"ClustererPanel_VisualizeClusterer_LoadClusterer_OutBuffer_Text_Thirteenth"));
}
outBuff
.append(Messages
.getInstance()
.getString(
"ClustererPanel_VisualizeClusterer_LoadClusterer_OutBuffer_Text_Fourteenth"));
outBuff.append(clusterer.toString() + "\n");
m_History.addResult(name, outBuff);
m_History.setSingle(name);
FastVector vv = new FastVector();
vv.addElement(clusterer);
if (trainHeader != null)
vv.addElement(trainHeader);
if (ignoredAtts != null)
vv.addElement(ignoredAtts);
// allow visualization of graphable classifiers
String grph = null;
if (clusterer instanceof Drawable) {
try {
grph = ((Drawable) clusterer).graph();
} catch (Exception ex) {
}
}
if (grph != null)
vv.addElement(grph);
m_History.addObject(name, vv);
}
}
}
/**
* Re-evaluates the named clusterer with the current test set. Unpredictable
* things will happen if the data set is not compatible with the clusterer.
*
* @param name the name of the clusterer entry
* @param clusterer the clusterer to evaluate
* @param trainHeader the header of the training set
* @param ignoredAtts ignored attributes
*/
protected void reevaluateModel(final String name, final Clusterer clusterer,
final Instances trainHeader, final int[] ignoredAtts) {
if (m_RunThread == null) {
m_StartBut.setEnabled(false);
m_StopBut.setEnabled(true);
m_ignoreBut.setEnabled(false);
m_RunThread = new Thread() {
@Override
public void run() {
// Copy the current state of things
m_Log
.statusMessage(Messages
.getInstance()
.getString(
"ClustererPanel_ReEvaluateModel_Run_Log_StatusMessage_Text_First"));
StringBuffer outBuff = m_History.getNamedBuffer(name);
Instances userTest = null;
PlotData2D predData = null;
if (m_TestInstances != null) {
userTest = new Instances(m_TestInstances);
}
boolean saveVis = m_StorePredictionsBut.isSelected();
String grph = null;
try {
if (userTest == null) {
throw new Exception(Messages.getInstance().getString(
"ClustererPanel_ReEvaluateModel_Run_Exception_Text_First"));
}
if (trainHeader != null && !trainHeader.equalHeaders(userTest)) {
throw new Exception(Messages.getInstance().getString(
"ClustererPanel_ReEvaluateModel_Run_Exception_Text_Second"));
}
m_Log
.statusMessage(Messages
.getInstance()
.getString(
"ClustererPanel_ReEvaluateModel_Run_Log_StatusMessage_Text_Second"));
m_Log
.logMessage(Messages
.getInstance()
.getString(
"ClustererPanel_ReEvaluateModel_Run_Log_LogMessage_Text_First")
+ name
+ Messages
.getInstance()
.getString(
"ClustererPanel_ReEvaluateModel_Run_Log_LogMessage_Text_Second"));
m_Log
.logMessage(Messages
.getInstance()
.getString(
"ClustererPanel_ReEvaluateModel_Run_Log_LogMessage_Text_Third"));
if (m_Log instanceof TaskLogger) {
((TaskLogger) m_Log).taskStarted();
}
ClusterEvaluation eval = new ClusterEvaluation();
eval.setClusterer(clusterer);
Instances userTestT = new Instances(userTest);
if (ignoredAtts != null) {
userTestT = removeIgnoreCols(userTestT, ignoredAtts);
}
eval.evaluateClusterer(userTestT);
predData = setUpVisualizableInstances(userTest, eval);
outBuff.append(Messages.getInstance().getString(
"ClustererPanel_ReEvaluateModel_Run_OutBuffer_Text_First"));
outBuff.append(Messages.getInstance().getString(
"ClustererPanel_ReEvaluateModel_Run_OutBuffer_Text_Second"));
outBuff
.append(Messages.getInstance().getString(
"ClustererPanel_ReEvaluateModel_Run_OutBuffer_Text_Third")
+ userTest.relationName()
+ Messages
.getInstance()
.getString(
"ClustererPanel_ReEvaluateModel_Run_OutBuffer_Text_Fourth"));
outBuff.append(Messages.getInstance().getString(
"ClustererPanel_ReEvaluateModel_Run_OutBuffer_Text_Fifth")
+ userTest.numInstances()
+ Messages.getInstance().getString(
"ClustererPanel_ReEvaluateModel_Run_OutBuffer_Text_Sixth"));
outBuff
.append(Messages
.getInstance()
.getString(
"ClustererPanel_ReEvaluateModel_Run_OutBuffer_Text_Seventh")
+ userTest.numAttributes()
+ Messages
.getInstance()
.getString(
"ClustererPanel_ReEvaluateModel_Run_OutBuffer_Text_Eighth"));
if (trainHeader == null)
outBuff.append(Messages.getInstance().getString(
"ClustererPanel_ReEvaluateModel_Run_OutBuffer_Text_Nineth"));
outBuff.append(eval.clusterResultsToString());
outBuff.append(Messages.getInstance().getString(
"ClustererPanel_ReEvaluateModel_Run_OutBuffer_Text_Tenth"));
m_History.updateResult(name);
m_Log
.logMessage(Messages
.getInstance()
.getString(
"ClustererPanel_ReEvaluateModel_Run_Log_LogMessage_Text_Fourth"));
m_Log
.statusMessage(Messages
.getInstance()
.getString(
"ClustererPanel_ReEvaluateModel_Run_Log_StatusMessage_Text_Third"));
} catch (Exception ex) {
ex.printStackTrace();
m_Log.logMessage(ex.getMessage());
JOptionPane
.showMessageDialog(
ClustererPanel.this,
Messages
.getInstance()
.getString(
"ClustererPanel_ReEvaluateModel_Run_JOptionPaneShowMessageDialog_Text_First")
+ ex.getMessage(),
Messages
.getInstance()
.getString(
"ClustererPanel_ReEvaluateModel_Run_JOptionPaneShowMessageDialog_Text_Second"),
JOptionPane.ERROR_MESSAGE);
m_Log
.statusMessage(Messages
.getInstance()
.getString(
"ClustererPanel_ReEvaluateModel_Run_Log_StatusMessage_Text_Fourth"));
} finally {
if (predData != null) {
m_CurrentVis = new VisualizePanel();
m_CurrentVis.setName(name + " (" + userTest.relationName() + ")");
m_CurrentVis.setLog(m_Log);
predData.setPlotName(name + " (" + userTest.relationName() + ")");
try {
m_CurrentVis.addPlot(predData);
} catch (Exception ex) {
System.err.println(ex);
}
FastVector vv = new FastVector();
vv.addElement(clusterer);
if (trainHeader != null)
vv.addElement(trainHeader);
if (ignoredAtts != null)
vv.addElement(ignoredAtts);
if (saveVis) {
vv.addElement(m_CurrentVis);
if (grph != null) {
vv.addElement(grph);
}
}
m_History.addObject(name, vv);
}
if (isInterrupted()) {
m_Log
.logMessage(Messages
.getInstance()
.getString(
"ClustererPanel_ReEvaluateModel_Run_Log_LogMessage_Text_Fifth"));
m_Log
.statusMessage(Messages
.getInstance()
.getString(
"ClustererPanel_ReEvaluateModel_Run_Log_StatusMessage_Text_Fifth"));
}
m_RunThread = null;
m_StartBut.setEnabled(true);
m_StopBut.setEnabled(false);
m_ignoreBut.setEnabled(true);
if (m_Log instanceof TaskLogger) {
((TaskLogger) m_Log).taskFinished();
}
}
}
};
m_RunThread.setPriority(Thread.MIN_PRIORITY);
m_RunThread.start();
}
}
/**
* updates the capabilities filter of the GOE
*
* @param filter the new filter to use
*/
protected void updateCapabilitiesFilter(Capabilities filter) {
Instances tempInst;
Capabilities filterClass;
if (filter == null) {
m_ClustererEditor.setCapabilitiesFilter(new Capabilities(null));
return;
}
if (!ExplorerDefaults.getInitGenericObjectEditorFilter())
tempInst = new Instances(m_Instances, 0);
else
tempInst = new Instances(m_Instances);
tempInst.setClassIndex(-1);
if (!m_ignoreKeyList.isSelectionEmpty()) {
tempInst = removeIgnoreCols(tempInst);
}
if (m_ClassesToClustersBut.isSelected()) {
// remove the class too
String classSelection = m_ClassCombo.getSelectedItem().toString();
classSelection = classSelection
.substring(classSelection.indexOf(")") + 1).trim();
int classIndex = tempInst.attribute(classSelection).index();
Remove rm = new Remove();
rm.setAttributeIndices("" + (classIndex + 1));
try {
rm.setInputFormat(tempInst);
tempInst = Filter.useFilter(tempInst, rm);
} catch (Exception e) {
e.printStackTrace();
}
}
try {
filterClass = Capabilities.forInstances(tempInst);
} catch (Exception e) {
filterClass = new Capabilities(null);
}
m_ClustererEditor.setCapabilitiesFilter(filterClass);
// check capabilities
m_StartBut.setEnabled(true);
Capabilities currentFilter = m_ClustererEditor.getCapabilitiesFilter();
Clusterer clusterer = (Clusterer) m_ClustererEditor.getValue();
Capabilities currentSchemeCapabilities = null;
if (clusterer != null && currentFilter != null
&& (clusterer instanceof CapabilitiesHandler)) {
currentSchemeCapabilities = ((CapabilitiesHandler) clusterer)
.getCapabilities();
if (!currentSchemeCapabilities.supportsMaybe(currentFilter)
&& !currentSchemeCapabilities.supports(currentFilter)) {
m_StartBut.setEnabled(false);
}
}
}
/**
* method gets called in case of a change event
*
* @param e the associated change event
*/
public void capabilitiesFilterChanged(CapabilitiesFilterChangeEvent e) {
if (e.getFilter() == null)
updateCapabilitiesFilter(null);
else
updateCapabilitiesFilter((Capabilities) e.getFilter().clone());
}
/**
* Sets the Explorer to use as parent frame (used for sending notifications
* about changes in the data)
*
* @param parent the parent frame
*/
public void setExplorer(Explorer parent) {
m_Explorer = parent;
}
/**
* returns the parent Explorer frame
*
* @return the parent
*/
public Explorer getExplorer() {
return m_Explorer;
}
/**
* Returns the title for the tab in the Explorer
*
* @return the title of this tab
*/
public String getTabTitle() {
return Messages.getInstance().getString("ClustererPanel_GetTabTitle_Text");
}
/**
* Returns the tooltip for the tab in the Explorer
*
* @return the tooltip of this tab
*/
public String getTabTitleToolTip() {
return Messages.getInstance().getString(
"ClustererPanel_GetTabTitleToolTip_Text");
}
/**
* Tests out the clusterer panel from the command line.
*
* @param args may optionally contain the name of a dataset to load.
*/
public static void main(String[] args) {
try {
final javax.swing.JFrame jf = new javax.swing.JFrame(Messages
.getInstance().getString("ClustererPanel_Main_JFrame_Text"));
jf.getContentPane().setLayout(new BorderLayout());
final ClustererPanel sp = new ClustererPanel();
jf.getContentPane().add(sp, BorderLayout.CENTER);
weka.gui.LogPanel lp = new weka.gui.LogPanel();
sp.setLog(lp);
jf.getContentPane().add(lp, BorderLayout.SOUTH);
jf.addWindowListener(new java.awt.event.WindowAdapter() {
@Override
public void windowClosing(java.awt.event.WindowEvent e) {
jf.dispose();
System.exit(0);
}
});
jf.pack();
jf.setSize(800, 600);
jf.setVisible(true);
if (args.length == 1) {
System.err.println(Messages.getInstance().getString(
"ClustererPanel_Main_Error_Text_First")
+ args[0]);
java.io.Reader r = new java.io.BufferedReader(new java.io.FileReader(
args[0]));
Instances i = new Instances(r);
sp.setInstances(i);
}
} catch (Exception ex) {
ex.printStackTrace();
System.err.println(ex.getMessage());
}
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy