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

weka.gui.knowledgeflow.StepEditorDialog 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 .
 */

/*
 *    StepEditorDialog.java
 *    Copyright (C) 2015 University of Waikato, Hamilton, New Zealand
 *
 */

package weka.gui.knowledgeflow;

import weka.core.Environment;
import weka.core.EnvironmentHandler;
import weka.core.Utils;
import weka.gui.PropertyDialog;
import weka.gui.SettingsEditor;
import weka.knowledgeflow.steps.Step;

import javax.swing.BorderFactory;
import javax.swing.JButton;
import javax.swing.JDialog;
import javax.swing.JPanel;
import javax.swing.JScrollPane;
import javax.swing.JTextArea;
import javax.swing.JTextPane;
import java.awt.BorderLayout;
import java.awt.Dimension;
import java.awt.Font;
import java.awt.Frame;
import java.awt.GridLayout;
import java.awt.Window;
import java.awt.event.ActionEvent;
import java.awt.event.ActionListener;
import java.awt.event.WindowAdapter;
import java.awt.event.WindowEvent;
import java.io.IOException;
import java.lang.reflect.Method;

/**
 * Base class for step editor dialogs. Clients can extend this in order to
 * supply their own custom dialogs. The constructor for this class provides OK
 * and Cancel buttons in the SOUTH location of a BorderLayout. To do meaningful
 * things on click of OK or Cancel, subclasses should override okPressed()
 * and/or cancelPressed(). If setStepToEdit() is not overridden, then this class
 * will also provide an "about" panel in the NORTH location of BorderLayout.
 * This then leaves the CENTER of the BorderLayout for custom widgets.
 * Subclasses should typically override the no-op layoutEditor() method to add
 * their widgets.
 *
 * @author Mark Hall (mhall{[at]}pentaho{[dot]}com)
 * @version $Revision: $
 */
