weka.gui.visualize.PrintableComponent 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 3 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, see .
*/
/*
* PrintableComponent.java
* Copyright (C) 2005-2012 University of Waikato, Hamilton, New Zealand
*
*/
package weka.gui.visualize;
import java.awt.Dimension;
import java.awt.event.ItemEvent;
import java.awt.event.ItemListener;
import java.awt.event.MouseAdapter;
import java.awt.event.MouseEvent;
import java.io.BufferedOutputStream;
import java.io.File;
import java.io.FileOutputStream;
import java.util.Collections;
import java.util.Enumeration;
import java.util.Hashtable;
import java.util.Properties;
import java.util.Vector;
import javax.swing.JCheckBox;
import javax.swing.JComponent;
import javax.swing.JFileChooser;
import javax.swing.JLabel;
import javax.swing.JOptionPane;
import javax.swing.JPanel;
import javax.swing.JTextField;
import javax.swing.event.DocumentEvent;
import javax.swing.event.DocumentListener;
import weka.gui.ExtensionFileFilter;
import weka.gui.GenericObjectEditor;
/**
* This class extends the component which is handed over in the constructor by a
* print dialog. The Print dialog is accessible via Alt+Shift+LeftMouseClick.
*
* The individual JComponentWriter-descendants can be accessed by the
* getWriter(String)
method, if the parameters need to be changed.
*
* @see #getWriters()
* @see #getWriter(String)
* @author FracPete (fracpete at waikato dot ac dot nz)
* @version $Revision: 10222 $
*/
public class PrintableComponent implements PrintableHandler {
/** the parent component of this print dialog. */
protected JComponent m_Component;
/** the filechooser for saving the panel. */
protected static JFileChooser m_FileChooserPanel;
/** the checkbox for the custom dimensions. */
protected static JCheckBox m_CustomDimensionsCheckBox;
/** the edit field for the custom width. */
protected static JTextField m_CustomWidthText;
/** the edit field for the custom height. */
protected static JTextField m_CustomHeightText;
/** the checkbox for keeping the aspect ration. */
protected static JCheckBox m_AspectRatioCheckBox;
/** the title of the save dialog. */
protected String m_SaveDialogTitle = "Save as...";
/** the x scale factor. */
protected double m_xScale = 1.0;
/** the y scale factor. */
protected double m_yScale = 1.0;
/** the aspect ratio. */
protected double m_AspectRatio;
/** whether to ignore the update of the text field (in case of "keep ratio"). */
protected boolean m_IgnoreChange;
/** whether to print some debug information. */
private static final boolean DEBUG = false;
/** whether the user was already asked about the tooltip behavior. */
protected static boolean m_ToolTipUserAsked = false;
/** the property name for showing the tooltip. */
protected final static String PROPERTY_SHOW = "PrintableComponentToolTipShow";
/** the property name whether the user was already asked. */
protected final static String PROPERTY_USERASKED = "PrintableComponentToolTipUserAsked";
/** whether to display the tooltip or not. */
protected static boolean m_ShowToolTip = true;
static {
try {
m_ShowToolTip = Boolean.valueOf(
VisualizeUtils.VISUALIZE_PROPERTIES.getProperty(PROPERTY_SHOW, "true"))
.booleanValue();
m_ToolTipUserAsked = Boolean.valueOf(
VisualizeUtils.VISUALIZE_PROPERTIES.getProperty(PROPERTY_USERASKED,
"false")).booleanValue();
} catch (Exception e) {
// ignore exception
m_ToolTipUserAsked = false;
m_ShowToolTip = true;
}
}
/** output if we're in debug mode */
static {
if (DEBUG) {
System.err.println(PrintablePanel.class.getName() + ": DEBUG ON");
}
}
/**
* initializes the panel.
*
* @param component the component to enhance with printing functionality
*/
public PrintableComponent(JComponent component) {
super();
m_Component = component;
m_AspectRatio = Double.NaN;
getComponent().addMouseListener(new PrintMouseListener(this));
getComponent().setToolTipText(getToolTipText(this));
initFileChooser();
}
/**
* returns the GUI component this print dialog is part of.
*
* @return the GUI component
*/
public JComponent getComponent() {
return m_Component;
}
/**
* Returns a tooltip only if the user wants it. If retrieved for the first, a
* dialog pops up and asks the user whether the tooltip should always appear
* or not. The weka/gui/visualize/Visualize.props is then written in the
* user's home directory.
*
* @param component the PrintableComponent to ask for
* @return null if the user doesn't want the tooltip, otherwise the text
*/
@SuppressWarnings("unused")
public static String getToolTipText(PrintableComponent component) {
String result;
int retVal;
Properties props;
String name;
Enumeration> names;
String filename;
// ToolTip is disabled for the moment...
if (true) {
return null;
}
// ask user whether the tooltip should be shown
if (!m_ToolTipUserAsked) {
m_ToolTipUserAsked = true;
retVal = JOptionPane.showConfirmDialog(component.getComponent(),
"Some panels enable the user to save the content as JPEG or EPS.\n"
+ "In order to see which panels support this, a tooltip can be "
+ "displayed. Enable tooltip?", "ToolTip for Panels...",
JOptionPane.YES_NO_OPTION);
m_ShowToolTip = (retVal == JOptionPane.YES_OPTION);
// save props file
VisualizeUtils.VISUALIZE_PROPERTIES.setProperty(PROPERTY_SHOW, ""
+ m_ShowToolTip);
VisualizeUtils.VISUALIZE_PROPERTIES.setProperty(PROPERTY_USERASKED, ""
+ m_ToolTipUserAsked);
try {
// NOTE: properties that got inherited from another props file don't
// get saved. I.e., one could overwrite the existing props
// file with an (nearly) empty one.
// => transfer all properties into a new one
props = new Properties();
names = VisualizeUtils.VISUALIZE_PROPERTIES.propertyNames();
while (names.hasMoreElements()) {
name = names.nextElement().toString();
props.setProperty(name,
VisualizeUtils.VISUALIZE_PROPERTIES.getProperty(name, ""));
}
filename = System.getProperty("user.home") + "/Visualize.props";
props.store(new BufferedOutputStream(new FileOutputStream(filename)),
null);
// inform user about location of props file and name of property
JOptionPane
.showMessageDialog(
component.getComponent(),
"You can still manually enable or disable the ToolTip via the following property\n"
+ " "
+ PROPERTY_SHOW
+ "\n"
+ "in the following file\n"
+ " " + filename);
} catch (Exception e) {
JOptionPane
.showMessageDialog(
component.getComponent(),
"Error saving the props file!\n"
+ e.getMessage()
+ "\n\n"
+ "Note:\n"
+ "If you want to disable these messages from popping up, place a file\n"
+ "called 'Visualize.props' either in your home directory or in the directory\n"
+ "you're starting Weka from and add the following lines:\n"
+ " " + PROPERTY_USERASKED + "=true\n" + " "
+ PROPERTY_SHOW + "=" + m_ShowToolTip, "Error...",
JOptionPane.ERROR_MESSAGE);
}
}
if (m_ShowToolTip) {
result = "Click left mouse button while holding and to display a save dialog.";
} else {
result = null;
}
return result;
}
/**
* initializes the filechooser, i.e. locates all the available writers in the
* current package
*/
protected void initFileChooser() {
Vector writerNames;
int i;
Class> cls;
JComponentWriter writer;
JPanel accessory;
JLabel label;
// already initialized?
if (m_FileChooserPanel != null) {
return;
}
m_FileChooserPanel = new JFileChooser();
m_FileChooserPanel.resetChoosableFileFilters();
m_FileChooserPanel.setAcceptAllFileFilterUsed(false);
// setup the accessory
accessory = new JPanel();
accessory.setLayout(null);
accessory.setPreferredSize(new Dimension(200, 200));
accessory.revalidate();
m_FileChooserPanel.setAccessory(accessory);
m_CustomDimensionsCheckBox = new JCheckBox("Use custom dimensions");
m_CustomDimensionsCheckBox.setBounds(14, 7, 200, 21);
m_CustomDimensionsCheckBox.addItemListener(new ItemListener() {
@Override
public void itemStateChanged(ItemEvent e) {
boolean custom = m_CustomDimensionsCheckBox.isSelected();
m_CustomWidthText.setEnabled(custom);
m_CustomHeightText.setEnabled(custom);
m_AspectRatioCheckBox.setEnabled(custom);
if (custom) {
m_IgnoreChange = true;
m_CustomWidthText.setText("" + m_Component.getWidth());
m_CustomHeightText.setText("" + m_Component.getHeight());
m_IgnoreChange = false;
} else {
m_IgnoreChange = true;
m_CustomWidthText.setText("-1");
m_CustomHeightText.setText("-1");
m_IgnoreChange = false;
}
}
});
accessory.add(m_CustomDimensionsCheckBox);
m_CustomWidthText = new JTextField(5);
m_CustomWidthText.setText("-1");
m_CustomWidthText.setEnabled(false);
m_CustomWidthText.setBounds(65, 35, 50, 21);
m_CustomWidthText.getDocument().addDocumentListener(new DocumentListener() {
@Override
public void changedUpdate(DocumentEvent e) {
updateDimensions(m_CustomWidthText);
}
@Override
public void insertUpdate(DocumentEvent e) {
updateDimensions(m_CustomWidthText);
}
@Override
public void removeUpdate(DocumentEvent e) {
updateDimensions(m_CustomWidthText);
}
});
label = new JLabel("Width");
label.setLabelFor(m_CustomWidthText);
label.setDisplayedMnemonic('W');
label.setBounds(14, 35, 50, 21);
accessory.add(label);
accessory.add(m_CustomWidthText);
m_CustomHeightText = new JTextField(5);
m_CustomHeightText.setText("-1");
m_CustomHeightText.setEnabled(false);
m_CustomHeightText.setBounds(65, 63, 50, 21);
m_CustomHeightText.getDocument().addDocumentListener(
new DocumentListener() {
@Override
public void changedUpdate(DocumentEvent e) {
updateDimensions(m_CustomHeightText);
}
@Override
public void insertUpdate(DocumentEvent e) {
updateDimensions(m_CustomHeightText);
}
@Override
public void removeUpdate(DocumentEvent e) {
updateDimensions(m_CustomHeightText);
}
});
label = new JLabel("Height");
label.setLabelFor(m_CustomHeightText);
label.setDisplayedMnemonic('H');
label.setBounds(14, 63, 50, 21);
accessory.add(label);
accessory.add(m_CustomHeightText);
m_AspectRatioCheckBox = new JCheckBox("Keep aspect ratio");
m_AspectRatioCheckBox.setBounds(14, 91, 200, 21);
m_AspectRatioCheckBox.setEnabled(false);
m_AspectRatioCheckBox.setSelected(true);
m_AspectRatioCheckBox.addItemListener(new ItemListener() {
@Override
public void itemStateChanged(ItemEvent e) {
boolean keep = m_AspectRatioCheckBox.isSelected();
if (keep) {
m_IgnoreChange = true;
m_CustomWidthText.setText("" + m_Component.getWidth());
m_CustomHeightText.setText("" + m_Component.getHeight());
m_IgnoreChange = false;
}
}
});
accessory.add(m_AspectRatioCheckBox);
// determine all available writers and add them to the filechooser
writerNames = GenericObjectEditor.getClassnames(JComponentWriter.class
.getName());
Collections.sort(writerNames);
for (i = 0; i < writerNames.size(); i++) {
try {
cls = Class.forName(writerNames.get(i).toString());
writer = (JComponentWriter) cls.newInstance();
m_FileChooserPanel
.addChoosableFileFilter(new JComponentWriterFileFilter(writer
.getExtension(), writer.getDescription() + " (*"
+ writer.getExtension() + ")", writer));
} catch (Exception e) {
System.err.println(writerNames.get(i) + ": " + e);
}
}
// set first filter as active filter
if (m_FileChooserPanel.getChoosableFileFilters().length > 0) {
m_FileChooserPanel.setFileFilter(m_FileChooserPanel
.getChoosableFileFilters()[0]);
}
}
/**
* updates the dimensions if necessary (i.e., if aspect ratio is to be kept).
*
* @param sender the JTextField which send the notification to update
*/
protected void updateDimensions(JTextField sender) {
int newValue;
int baseValue;
// some sanity checks
if (!m_AspectRatioCheckBox.isSelected() || m_IgnoreChange) {
return;
}
if (!(sender instanceof JTextField) || (sender == null)) {
return;
}
if (sender.getText().length() == 0) {
return;
}
// is it a valid integer, greater than 0?
try {
baseValue = Integer.parseInt(sender.getText());
newValue = 0;
if (baseValue <= 0) {
return;
}
if (Double.isNaN(m_AspectRatio)) {
m_AspectRatio = (double) getComponent().getWidth()
/ (double) getComponent().getHeight();
}
} catch (Exception e) {
// we can't parse the string!
return;
}
// computer and update
m_IgnoreChange = true;
if (sender == m_CustomWidthText) {
newValue = (int) ((baseValue) * (1 / m_AspectRatio));
m_CustomHeightText.setText("" + newValue);
} else if (sender == m_CustomHeightText) {
newValue = (int) ((baseValue) * m_AspectRatio);
m_CustomWidthText.setText("" + newValue);
}
m_IgnoreChange = false;
}
/**
* returns a Hashtable with the current available JComponentWriters in the
* save dialog. the key of the Hashtable is the description of the writer.
*
* @return all currently available JComponentWriters
* @see JComponentWriter#getDescription()
*/
@Override
public Hashtable getWriters() {
Hashtable result;
int i;
JComponentWriter writer;
result = new Hashtable();
for (i = 0; i < m_FileChooserPanel.getChoosableFileFilters().length; i++) {
writer = ((JComponentWriterFileFilter) m_FileChooserPanel
.getChoosableFileFilters()[i]).getWriter();
result.put(writer.getDescription(), writer);
}
return result;
}
/**
* returns the JComponentWriter associated with the given name, is
* null
if not found.
*
* @param name the name of the writer
* @return the writer associated with the given name
* @see JComponentWriter#getDescription()
*/
@Override
public JComponentWriter getWriter(String name) {
return getWriters().get(name);
}
/**
* sets the title for the save dialog.
*
* @param title the title of the save dialog
*/
@Override
public void setSaveDialogTitle(String title) {
m_SaveDialogTitle = title;
}
/**
* returns the title for the save dialog.
*
* @return the title of the save dialog
*/
@Override
public String getSaveDialogTitle() {
return m_SaveDialogTitle;
}
/**
* sets the scale factor.
*
* @param x the scale factor for the x-axis
* @param y the scale factor for the y-axis
*/
@Override
public void setScale(double x, double y) {
m_xScale = x;
m_yScale = y;
if (DEBUG) {
System.err.println("x = " + x + ", y = " + y);
}
}
/**
* returns the scale factor for the x-axis.
*
* @return the scale factor
*/
@Override
public double getXScale() {
return m_xScale;
}
/**
* returns the scale factor for the y-axis.
*
* @return the scale factor
*/
@Override
public double getYScale() {
return m_xScale;
}
/**
* displays a save dialog for saving the panel to a file. Fixes a bug with the
* Swing JFileChooser: if you entered a new filename in the save dialog and
* press Enter the getSelectedFile
method returns
* null
instead of the filename.
* To solve this annoying behavior we call the save dialog once again s.t. the
* filename is set. Might look a little bit strange to the user, but no
* NullPointerException! ;-)
*/
@Override
public void saveComponent() {
int result;
JComponentWriter writer;
File file;
JComponentWriterFileFilter filter;
// display save dialog
m_FileChooserPanel.setDialogTitle(getSaveDialogTitle());
do {
result = m_FileChooserPanel.showSaveDialog(getComponent());
if (result != JFileChooser.APPROVE_OPTION) {
return;
}
} while (m_FileChooserPanel.getSelectedFile() == null);
// save the file
try {
filter = (JComponentWriterFileFilter) m_FileChooserPanel.getFileFilter();
file = m_FileChooserPanel.getSelectedFile();
writer = filter.getWriter();
if (!file.getAbsolutePath().toLowerCase()
.endsWith(writer.getExtension().toLowerCase())) {
file = new File(file.getAbsolutePath() + writer.getExtension());
}
writer.setComponent(getComponent());
writer.setFile(file);
writer.setScale(getXScale(), getYScale());
writer.setUseCustomDimensions(m_CustomDimensionsCheckBox.isSelected());
if (m_CustomDimensionsCheckBox.isSelected()) {
writer.setCustomWidth(Integer.parseInt(m_CustomWidthText.getText()));
writer.setCustomHeight(Integer.parseInt(m_CustomHeightText.getText()));
} else {
writer.setCustomWidth(-1);
writer.setCustomHeight(-1);
}
writer.toOutput();
} catch (Exception e) {
e.printStackTrace();
}
}
/**
* a specialized filter that also contains the associated filter class.
*/
protected class JComponentWriterFileFilter extends ExtensionFileFilter {
/** ID added to avoid warning */
private static final long serialVersionUID = 8540426888094207515L;
/** the associated writer. */
private final JComponentWriter m_Writer;
/**
* Creates the ExtensionFileFilter.
*
* @param extension the extension of accepted files.
* @param description a text description of accepted files.
* @param writer the associated writer
*/
public JComponentWriterFileFilter(String extension, String description,
JComponentWriter writer) {
super(extension, description);
m_Writer = writer;
}
/**
* returns the associated writer.
*
* @return the writer
*/
public JComponentWriter getWriter() {
return m_Writer;
}
}
/**
* The listener to wait for Ctrl-Shft-Left Mouse Click.
*/
private class PrintMouseListener extends MouseAdapter {
/** the listener's component. */
private final PrintableComponent m_Component;
/**
* initializes the listener.
*
* @param component the component for which to create the listener
*/
public PrintMouseListener(PrintableComponent component) {
m_Component = component;
}
/**
* Invoked when the mouse has been clicked on a component.
*
* @param e the event
*/
@Override
public void mouseClicked(MouseEvent e) {
int modifiers = e.getModifiers();
if (((modifiers & MouseEvent.SHIFT_MASK) == MouseEvent.SHIFT_MASK)
&& ((modifiers & MouseEvent.ALT_MASK) == MouseEvent.ALT_MASK)
&& ((modifiers & MouseEvent.BUTTON1_MASK) == MouseEvent.BUTTON1_MASK)) {
e.consume();
m_Component.saveComponent();
}
}
}
}