/* * RapidMiner * * Copyright (C) 2001-2011 by Rapid-I and the contributors * * Complete list of developers available at our web site: * * http://rapid-i.com * * This program is free software: you can redistribute it and/or modify * it under the terms of the GNU Affero 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 Affero General Public License for more details. * * You should have received a copy of the GNU Affero General Public License * along with this program. If not, see http://www.gnu.org/licenses/. */ package com.rapidminer.gui.operatormenu; import java.util.HashMap; import java.util.List; import java.util.Map; import com.rapidminer.gui.RapidMinerGUI; import com.rapidminer.gui.tools.SwingTools; import com.rapidminer.operator.ExecutionUnit; import com.rapidminer.operator.Operator; import com.rapidminer.operator.OperatorChain; import com.rapidminer.operator.OperatorDescription; import com.rapidminer.operator.ports.InputPort; import com.rapidminer.operator.ports.OutputPort; import com.rapidminer.tools.OperatorService; /** * An operator menu which can be used to replace the currently selected operator * by one of the same type. Simple operators can be by other simple operators or * operator chains, operator chains can only be replaced by other chains. This * operator menu is available in the context menu of an operator in tree view. * * @author Ingo Mierswa, Simon Fischer, Tobias Malbrecht */ public class ReplaceOperatorMenu extends OperatorMenu { private static final long serialVersionUID = -663404687013352042L; protected ReplaceOperatorMenu(boolean onlyChains) { super("replace_operator", onlyChains); } @Override public void performAction(OperatorDescription description) { try { Operator operator = OperatorService.createOperator(description); replace(operator); } catch (Exception e) { SwingTools.showSimpleErrorMessage("cannot_instantiate", e, description.getName()); } } /** The currently selected operator will be replaced by the given operator. */ private void replace(Operator operator) { List<Operator> selection = RapidMinerGUI.getMainFrame().getSelectedOperators(); if (selection.isEmpty()) { return; } Operator selectedOperator = selection.get(0); ExecutionUnit parent = selectedOperator.getExecutionUnit(); if (parent == null) { return; } // remember source and sink connections so we can reconnect them later. Map<String,InputPort> inputPortMap = new HashMap<String,InputPort>(); Map<String,OutputPort> outputPortMap = new HashMap<String,OutputPort>(); for (OutputPort source : selectedOperator.getOutputPorts().getAllPorts()) { if (source.isConnected()) { inputPortMap.put(source.getName(), source.getDestination()); } } for (InputPort sink : selectedOperator.getInputPorts().getAllPorts()) { if (sink.isConnected()) { outputPortMap.put(sink.getName(), sink.getSource()); } } selectedOperator.getOutputPorts().disconnectAll(); selectedOperator.getInputPorts().disconnectAll(); int failedReconnects = 0; // copy children if possible if ((selectedOperator instanceof OperatorChain) && (operator instanceof OperatorChain)) { OperatorChain oldChain = (OperatorChain) selectedOperator; OperatorChain newChain = (OperatorChain) operator; int numCommonSubprocesses = Math.min(oldChain.getNumberOfSubprocesses(), newChain.getNumberOfSubprocesses()); for (int i = 0; i < numCommonSubprocesses; i++) { ExecutionUnit oldSubprocess = oldChain.getSubprocess(i); ExecutionUnit newSubprocess = newChain.getSubprocess(i); failedReconnects += newSubprocess.stealOperatorsFrom(oldSubprocess); } } int oldPos = parent.getOperators().indexOf(selectedOperator); selectedOperator.remove(); parent.addOperator(operator, oldPos); // Rewire sources and sinks for (Map.Entry<String,InputPort> entry : inputPortMap.entrySet()) { OutputPort mySource = operator.getOutputPorts().getPortByName(entry.getKey()); if (mySource != null) { mySource.connectTo(entry.getValue()); } else { failedReconnects++; } } operator.getOutputPorts().unlockPortExtenders(); for (Map.Entry<String,OutputPort> entry : outputPortMap.entrySet()) { InputPort mySink = operator.getInputPorts().getPortByName(entry.getKey()); if (mySink != null) { entry.getValue().connectTo(mySink); } else { failedReconnects++; } } operator.getInputPorts().unlockPortExtenders(); RapidMinerGUI.getMainFrame().selectOperator(operator.getParent()); if (failedReconnects > 0) { SwingTools.showVerySimpleErrorMessage("op_replaced_failed_connections_restored", failedReconnects); } } }