public abstract class StepEditorDialog extends JPanel implements
  EnvironmentHandler {

  /** For serialization */
  private static final long serialVersionUID = -4860182109190301676L;

  /** True if the step's properties have been altered */
  protected boolean m_isEdited;

  /** Environment variables */
  protected Environment m_env = Environment.getSystemWide();

  /** Holder for buttons */
  protected JPanel m_buttonHolder = new JPanel(new GridLayout(1, 0));

  /** OK button */
  protected JButton m_okBut = new JButton("OK");

  /** Cancel button */
  protected JButton m_cancelBut = new JButton("Cancel");

  /** Settings button */
  protected JButton m_settingsBut = new JButton("Settings");

  /** Reference to the main perspective */
  protected MainKFPerspective m_mainPerspective;

  /** Parent window */
  protected Window m_parent;

  /** Listener to be informed when the window closes */
  protected ClosingListener m_closingListener;

  /** The step to edit */
  protected Step m_stepToEdit;

  /** Buffer to hold the help text */
  protected StringBuilder m_helpText = new StringBuilder();

  /** About button */
  protected JButton m_helpBut = new JButton("About");

  /** The handler for graphical commands */
  protected KFGraphicalEnvironmentCommandHandler m_commandHandler;

  /**
   * Constructor
   */
  public StepEditorDialog() {
    setLayout(new BorderLayout());

    m_buttonHolder.add(m_okBut);
    m_buttonHolder.add(m_cancelBut);
    add(m_buttonHolder, BorderLayout.SOUTH);

    m_okBut.addActionListener(new ActionListener() {
      @Override
      public void actionPerformed(ActionEvent e) {
        ok();
      }
    });

    m_cancelBut.addActionListener(new ActionListener() {
      @Override
      public void actionPerformed(ActionEvent e) {
        cancel();
      }
    });
  }

  /**
   * Set the main perspective
   *
   * @param main the main perspective
   */
  protected void setMainPerspective(MainKFPerspective main) {
    m_mainPerspective = main;
    m_commandHandler =
      new KFGraphicalEnvironmentCommandHandler(m_mainPerspective);
  }

  /**
   * Get the main knowledgeflow perspective
   *
   * @return the main knowledge flow perspective
   */
  protected MainKFPerspective getMainPerspective() {
    return m_mainPerspective;
  }

  /**
   * Get the environment for performing commands at the application-level in a
   * graphical environment.
   *
   * @return the graphical environment command handler.
   */
  protected KFGraphicalEnvironmentCommandHandler
    getGraphicalEnvironmentCommandHandler() {
    return m_commandHandler;
  }

  /**
   * Show an error dialog
   *
   * @param cause an exception to show in the dialog
   */
  protected void showErrorDialog(Exception cause) {
    m_mainPerspective.showErrorDialog(cause);
  }

  /**
   * Show an information dialog
   * 
   * @param information the information to show
   * @param title the title for the dialog
   * @param isWarning true if this is a warning rather than general information
   */
  protected void showInfoDialog(Object information, String title,
    boolean isWarning) {
    m_mainPerspective.showInfoDialog(information, title, isWarning);
  }

  /**
   * Get the step being edited
   *
   * @return
   */
  protected Step getStepToEdit() {
    return m_stepToEdit;
  }

  /**
   * Set the step to edit
   *
   * @param step the step to edit
   */
  protected void setStepToEdit(Step step) {
    m_stepToEdit = step;

    createAboutPanel(step);
    if (step.getDefaultSettings() != null) {
      addSettingsButton();
    }
    layoutEditor();
  }

  /**
   * Layout the editor. This is a no-op method that subclasses should override
   */
  protected void layoutEditor() {
    // subclasses can override to add their
    // stuff to the center of the borderlayout
  }

  /**
   * Adds a button for popping up a settings editor. Is only visible if the step
   * being edited provides default settings
   */
  protected void addSettingsButton() {
    getMainPerspective().getMainApplication().getApplicationSettings()
      .applyDefaults(getStepToEdit().getDefaultSettings());
    m_buttonHolder.add(m_settingsBut);
    m_settingsBut.addActionListener(new ActionListener() {
      @Override
      public void actionPerformed(ActionEvent e) {
        try {
          SettingsEditor.showSingleSettingsEditor(getMainPerspective()
            .getMainApplication().getApplicationSettings(), getStepToEdit()
            .getDefaultSettings().getID(), getStepToEdit().getName(),
            StepEditorDialog.this);
        } catch (IOException ex) {
          showErrorDialog(ex);
        }
      }
    });
  }

  /**
   * Set the parent window of this dialog
   *
   * @param parent the parent window
   */
  protected void setParentWindow(Window parent) {
    m_parent = parent;
  }

  /**
   * Set a closing listener
   *
   * @param c the closing listener
   */
  protected void setClosingListener(ClosingListener c) {
    m_closingListener = c;
  }

  /**
   * Returns true if the properties of the step being edited have been changed
   *
   * @return true if the step has been edited
   */
  protected boolean isEdited() {
    return m_isEdited;
  }

  /**
   * Set whether the step's properties have changed or not
   *
   * @param edited true if the step has been edited
   */
  protected void setEdited(boolean edited) {
    m_isEdited = edited;
  }

  private void ok() {
    setEdited(true);

    okPressed();
    if (m_parent != null) {
      m_parent.dispose();
    }

    if (m_closingListener != null) {
      m_closingListener.closing();
    }
  }

  /**
   * Called when the OK button is pressed. This is a no-op method - subclasses
   * should override
   */
  protected void okPressed() {
    // subclasses to override
  }

  /**
   * Called when the Cancel button is pressed. This is a no-op method -
   * subclasses should override
   */
  protected void cancelPressed() {
    // subclasses to override
  }

  private void cancel() {
    setEdited(false);

    cancelPressed();
    if (m_parent != null) {
      m_parent.dispose();
    }

    if (m_closingListener != null) {
      m_closingListener.closing();
    }
  }

  /**
   * Creates an "about" panel to add to the dialog
   *
   * @param step the step from which to extract help info
   */
  protected void createAboutPanel(Step step) {
    String globalFirstSentence = "";
    String globalInfo = Utils.getGlobalInfo(step, false);
    if (globalInfo == null) {
      globalInfo = "No info available";
      globalFirstSentence = globalInfo;
    } else {
      globalInfo = globalInfo.replace("font color=blue", "font color=black");
      try {
        Method gI = step.getClass().getMethod("globalInfo");
        String globalInfoNoHTML = gI.invoke(step).toString();
        globalFirstSentence =
          globalInfoNoHTML.contains(".") ? globalInfoNoHTML.substring(0,
            globalInfoNoHTML.indexOf('.')) : globalInfoNoHTML;
      } catch (Exception ex) {
        ex.printStackTrace();
      }
    }

    createAboutPanel(globalInfo, globalFirstSentence);
  }

  private void createAboutPanel(String about, String firstSentence) {
    JTextArea jt = new JTextArea();

    m_helpText.append(about);
    jt.setColumns(30);
    // jt.setContentType("text/html");
    jt.setFont(new Font("SansSerif", Font.PLAIN, 12));
    jt.setEditable(false);
    jt.setLineWrap(true);
    jt.setWrapStyleWord(true);

    jt.setText(firstSentence);
    jt.setBackground(getBackground());

    String className = m_stepToEdit.getClass().getName();
    className =
      className.substring(className.lastIndexOf('.') + 1, className.length());
    m_helpBut.setToolTipText("More information about " + className);

    final JPanel jp = new JPanel();
    jp.setBorder(BorderFactory.createCompoundBorder(
      BorderFactory.createTitledBorder("About"),
      BorderFactory.createEmptyBorder(5, 5, 5, 5)));
    jp.setLayout(new BorderLayout());
    jp.add(new JScrollPane(jt), BorderLayout.CENTER);

    JPanel p2 = new JPanel();
    p2.setLayout(new BorderLayout());
    p2.add(m_helpBut, BorderLayout.NORTH);
    jp.add(p2, BorderLayout.EAST);

    add(jp, BorderLayout.NORTH);

    int preferredWidth = jt.getPreferredSize().width;
    jt.setSize(new Dimension(Math.min(preferredWidth, 600), Short.MAX_VALUE));
    Dimension d = jt.getPreferredSize();
    jt.setPreferredSize(new Dimension(Math.min(preferredWidth, 600), d.height));

    m_helpBut.addActionListener(new ActionListener() {
      @Override
      public void actionPerformed(ActionEvent a) {
        openHelpFrame(jp);
        m_helpBut.setEnabled(false);
      }
    });
  }

  private void openHelpFrame(JPanel aboutPanel) {
    JTextPane ta = new JTextPane();
    ta.setBorder(BorderFactory.createEmptyBorder(5, 5, 5, 5));
    ta.setContentType("text/html");
    // ta.setLineWrap(true);
    // ta.setWrapStyleWord(true);
    // ta.setBackground(getBackground());
    ta.setEditable(false);
    ta.setText(m_helpText.toString());
    ta.setCaretPosition(0);
    JDialog jdtmp;
    if (PropertyDialog.getParentDialog(this) != null) {
      jdtmp = new JDialog(PropertyDialog.getParentDialog(this), "Information");
    } else if (PropertyDialog.getParentFrame(this) != null) {
      jdtmp = new JDialog(PropertyDialog.getParentFrame(this), "Information");
    } else {
      jdtmp = new JDialog((Frame) null, "Information");
    }
    final JDialog jd = jdtmp;
    jd.addWindowListener(new WindowAdapter() {
      @Override
      public void windowClosing(WindowEvent e) {
        jd.dispose();
        m_helpBut.setEnabled(true);
      }
    });
    jd.getContentPane().setLayout(new BorderLayout());
    jd.getContentPane().add(new JScrollPane(ta), BorderLayout.CENTER);
    jd.pack();
    jd.setSize(400, 350);
    jd.setLocation(aboutPanel.getTopLevelAncestor().getLocationOnScreen().x
      + aboutPanel.getTopLevelAncestor().getSize().width, aboutPanel
      .getTopLevelAncestor().getLocationOnScreen().y);
    jd.setVisible(true);
  }

  /**
   * Get environment variables
   *
   * @return environment variables
   */
  public Environment getEnvironment() {
    return m_env;
  }

  /**
   * Set environment variables
   *
   * @param env the environment variables to use
   */
  @Override
  public void setEnvironment(Environment env) {
    m_env = env;
  }

  /**
   * Substitute the values of any environment variables present in the supplied
   * string
   *
   * @param source the source string to substitute vars in
   * @return the string with any environment variables substituted
   */
  public String environmentSubstitute(String source) {
    String result = source;
    if (result != null) {
      try {
        result = m_env.substitute(result);
      } catch (Exception ex) {
        // ignore
      }
    }

    return result;
  }

  /**
   * Interface for those that want to be notified when this dialog closes
   */
  public interface ClosingListener {
    void closing();
  }
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy