weka.gui.knowledgeflow.LayoutPanel 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 .
*/
/*
* LayoutPanel.java
* Copyright (C) 2015 University of Waikato, Hamilton, New Zealand
*
*/
package weka.gui.knowledgeflow;
import weka.core.EnvironmentHandler;
import weka.core.Instances;
import weka.core.WekaException;
import weka.core.WekaPackageClassLoaderManager;
import weka.core.converters.FileSourcedConverter;
import weka.gui.Perspective;
import weka.gui.knowledgeflow.VisibleLayout.LayoutOperation;
import weka.gui.visualize.PrintablePanel;
import weka.knowledgeflow.KFDefaults;
import weka.knowledgeflow.StepManager;
import weka.knowledgeflow.StepManagerImpl;
import weka.knowledgeflow.steps.Loader;
import weka.knowledgeflow.steps.Note;
import javax.swing.JComponent;
import javax.swing.JDialog;
import javax.swing.JFrame;
import javax.swing.JOptionPane;
import java.awt.BasicStroke;
import java.awt.BorderLayout;
import java.awt.Color;
import java.awt.Cursor;
import java.awt.Dialog.ModalityType;
import java.awt.Dimension;
import java.awt.Font;
import java.awt.FontMetrics;
import java.awt.Graphics;
import java.awt.Graphics2D;
import java.awt.Menu;
import java.awt.MenuItem;
import java.awt.Point;
import java.awt.PopupMenu;
import java.awt.RenderingHints;
import java.awt.Stroke;
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.awt.event.MouseMotionAdapter;
import java.io.File;
import java.io.IOException;
import java.util.ArrayList;
import java.util.List;
import java.util.Map;
import java.util.Map.Entry;
/**
* Provides a panel just for laying out a Knowledge Flow graph. Also listens for
* mouse events for editing the flow.
*
* @author Mark Hall (mhall{[at]}pentaho{[dot]}com)
* @version $Revision: $
*/
public class LayoutPanel extends PrintablePanel {
/** For serialization */
private static final long serialVersionUID = 4988098224376217099L;
/** Grid spacing */
protected int m_gridSpacing;
/** The flow contained in this LayoutPanel as a visible (graphical) flow */
protected VisibleLayout m_visLayout;
protected int m_currentX;
protected int m_currentY;
protected int m_oldX;
protected int m_oldY;
/** Thread for loading data for perspectives */
protected Thread m_perspectiveDataLoadThread;
/**
* Constructor
*
* @param vis the {@code VisibleLayout} to display
*/
public LayoutPanel(VisibleLayout vis) {
super();
m_visLayout = vis;
setLayout(null);
setupMouseListener();
setupMouseMotionListener();
m_gridSpacing =
m_visLayout.getMainPerspective().getSetting(KFDefaults.GRID_SPACING_KEY,
KFDefaults.GRID_SPACING);
}
/**
* Configure mouse listener
*/
protected void setupMouseListener() {
addMouseListener(new MouseAdapter() {
@Override
public void mousePressed(MouseEvent me) {
LayoutPanel.this.requestFocusInWindow();
double z = m_visLayout.getZoomSetting() / 100.0;
double px = me.getX();
double py = me.getY();
py /= z;
px /= z;
if (m_visLayout.getMainPerspective().getPalleteSelectedStep() == null) {
if (((me.getModifiers() & InputEvent.BUTTON1_MASK) == InputEvent.BUTTON1_MASK)
&& m_visLayout.getFlowLayoutOperation() == VisibleLayout.LayoutOperation.NONE) {
StepVisual step =
m_visLayout.findStep(new Point((int) px, (int) py));
if (step != null) {
m_visLayout.setEditStep(step);
m_oldX = (int) px;
m_oldY = (int) py;
m_visLayout.setFlowLayoutOperation(LayoutOperation.MOVING);
}
if (m_visLayout.getFlowLayoutOperation() != LayoutOperation.MOVING) {
m_visLayout.setFlowLayoutOperation(LayoutOperation.SELECTING);
m_oldX = (int) px;
m_oldY = (int) py;
m_currentX = m_oldX;
m_currentY = m_oldY;
Graphics2D gx = (Graphics2D) LayoutPanel.this.getGraphics();
gx.setXORMode(java.awt.Color.white);
gx.dispose();
}
}
}
}
@Override
public void mouseReleased(MouseEvent me) {
LayoutPanel.this.requestFocusInWindow();
if (m_visLayout.getEditStep() != null
&& m_visLayout.getFlowLayoutOperation() == LayoutOperation.MOVING) {
if (m_visLayout.getMainPerspective().getSnapToGrid()) {
int x = snapToGrid(m_visLayout.getEditStep().getX());
int y = snapToGrid(m_visLayout.getEditStep().getY());
m_visLayout.getEditStep().setX(x);
m_visLayout.getEditStep().setY(y);
snapSelectedToGrid();
}
m_visLayout.setEditStep(null);
revalidate();
repaint();
m_visLayout.setFlowLayoutOperation(LayoutOperation.NONE);
}
if (m_visLayout.getFlowLayoutOperation() == LayoutOperation.SELECTING) {
revalidate();
repaint();
m_visLayout.setFlowLayoutOperation(LayoutOperation.NONE);
double z = m_visLayout.getZoomSetting() / 100.0;
double px = me.getX();
double py = me.getY();
py /= z;
px /= z;
highlightSubFlow(m_currentX, m_currentY, (int) px, (int) py);
}
}
@Override
public void mouseClicked(MouseEvent me) {
LayoutPanel.this.requestFocusInWindow();
Point p = me.getPoint();
Point np = new Point();
double z = m_visLayout.getZoomSetting() / 100.0;
double px = me.getX();
double py = me.getY();
px /= z;
py /= z;
np.setLocation(p.getX() / z, p.getY() / z);
StepVisual step = m_visLayout.findStep(np);
if (m_visLayout.getFlowLayoutOperation() == LayoutOperation.ADDING
|| m_visLayout.getFlowLayoutOperation() == LayoutOperation.NONE) {
// try and popup a context sensitive menu if we've been
// clicked over a step
if (step != null) {
if (me.getClickCount() == 2) {
if (!step.getStepManager().isStepBusy()
&& !m_visLayout.isExecuting()) {
popupStepEditorDialog(step);
}
} else if ((me.getModifiers() & InputEvent.BUTTON1_MASK) != InputEvent.BUTTON1_MASK
|| me.isAltDown()) {
stepContextualMenu(step, (int) (p.getX() / z),
(int) (p.getY() / z));
return;
} else {
// just select this step
List v = m_visLayout.getSelectedSteps();
if (!me.isShiftDown()) {
v = new ArrayList();
}
v.add(step);
m_visLayout.setSelectedSteps(v);
return;
}
} else {
if ((me.getModifiers() & InputEvent.BUTTON1_MASK) != InputEvent.BUTTON1_MASK
|| me.isAltDown()) {
if (!m_visLayout.isExecuting()) {
canvasContextualMenu((int) px, (int) py);
revalidate();
repaint();
m_visLayout.getMainPerspective().notifyIsDirty();
}
return;
} else if (m_visLayout.getMainPerspective()
.getPalleteSelectedStep() != null) {
// if there is a user-selected step from the design palette then
// add the step
// snap to grid
double x = px;
double y = py;
if (m_visLayout.getMainPerspective().getSnapToGrid()) {
x = snapToGrid((int) x);
y = snapToGrid((int) y);
}
m_visLayout.addUndoPoint();
m_visLayout.addStep(m_visLayout.getMainPerspective()
.getPalleteSelectedStep(), (int) x, (int) y);
m_visLayout.getMainPerspective().clearDesignPaletteSelection();
m_visLayout.getMainPerspective().setPalleteSelectedStep(null);
m_visLayout.setFlowLayoutOperation(LayoutOperation.NONE);
m_visLayout.setEdited(true);
}
}
revalidate();
repaint();
m_visLayout.getMainPerspective().notifyIsDirty();
}
if (m_visLayout.getFlowLayoutOperation() == LayoutOperation.PASTING
&& m_visLayout.getMainPerspective().getPasteBuffer().length() > 0) {
try {
m_visLayout.pasteFromClipboard((int) px, (int) py);
} catch (WekaException e) {
m_visLayout.getMainPerspective().showErrorDialog(e);
}
m_visLayout.setFlowLayoutOperation(LayoutOperation.NONE);
m_visLayout.getMainPerspective().setCursor(
Cursor.getPredefinedCursor(Cursor.DEFAULT_CURSOR));
return;
}
if (m_visLayout.getFlowLayoutOperation() == LayoutOperation.CONNECTING) {
// turn off connector points and remove connecting line
repaint();
for (StepVisual v : m_visLayout.getRenderGraph()) {
v.setDisplayConnectors(false);
}
if (step != null
&& step.getStepManager() != m_visLayout.getEditStep()
.getStepManager()) {
// connection is valid because only valid connections will
// have appeared in the contextual popup
m_visLayout.addUndoPoint();
m_visLayout.connectSteps(
m_visLayout.getEditStep().getStepManager(),
step.getStepManager(), m_visLayout.getEditConnection());
m_visLayout.setEdited(true);
repaint();
}
m_visLayout.setFlowLayoutOperation(LayoutOperation.NONE);
m_visLayout.setEditStep(null);
m_visLayout.setEditConnection(null);
}
if (m_visLayout.getSelectedSteps().size() > 0) {
m_visLayout.setSelectedSteps(new ArrayList());
}
}
});
}
/**
* Configure mouse motion listener
*/
protected void setupMouseMotionListener() {
addMouseMotionListener(new MouseMotionAdapter() {
@Override
public void mouseDragged(MouseEvent me) {
double z = m_visLayout.getZoomSetting() / 100.0;
double px = me.getX();
double py = me.getY();
px /= z;
py /= z;
if (m_visLayout.getEditStep() != null
&& m_visLayout.getFlowLayoutOperation() == LayoutOperation.MOVING) {
int deltaX = (int) px - m_oldX;
int deltaY = (int) py - m_oldY;
m_visLayout.getEditStep().setX(
m_visLayout.getEditStep().getX() + deltaX);
m_visLayout.getEditStep().setY(
m_visLayout.getEditStep().getY() + deltaY);
for (StepVisual v : m_visLayout.getSelectedSteps()) {
if (v != m_visLayout.getEditStep()) {
v.setX(v.getX() + deltaX);
v.setY(v.getY() + deltaY);
}
}
repaint();
m_oldX = (int) px;
m_oldY = (int) py;
m_visLayout.setEdited(true);
}
if (m_visLayout.getFlowLayoutOperation() == LayoutOperation.SELECTING) {
repaint();
m_oldX = (int) px;
m_oldY = (int) py;
}
}
@Override
public void mouseMoved(MouseEvent me) {
double z = m_visLayout.getZoomSetting() / 100.0;
double px = me.getX();
double py = me.getY();
px /= z;
py /= z;
if (m_visLayout.getFlowLayoutOperation() == LayoutOperation.CONNECTING) {
repaint();
m_oldX = (int) px;
m_oldY = (int) py;
}
}
});
}
@Override
public void paintComponent(Graphics gx) {
Color backG =
m_visLayout.getMainPerspective().getSetting(KFDefaults.LAYOUT_COLOR_KEY,
KFDefaults.LAYOUT_COLOR);
if (!backG.equals(getBackground())) {
setBackground(backG);
}
double lz = m_visLayout.getZoomSetting() / 100.0;
((Graphics2D) gx).scale(lz, lz);
if (m_visLayout.getZoomSetting() < 100) {
((Graphics2D) gx).setStroke(new BasicStroke(2));
}
super.paintComponent(gx);
((Graphics2D) gx).setRenderingHint(RenderingHints.KEY_ANTIALIASING,
RenderingHints.VALUE_ANTIALIAS_ON);
((Graphics2D) gx).setRenderingHint(RenderingHints.KEY_TEXT_ANTIALIASING,
RenderingHints.VALUE_TEXT_ANTIALIAS_GASP);
paintStepLabels(gx);
paintConnections(gx);
if (m_visLayout.getFlowLayoutOperation() == LayoutOperation.CONNECTING) {
gx.drawLine(m_currentX, m_currentY, m_oldX, m_oldY);
} else if (m_visLayout.getFlowLayoutOperation() == LayoutOperation.SELECTING) {
gx.drawRect(m_currentX < m_oldX ? m_currentX : m_oldX,
m_currentY < m_oldY ? m_currentY : m_oldY,
Math.abs(m_oldX - m_currentX), Math.abs(m_oldY - m_currentY));
}
if (m_visLayout.getMainPerspective().getSetting(KFDefaults.SHOW_GRID_KEY,
KFDefaults.SHOW_GRID)) {
Color gridColor =
m_visLayout.getMainPerspective().getSetting(KFDefaults.GRID_COLOR_KEY,
KFDefaults.GRID_COLOR);
gx.setColor(gridColor);
int gridSpacing =
m_visLayout.getMainPerspective().getSetting(
KFDefaults.GRID_SPACING_KEY, KFDefaults.GRID_SPACING);
int layoutWidth =
m_visLayout.getMainPerspective().getSetting(
KFDefaults.LAYOUT_WIDTH_KEY, KFDefaults.LAYOUT_WIDTH);
int layoutHeight =
m_visLayout.getMainPerspective().getSetting(
KFDefaults.LAYOUT_HEIGHT_KEY, KFDefaults.LAYOUT_HEIGHT);
Stroke original = ((Graphics2D) gx).getStroke();
Stroke dashed =
new BasicStroke(1, BasicStroke.CAP_BUTT, BasicStroke.JOIN_BEVEL, 0,
new float[] { 3 }, 0);
((Graphics2D) gx).setStroke(dashed);
for (int i = gridSpacing; i < layoutWidth / lz; i += gridSpacing) {
gx.drawLine(i, 0, i, (int) (layoutHeight / lz));
}
for (int i = gridSpacing; i < layoutHeight / lz; i += gridSpacing) {
gx.drawLine(0, i, (int) (layoutWidth / lz), i);
}
((Graphics2D) gx).setStroke(original);
}
}
@Override
public void doLayout() {
super.doLayout();
for (StepVisual v : m_visLayout.getRenderGraph()) {
Dimension d = v.getPreferredSize();
v.setBounds(v.getX(), v.getY(), d.width, d.height);
v.revalidate();
}
}
/**
* Render the labels for each step in the layout
*
* @param gx the graphics context to use
*/
protected void paintStepLabels(Graphics gx) {
gx.setFont(new Font(null, Font.PLAIN, m_visLayout.getMainPerspective()
.getSetting(KFDefaults.STEP_LABEL_FONT_SIZE_KEY,
KFDefaults.STEP_LABEL_FONT_SIZE)));
FontMetrics fm = gx.getFontMetrics();
int hf = fm.getAscent();
for (StepVisual v : m_visLayout.getRenderGraph()) {
if (!v.getDisplayStepLabel()) {
continue;
}
int cx = v.getX();
int cy = v.getY();
int width = v.getWidth();
int height = v.getHeight();
String label = v.getStepName();
int labelwidth = fm.stringWidth(label);
if (labelwidth < width) {
gx.drawString(label, (cx + (width / 2)) - (labelwidth / 2), cy + height
+ hf + 2);
} else {
// split label
// find mid point
int mid = label.length() / 2;
// look for split point closest to the mid
int closest = label.length();
int closestI = -1;
for (int z = 0; z < label.length(); z++) {
if (label.charAt(z) < 'a') {
if (Math.abs(mid - z) < closest) {
closest = Math.abs(mid - z);
closestI = z;
}
}
}
if (closestI != -1) {
String left = label.substring(0, closestI);
String right = label.substring(closestI, label.length());
if (left.length() > 1 && right.length() > 1) {
gx.drawString(left,
(cx + (width / 2)) - (fm.stringWidth(left) / 2), cy + height
+ (hf * 1) + 2);
gx.drawString(right, (cx + (width / 2))
- (fm.stringWidth(right) / 2), cy + height + (hf * 2) + 2);
} else {
gx.drawString(label, (cx + (width / 2))
- (fm.stringWidth(label) / 2), cy + height + (hf * 1) + 2);
}
} else {
gx.drawString(label,
(cx + (width / 2)) - (fm.stringWidth(label) / 2), cy + height
+ (hf * 1) + 2);
}
}
}
}
/**
* Render the connections between steps
*
* @param gx the graphics object to use
*/
protected void paintConnections(Graphics gx) {
for (StepVisual stepVis : m_visLayout.getRenderGraph()) {
Map> outConns =
stepVis.getStepManager().getOutgoingConnections();
if (outConns.size() > 0) {
List generatableOutputConnections =
stepVis.getStepManager().getStepOutgoingConnectionTypes();
// iterate over the outgoing connections and paint
// with color according to what is present in
// generatableOutputConnections
int count = 0;
for (Entry> e : outConns.entrySet()) {
String connName = e.getKey();
List connectedSteps = e.getValue();
if (connectedSteps.size() > 0) {
int sX = stepVis.getX();
int sY = stepVis.getY();
int sWidth = stepVis.getWidth();
int sHeight = stepVis.getHeight();
for (StepManager target : connectedSteps) {
StepManagerImpl targetI = (StepManagerImpl) target;
int tX = targetI.getStepVisual().getX();
int tY = targetI.getStepVisual().getY();
int tWidth = targetI.getStepVisual().getWidth();
int tHeight = targetI.getStepVisual().getHeight();
Point bestSourcePoint =
stepVis.getClosestConnectorPoint(new Point(tX + (tWidth / 2),
tY + (tHeight / 2)));
Point bestTargetPoint =
targetI.getStepVisual().getClosestConnectorPoint(
new Point(sX + (sWidth / 2), sY + (sHeight / 2)));
gx.setColor(Color.red);
boolean active =
generatableOutputConnections == null
|| !generatableOutputConnections.contains(connName) ? false
: true;
if (!active) {
gx.setColor(Color.gray);
}
gx.drawLine((int) bestSourcePoint.getX(),
(int) bestSourcePoint.getY(), (int) bestTargetPoint.getX(),
(int) bestTargetPoint.getY());
// paint an arrow head
double angle;
try {
double a =
(bestSourcePoint.getY() - bestTargetPoint.getY())
/ (bestSourcePoint.getX() - bestTargetPoint.getX());
angle = Math.atan(a);
} catch (Exception ex) {
angle = Math.PI / 2;
}
Point arrowstart =
new Point(bestTargetPoint.x, bestTargetPoint.y);
Point arrowoffset =
new Point((int) (7 * Math.cos(angle)),
(int) (7 * Math.sin(angle)));
Point arrowend;
if (bestSourcePoint.getX() >= bestTargetPoint.getX()) {
arrowend =
new Point(arrowstart.x + arrowoffset.x, arrowstart.y
+ arrowoffset.y);
} else {
arrowend =
new Point(arrowstart.x - arrowoffset.x, arrowstart.y
- arrowoffset.y);
}
int xs[] =
{ arrowstart.x,
arrowend.x + (int) (7 * Math.cos(angle + (Math.PI / 2))),
arrowend.x + (int) (7 * Math.cos(angle - (Math.PI / 2))) };
int ys[] =
{ arrowstart.y,
arrowend.y + (int) (7 * Math.sin(angle + (Math.PI / 2))),
arrowend.y + (int) (7 * Math.sin(angle - (Math.PI / 2))) };
gx.fillPolygon(xs, ys, 3);
// paint the connection name
int midx = (int) bestSourcePoint.getX();
midx +=
(int) ((bestTargetPoint.getX() - bestSourcePoint.getX()) / 2);
int midy = (int) bestSourcePoint.getY();
midy +=
(int) ((bestTargetPoint.getY() - bestSourcePoint.getY()) / 2) - 2;
gx.setColor((active) ? Color.blue : Color.gray);
// check to see if there is more than one connection
// between the source and target
if (m_visLayout.previousConn(outConns, targetI, count)) {
midy -= 15;
}
gx.drawString(connName, midx, midy);
}
}
count++;
}
}
}
}
/**
* Popup a contextual menu on the canvas that provides options for cutting,
* pasting, deleting selected steps etc.
*
* @param x the x coordinate to pop up at
* @param y the y coordinate to pop up at
*/
protected void canvasContextualMenu(final int x, final int y) {
Map> closestConnections =
m_visLayout.findClosestConnections(new Point(x, y), 10);
PopupMenu contextualMenu = new PopupMenu();
int menuItemCount = 0;
if (m_visLayout.getSelectedSteps().size() > 0) {
MenuItem snapItem = new MenuItem("Snap selected to grid");
snapItem.addActionListener(new ActionListener() {
@Override
public void actionPerformed(ActionEvent e) {
snapSelectedToGrid();
}
});
contextualMenu.add(snapItem);
menuItemCount++;
MenuItem copyItem = new MenuItem("Copy selected");
copyItem.addActionListener(new ActionListener() {
@Override
public void actionPerformed(ActionEvent e) {
try {
m_visLayout.copySelectedStepsToClipboard();
m_visLayout.setSelectedSteps(new ArrayList());
} catch (WekaException ex) {
m_visLayout.getMainPerspective().showErrorDialog(ex);
}
}
});
contextualMenu.add(copyItem);
menuItemCount++;
MenuItem cutItem = new MenuItem("Cut selected");
cutItem.addActionListener(new ActionListener() {
@Override
public void actionPerformed(ActionEvent e) {
try {
m_visLayout.copySelectedStepsToClipboard();
m_visLayout.removeSelectedSteps();
} catch (WekaException ex) {
m_visLayout.getMainPerspective().showErrorDialog(ex);
}
}
});
contextualMenu.add(cutItem);
menuItemCount++;
MenuItem deleteSelected = new MenuItem("Delete selected");
deleteSelected.addActionListener(new ActionListener() {
@Override
public void actionPerformed(ActionEvent e) {
try {
m_visLayout.removeSelectedSteps();
} catch (WekaException ex) {
m_visLayout.getMainPerspective().showErrorDialog(ex);
}
}
});
contextualMenu.add(deleteSelected);
menuItemCount++;
}
if (m_visLayout.getMainPerspective().getPasteBuffer() != null
&& m_visLayout.getMainPerspective().getPasteBuffer().length() > 0) {
contextualMenu.addSeparator();
menuItemCount++;
MenuItem pasteItem = new MenuItem("Paste");
pasteItem.addActionListener(new ActionListener() {
@Override
public void actionPerformed(ActionEvent e) {
try {
m_visLayout.pasteFromClipboard(x, y);
} catch (WekaException ex) {
m_visLayout.getMainPerspective().showErrorDialog(ex);
}
}
});
contextualMenu.add(pasteItem);
menuItemCount++;
}
if (closestConnections.size() > 0) {
contextualMenu.addSeparator();
menuItemCount++;
MenuItem deleteConnection = new MenuItem("Delete connection:");
deleteConnection.setEnabled(false);
contextualMenu.insert(deleteConnection, menuItemCount++);
for (Map.Entry> e : closestConnections
.entrySet()) {
final String connName = e.getKey();
for (StepManagerImpl[] cons : e.getValue()) {
final StepManagerImpl source = cons[0];
final StepManagerImpl target = cons[1];
MenuItem deleteItem =
new MenuItem(connName + "-->" + target.getName());
deleteItem.addActionListener(new ActionListener() {
@Override
public void actionPerformed(ActionEvent e) {
m_visLayout.addUndoPoint();
source.disconnectStepWithConnection(target.getManagedStep(),
connName);
target.disconnectStepWithConnection(source.getManagedStep(),
connName);
if (m_visLayout.getSelectedSteps().size() > 0) {
m_visLayout.setSelectedSteps(new ArrayList());
}
m_visLayout.setEdited(true);
revalidate();
repaint();
m_visLayout.getMainPerspective().notifyIsDirty();
}
});
contextualMenu.add(deleteItem);
menuItemCount++;
}
}
}
if (menuItemCount > 0) {
contextualMenu.addSeparator();
menuItemCount++;
}
MenuItem noteItem = new MenuItem("New note");
noteItem.addActionListener(new ActionListener() {
@Override
public void actionPerformed(ActionEvent e) {
initiateAddNote();
}
});
contextualMenu.add(noteItem);
this.add(contextualMenu);
// make sure that popup location takes current scaling into account
double z = m_visLayout.getZoomSetting() / 100.0;
double px = x * z;
double py = y * z;
contextualMenu.show(this, (int) px, (int) py);
}
/**
* Initiate the process of adding a note to the canvas
*/
protected void initiateAddNote() {
Note n = new Note();
StepManagerImpl noteManager = new StepManagerImpl(n);
StepVisual noteVisual = StepVisual.createVisual(noteManager);
m_visLayout.getMainPerspective().setPalleteSelectedStep(
noteVisual.getStepManager());
m_visLayout.setFlowLayoutOperation(LayoutOperation.ADDING);
m_visLayout.getMainPerspective().setCursor(
Cursor.getPredefinedCursor(Cursor.CROSSHAIR_CURSOR));
}
/**
* Popup the contextual menu for when a step is right clicked on
*
* @param step the step that was clicked on
* @param x the x coordiante to pop up at
* @param y the y coordinate to pop up at
*/
protected void stepContextualMenu(final StepVisual step, final int x,
final int y) {
PopupMenu stepContextMenu = new PopupMenu();
boolean executing = m_visLayout.isExecuting();
int menuItemCount = 0;
MenuItem edit = new MenuItem("Edit:");
edit.setEnabled(false);
stepContextMenu.insert(edit, menuItemCount);
menuItemCount++;
if (m_visLayout.getSelectedSteps().size() > 0) {
MenuItem copyItem = new MenuItem("Copy");
copyItem.addActionListener(new ActionListener() {
@Override
public void actionPerformed(ActionEvent e) {
try {
m_visLayout.copySelectedStepsToClipboard();
} catch (WekaException ex) {
m_visLayout.getMainPerspective().showErrorDialog(ex);
}
m_visLayout.setSelectedSteps(new ArrayList());
}
});
stepContextMenu.add(copyItem);
copyItem.setEnabled(!executing);
menuItemCount++;
}
MenuItem deleteItem = new MenuItem("Delete");
deleteItem.addActionListener(new ActionListener() {
@Override
public void actionPerformed(ActionEvent e) {
m_visLayout.addUndoPoint();
try {
m_visLayout.removeStep(step);
} catch (WekaException ex) {
m_visLayout.getMainPerspective().showErrorDialog(ex);
}
// step.getStepManager().clearAllConnections();
// LayoutPanel.this.remove(step);
String key =
step.getStepName() + "$"
+ step.getStepManager().getManagedStep().hashCode();
m_visLayout.getLogPanel().statusMessage(key + "|remove");
LayoutPanel.this.revalidate();
LayoutPanel.this.repaint();
m_visLayout.setEdited(true);
m_visLayout.getMainPerspective().notifyIsDirty();
m_visLayout
.getMainPerspective()
.getMainToolBar()
.enableWidget(
MainKFPerspectiveToolBar.Widgets.SELECT_ALL_BUTTON.toString(),
m_visLayout.getSelectedSteps().size() > 0);
}
});
deleteItem.setEnabled(!executing);
stepContextMenu.add(deleteItem);
menuItemCount++;
MenuItem nameItem = new MenuItem("Set name...");
nameItem.addActionListener(new ActionListener() {
@Override
public void actionPerformed(ActionEvent e) {
String oldName = step.getStepName();
String name =
JOptionPane.showInputDialog(m_visLayout.getMainPerspective(),
"Enter a name for this step", oldName);
if (name != null) {
m_visLayout.renameStep(oldName, name);
m_visLayout.setEdited(true);
revalidate();
repaint();
}
}
});
nameItem.setEnabled(!executing);
stepContextMenu.add(nameItem);
menuItemCount++;
MenuItem configItem = new MenuItem("Configure...");
configItem.addActionListener(new ActionListener() {
@Override
public void actionPerformed(ActionEvent e) {
popupStepEditorDialog(step);
m_visLayout.getMainPerspective().notifyIsDirty();
}
});
configItem.setEnabled(!executing);
stepContextMenu.add(configItem);
menuItemCount++;
// connections
List outgoingConnTypes =
step.getStepManager().getManagedStep().getOutgoingConnectionTypes();
if (outgoingConnTypes != null && outgoingConnTypes.size() > 0) {
MenuItem connections = new MenuItem("Connections:");
connections.setEnabled(false);
stepContextMenu.insert(connections, menuItemCount);
menuItemCount++;
for (final String connType : outgoingConnTypes) {
MenuItem conItem = new MenuItem(connType);
conItem.setEnabled(!executing);
conItem.addActionListener(new ActionListener() {
@Override
public void actionPerformed(ActionEvent e) {
initiateConnection(connType, step, x, y);
m_visLayout.getMainPerspective().notifyIsDirty();
}
});
stepContextMenu.add(conItem);
menuItemCount++;
}
}
// Any interactive views?
Map interactiveViews =
step.getStepManager().getManagedStep().getInteractiveViewers();
Map interactiveViewsImpls =
step.getStepManager().getManagedStep().getInteractiveViewersImpls();
if (interactiveViews != null && interactiveViews.size() > 0) {
MenuItem actions = new MenuItem("Actions:");
actions.setEnabled(false);
stepContextMenu.insert(actions, menuItemCount++);
for (Map.Entry e : interactiveViews.entrySet()) {
String command = e.getKey();
String viewerClassName = e.getValue();
addInteractiveViewMenuItem(step, e.getKey(), !executing, e.getValue(),
null, stepContextMenu);
menuItemCount++;
}
} else if (interactiveViewsImpls != null
&& interactiveViewsImpls.size() > 0) {
MenuItem actions = new MenuItem("Actions:");
actions.setEnabled(false);
stepContextMenu.insert(actions, menuItemCount++);
for (Map.Entry e : interactiveViewsImpls
.entrySet()) {
String command = e.getKey();
StepInteractiveViewer impl = e.getValue();
addInteractiveViewMenuItem(step, e.getKey(), !executing, null, impl,
stepContextMenu);
menuItemCount++;
}
}
// perspective stuff...
final List perspectives =
m_visLayout.getMainPerspective().getMainApplication()
.getPerspectiveManager().getVisiblePerspectives();
if (perspectives.size() > 0
&& step.getStepManager().getManagedStep() instanceof Loader) {
final weka.core.converters.Loader theLoader =
((Loader) step.getStepManager().getManagedStep()).getLoader();
boolean ok = true;
if (theLoader instanceof FileSourcedConverter) {
String fileName =
((FileSourcedConverter) theLoader).retrieveFile().getPath();
fileName = m_visLayout.environmentSubstitute(fileName);
File tempF = new File(fileName);
String fileNameFixedPathSep = fileName.replace(File.separatorChar, '/');
if (!tempF.isFile()
&& this.getClass().getClassLoader().getResource(fileNameFixedPathSep) == null) {
ok = false;
}
}
if (ok) {
stepContextMenu.addSeparator();
menuItemCount++;
if (perspectives.size() > 1) {
MenuItem sendToAllPerspectives =
new MenuItem("Send to all perspectives");
menuItemCount++;
sendToAllPerspectives.addActionListener(new ActionListener() {
@Override
public void actionPerformed(ActionEvent e) {
loadDataAndSendToPerspective(theLoader, perspectives, -1);
}
});
stepContextMenu.add(sendToAllPerspectives);
}
Menu sendToPerspective = new Menu("Send to perspective...");
stepContextMenu.add(sendToPerspective);
menuItemCount++;
for (int i = 0; i < perspectives.size(); i++) {
final int pIndex = i;
if (perspectives.get(i).acceptsInstances()
&& !perspectives.get(i).getPerspectiveID()
.equalsIgnoreCase(KFDefaults.MAIN_PERSPECTIVE_ID)) {
String pName = perspectives.get(i).getPerspectiveTitle();
final MenuItem pI = new MenuItem(pName);
pI.addActionListener(new ActionListener() {
@Override
public void actionPerformed(ActionEvent e) {
loadDataAndSendToPerspective(theLoader, perspectives, pIndex);
}
});
sendToPerspective.add(pI);
}
}
}
}
if (menuItemCount > 0) {
// make sure that popup location takes current scaling into account
double z = m_visLayout.getZoomSetting() / 100.0;
double px = x * z;
double py = y * z;
this.add(stepContextMenu);
stepContextMenu.show(this, (int) px, (int) py);
}
}
private synchronized void loadDataAndSendToPerspective(
final weka.core.converters.Loader loader,
final List perspectives, final int perspectiveIndex) {
if (m_perspectiveDataLoadThread == null) {
m_perspectiveDataLoadThread = new Thread() {
@Override
public void run() {
try {
if (loader instanceof EnvironmentHandler) {
((EnvironmentHandler) loader).setEnvironment(m_visLayout
.getEnvironment());
loader.reset();
}
m_visLayout.getLogPanel().statusMessage(
"@!@[KnowledgeFlow]|Sending data to perspective(s)...");
Instances data = loader.getDataSet();
if (data != null) {
m_visLayout.getMainPerspective().getMainApplication()
.showPerspectivesToolBar();
// need to disable all the perspective buttons on the toolbar
/*
* m_visLayout.getMainPerspective().getMainApplication()
* .getPerspectiveManager().disableAllPerspectiveTabs();
*/
if (perspectiveIndex < 0) {
// send to all
for (Perspective p : perspectives) {
if (p.acceptsInstances()) {
p.setInstances(data);
m_visLayout.getMainPerspective().getMainApplication()
.getPerspectiveManager()
.setEnablePerspectiveTab(p.getPerspectiveID(), true);
}
}
} else {
perspectives.get(perspectiveIndex).setInstances(data);
m_visLayout
.getMainPerspective()
.getMainApplication()
.getPerspectiveManager()
.setActivePerspective(
perspectives.get(perspectiveIndex).getPerspectiveID());
m_visLayout
.getMainPerspective()
.getMainApplication()
.getPerspectiveManager()
.setEnablePerspectiveTab(
perspectives.get(perspectiveIndex).getPerspectiveID(), true);
}
}
} catch (Exception ex) {
m_visLayout.getMainPerspective().showErrorDialog(ex);
} finally {
// re-enable the perspective buttons
/*
* m_visLayout.getMainPerspective().getMainApplication()
* .getPerspectiveManager().enableAllPerspectiveTabs();
*/
m_perspectiveDataLoadThread = null;
m_visLayout.getLogPanel().statusMessage("@!@[KnowledgeFlow]|OK");
}
}
};
m_perspectiveDataLoadThread.setPriority(Thread.MIN_PRIORITY);
m_perspectiveDataLoadThread.start();
}
}
/**
* Initiate the process of connecting two steps.
*
* @param connType the type of the connection to create
* @param source the source step of the connection
* @param x the x coordinate at which the connection starts
* @param y the y coordinate at which the connection starts
*/
protected void initiateConnection(String connType, StepVisual source, int x,
int y) {
// unselect any selected steps on the layaout
if (m_visLayout.getSelectedSteps().size() > 0) {
m_visLayout.setSelectedSteps(new ArrayList());
}
List connectableForConnType =
m_visLayout.findStepsThatCanAcceptConnection(connType);
for (StepManagerImpl sm : connectableForConnType) {
sm.getStepVisual().setDisplayConnectors(true);
}
if (connectableForConnType.size() > 0) {
source.setDisplayConnectors(true);
m_visLayout.setEditStep(source);
m_visLayout.setEditConnection(connType);
Point closest = source.getClosestConnectorPoint(new Point(x, y));
m_currentX = (int) closest.getX();
m_currentY = (int) closest.getY();
m_oldX = m_currentX;
m_oldY = m_currentY;
Graphics2D gx = (Graphics2D) this.getGraphics();
gx.setXORMode(java.awt.Color.white);
gx.drawLine(m_currentX, m_currentY, m_currentX, m_currentY);
gx.dispose();
m_visLayout.setFlowLayoutOperation(LayoutOperation.CONNECTING);
}
revalidate();
repaint();
m_visLayout.getMainPerspective().notifyIsDirty();
}
/**
* Add a menu item to a contextual menu for accessing any interactive viewers
* that a step might provide
*
* @param step the step in question
* @param entryText the text of the menu entry
* @param enabled true if the menu entry is to be enabled
* @param viewerClassName the fully qualified name of the viewer that is
* associated with the menu entry
* @param viewerImpl an instance of the viewer itself. If null, then the fully
* qualified viewer class name will be used to instantiate an
* instance
* @param stepContextMenu the menu to add the item to
*/
protected void addInteractiveViewMenuItem(final StepVisual step,
String entryText, boolean enabled, final String viewerClassName,
final StepInteractiveViewer viewerImpl, PopupMenu stepContextMenu) {
MenuItem viewItem = new MenuItem(entryText);
viewItem.setEnabled(enabled);
viewItem.addActionListener(new ActionListener() {
@Override
public void actionPerformed(ActionEvent e) {
popupStepInteractiveViewer(step, viewerClassName, viewerImpl);
}
});
stepContextMenu.add(viewItem);
}
/**
* Popup a dialog containing a particular interactive step viewer
*
* @param step the step that provides the viewer
* @param viewerClassName the fully qualified name of the viewer
* @param viewerImpl the actual instance of the viewer
*/
protected void popupStepInteractiveViewer(StepVisual step,
String viewerClassName, StepInteractiveViewer viewerImpl) {
try {
Object viewer =
viewerClassName != null ? WekaPackageClassLoaderManager
.objectForName(viewerClassName) : viewerImpl;
// viewerClassName != null ? Beans.instantiate(this.getClass()
// .getClassLoader(), viewerClassName) : viewerImpl;
if (!(viewer instanceof StepInteractiveViewer)) {
throw new WekaException("Interactive step viewer component "
+ viewerClassName + " must implement StepInteractiveViewer");
}
if (!(viewer instanceof JComponent)) {
throw new WekaException("Interactive step viewer component "
+ viewerClassName + " must be a JComponent");
}
String viewerName = ((StepInteractiveViewer) viewer).getViewerName();
((StepInteractiveViewer) viewer).setStep(step.getStepManager()
.getManagedStep());
((StepInteractiveViewer) viewer).setMainKFPerspective(m_visLayout
.getMainPerspective());
JFrame jf = new JFrame(viewerName);
((StepInteractiveViewer) viewer).setParentWindow(jf);
((StepInteractiveViewer) viewer).init();
jf.setLayout(new BorderLayout());
jf.add((JComponent) viewer, BorderLayout.CENTER);
jf.pack();
jf.setVisible(true);
((StepInteractiveViewer) viewer).nowVisible();
} catch (IOException e) {
e.printStackTrace();
} catch (Exception e) {
m_visLayout.getMainPerspective().showErrorDialog(e);
}
}
/**
* Popup an editor dialog for a given step
*
* @param step the step to popup the editor dialog for
*/
protected void popupStepEditorDialog(StepVisual step) {
String custEditor =
step.getStepManager().getManagedStep().getCustomEditorForStep();
StepEditorDialog toPopup = null;
if (custEditor != null && custEditor.length() > 0) {
try {
Object custPanel = WekaPackageClassLoaderManager.objectForName(custEditor);
// Beans.instantiate(getClass().getClassLoader(), custEditor);
if (!(custPanel instanceof StepEditorDialog)) {
throw new WekaException(
"Custom editor must be a subclass of StepEditorDialog");
}
toPopup = ((StepEditorDialog) custPanel);
} catch (Exception ex) {
m_visLayout.getMainPerspective().showErrorDialog(ex);
ex.printStackTrace();
}
} else {
// create an editor based on the GenericObjectEditor
toPopup = new GOEStepEditorDialog();
toPopup.setMainPerspective(m_visLayout.getMainPerspective());
}
final JDialog d =
new JDialog((java.awt.Frame) getTopLevelAncestor(),
ModalityType.DOCUMENT_MODAL);
d.setLayout(new BorderLayout());
d.getContentPane().add(toPopup, BorderLayout.CENTER);
final StepEditorDialog toPopupC = toPopup;
d.addWindowListener(new java.awt.event.WindowAdapter() {
@Override
public void windowClosing(java.awt.event.WindowEvent e) {
d.dispose();
}
});
toPopup.setParentWindow(d);
toPopup.setEnvironment(m_visLayout.getEnvironment());
toPopup.setMainPerspective(m_visLayout.getMainPerspective());
toPopup.setStepToEdit(step.getStepManager().getManagedStep());
toPopup.setClosingListener(new StepEditorDialog.ClosingListener() {
@Override
public void closing() {
if (toPopupC.isEdited()) {
m_visLayout.setEdited(true);
}
}
});
d.pack();
d.setLocationRelativeTo(m_visLayout.getMainPerspective());
d.setVisible(true);
}
private int snapToGrid(int val) {
int r = val % m_gridSpacing;
val /= m_gridSpacing;
if (r > (m_gridSpacing / 2)) {
val++;
}
val *= m_gridSpacing;
return val;
}
/**
* Snaps selected steps to the grid
*/
protected void snapSelectedToGrid() {
List selected = m_visLayout.getSelectedSteps();
for (StepVisual s : selected) {
int x = s.getX();
int y = s.getY();
s.setX(snapToGrid(x));
s.setY(snapToGrid(y));
}
revalidate();
repaint();
m_visLayout.setEdited(true);
m_visLayout.getMainPerspective().notifyIsDirty();
}
private void highlightSubFlow(int startX, int startY, int endX, int endY) {
java.awt.Rectangle r =
new java.awt.Rectangle((startX < endX) ? startX : endX,
(startY < endY) ? startY : endY, Math.abs(startX - endX),
Math.abs(startY - endY));
List selected = m_visLayout.findSteps(r);
m_visLayout.setSelectedSteps(selected);
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy