Beispiel #1
0
 private Widget createLabel(String text, int x, int y, Color color) {
   LabelWidget label = new LabelWidget(this, text);
   label.setOpaque(true);
   label.setBackground(color);
   label.setBorder(BorderFactory.createLineBorder(5));
   label.setPreferredLocation(new Point(x, y));
   label.getActions().addAction(ActionFactory.createMoveAction());
   mainLayer.addChild(label);
   return label;
 }
/** @author Jon Rose */
public class VisualScene extends GraphScene {

  public LayerWidget forgroundLayer = new LayerWidget(this);
  public LayerWidget mainLayer = new LayerWidget(this);
  public LayerWidget connectionLayer = new LayerWidget(this);
  public LayerWidget interactionLayer = new LayerWidget(this);
  public LayerWidget backgroundLayer = new LayerWidget(this);
  public static Port currentPort;
  private ArrayList<Widget> inputs = new ArrayList();
  private ArrayList<Widget> outputs = new ArrayList();
  private WidgetAction multiMove = ActionFactory.createMoveAction(null, new MultiMoveProvider());
  private WidgetAction moveAction =
      ActionFactory.createAlignWithMoveAction(mainLayer, interactionLayer, null, false);
  private WidgetAction resizeAction =
      ActionFactory.createAlignWithResizeAction(mainLayer, interactionLayer, null, false);
  private WidgetAction hoverAction = ActionFactory.createHoverAction(new MyHoverProvider(this));
  private UIMouseAction dragAction = new UIMouseAction();
  private SceneMouseAction sceneAction = new SceneMouseAction();
  public static final String EDIT = "editMode";
  public static final String CONNECTION_EDIT_TOOL = "connectionEditTool";
  public static final String USE = "alignMode";
  public static final String EDGE_CONTROL_MODE = "edgeControlMode";
  public static final String MULTI_MOVE = "multiMove";
  private NodeCreator dataNodeCreator;
  private DataScene dataScene = new DataScene(this);
  private boolean over = false;
  private double mouseX;
  private double mouseY;
  private double lastX;
  private double lastY;
  private double xVelocity;
  private double yVelocity;
  private Point lastPointCreatedAt = new Point(400, 400);
  private boolean mouseDragged = false;
  protected ArrayList<Connection> connections = new ArrayList<Connection>();
  private HashMap customRenderer = new HashMap();
  public Trie trie = new Trie();

  public VisualScene() {
    addChild(backgroundLayer);
    addChild(mainLayer);
    addChild(connectionLayer);
    addChild(interactionLayer);
    addChild(forgroundLayer);

    // set the initial tool
    this.setActiveTool(EDIT);

    getActions().addAction(ActionFactory.createZoomAction());
    getActions().addAction(ActionFactory.createPanAction());
    getActions()
        .addAction(
            hoverAction); // this hover action is needed for port highlights to turn them off(the
    // scene takes over the mosue)
    getActions().addAction(new SceneMouseAction());
    getActions().addAction(ActionFactory.createRectangularSelectAction(this, backgroundLayer));

    getPriorActions().addAction(new KeyEventAction()); // add the scene key event listener
  }

  public Trie getTrie() {
    return trie;
  }

  public boolean isMouseDragged() {
    return mouseDragged;
  }

  public void setMouseDragged(boolean mouseDragged) {
    this.mouseDragged = mouseDragged;
  }

  public double getLastX() {
    return lastX;
  }

  public void setLastX(double lastX) {
    this.lastX = lastX;
  }

  public double getLastY() {
    return lastY;
  }

  public void setLastY(double lastY) {
    this.lastY = lastY;
  }

  public double getMouseX() {
    return mouseX;
  }

  public void setMouseX(double mouseX) {
    this.mouseX = mouseX;
  }

  public double getMouseY() {
    return mouseY;
  }

  public void setMouseY(double mouseY) {
    this.mouseY = mouseY;
  }

  public double getxVelocity() {
    return xVelocity;
  }

  public void setxVelocity(double xVelocity) {
    this.xVelocity = xVelocity;
  }

  public double getyVelocity() {
    return yVelocity;
  }

  public void setyVelocity(double yVelocity) {
    this.yVelocity = yVelocity;
  }

  public Point getLastPointCreatedAt() {
    return lastPointCreatedAt;
  }

  public void setLastPointCreatedAt(Point lastPointCreatedAt) {
    this.lastPointCreatedAt = lastPointCreatedAt;
  }

  public void setDataScene(DataScene ds) {
    dataScene = ds;
  }

  public DataScene getDataScene() {
    return dataScene;
  }

  /**
   * Checks whether an object is registered as a node in the graph model.
   *
   * @param object the object; must not be a Widget
   * @return true, if the object is registered as a node
   */
  public LayerWidget getBackgroundLayer() {
    return backgroundLayer;
  }

  public LayerWidget getConnectionLayer() {
    return connectionLayer;
  }

  public LayerWidget getForgroundLayer() {
    return forgroundLayer;
  }

  public LayerWidget getInteractionLayer() {
    return interactionLayer;
  }

  public LayerWidget getMainLayer() {
    return mainLayer;
  }

  public boolean isInput(Object object) {
    return inputs.contains((Widget) object);
  }

  public boolean isOutput(Widget targetWidget) {
    return outputs.contains(targetWidget);
  }

  public void addConnection(ConnectionWidget conn) {
    this.addEdge(conn);
  }

  public void addInputConnection(PortInteractor input) {
    inputs.add(input);
  }

  public void addInputConnection(CustomPortInteractor input) {
    inputs.add(input);
  }

  public void addOutputConnection(PortInteractor output) {
    outputs.add(output);
  }

  public void addOutputConnection(CustomPortInteractor output) {
    outputs.add(output);
  }

  private Widget findOutput(Object sourceNode) {
    int id = outputs.indexOf(sourceNode);
    return outputs.get(id);
  }

  private Widget findInput(Object sourceNode) {
    int id = inputs.indexOf(sourceNode);
    return inputs.get(id);
  }

  public WidgetAction getHoverAction() {
    return hoverAction;
  }

  // not working
  public void deleteSelected() {
    Set objects = getSelectedObjects();
  }

  public void createNode(VisualNode node) {
    mainLayer.addChild(node);
  }

  public VisualScene getVisualScene() {
    return this;
  }

  @Override
  public void paintChildren() {
    super.paintChildren();
  }

  @Override
  protected Widget attachNodeWidget(Object n) {
    Widget widget = (Widget) n;
    widget.setToolTipText("flow node");
    widget.createActions(USE).addAction(0, dragAction);
    widget.createActions(EDIT).addAction(0, createSelectAction());
    widget.createActions(EDIT).addAction(1, multiMove);
    widget.createActions(EDIT).addAction(2, new CustomResizeAction());
    widget.createActions(EDIT).addAction(3, createObjectHoverAction());

    mainLayer.addChild(widget);

    return widget;
  }

  @Override
  protected Widget attachEdgeWidget(Object e) {
    // create the visual connection
    CustomConnectionWidget connection = (CustomConnectionWidget) e;
    connection.setPaintControlPoints(true);
    connection.setRouter(CustomRouterFactory.createFreeRouter());
    connection.setControlPointShape(FlowControlPointShape.CIRCLE_FILLED_SMALL);

    // Normal Edit Mode actions
    connection
        .createActions(EDGE_CONTROL_MODE)
        .addAction(0, ActionFactory.createAddRemoveControlPointAction());
    connection
        .createActions(EDGE_CONTROL_MODE)
        .addAction(1, ActionFactory.createFreeMoveControlPointAction());
    connection
        .createActions(EDGE_CONTROL_MODE)
        .addAction(2, ActionFactory.createAddRemoveControlPointAction());
    connection
        .createActions(EDGE_CONTROL_MODE)
        .addAction(3, ActionFactory.createFreeMoveControlPointAction());
    connection.createActions(EDGE_CONTROL_MODE).addAction(4, createObjectHoverAction());

    connectionLayer.addChild(connection);

    // create the data connection
    //        Port srcPort = (Port)
    // connection.getSourceAnchor().getRelatedWidget().getParentWidget();
    //        VisualNode src = srcPort.getParentNode();
    //
    //        Port tgtPort = (Port)
    // connection.getTargetAnchor().getRelatedWidget().getParentWidget();
    //        VisualNode tgt = tgtPort.getParentNode();

    //        dataScene.connect(src.getDataNode(), srcPort.getID(), tgt.getDataNode(),
    // tgtPort.getID());

    return connection;
  }

  @Override
  protected void attachEdgeSourceAnchor(Object edge, Object sourceNode, Object n1) {
    Widget w = sourceNode != null ? findOutput(sourceNode) : null;
    ((ConnectionWidget) findWidget(edge)).setSourceAnchor(AnchorFactory.createRectangularAnchor(w));
  }

  @Override
  protected void attachEdgeTargetAnchor(Object edge, Object n, Object targetNode) {
    Widget w = targetNode != null ? findInput(targetNode) : null;
    ((ConnectionWidget) findWidget(edge)).setTargetAnchor(AnchorFactory.createRectangularAnchor(w));
  }

  private void connect(VisualNode source, int sourcePort, VisualNode target, int targetPort) {
    CustomConnectionWidget connection = new CustomConnectionWidget(this);
    Widget src = source.getOutputPort(sourcePort);
    Widget tgt = target.getInputPort(targetPort);
    connection.setSourceAnchor(AnchorFactory.createCircularAnchor(src, 2));
    connection.setTargetAnchor(AnchorFactory.createCircularAnchor(tgt, 2));
    addEdge(connection);

    connections.add(new Connection(source, sourcePort, target, targetPort, connection));
  }

  /*
   * for use with CustomBorderPort testing
   */
  private void connect(
      BorderPortTestNode source, int sourcePort, BorderPortTestNode target, int targetPort) {
    CustomConnectionWidget connection = new CustomConnectionWidget(this);
    Widget src = source.getOutputPort(sourcePort);
    Widget tgt = target.getInputPort(targetPort);
    connection.setSourceAnchor(AnchorFactory.createCircularAnchor(src, 2));
    connection.setTargetAnchor(AnchorFactory.createCircularAnchor(tgt, 2));
    addEdge(connection);

    connections.add(new Connection(source, sourcePort, target, targetPort, connection));
  }

  public void connect(CustomPortInteractor source, CustomPortInteractor target) {
    CustomConnectionWidget connection = new CustomConnectionWidget(this);
    connection.setSourceAnchor(AnchorFactory.createCircularAnchor(source, 1));
    connection.setTargetAnchor(AnchorFactory.createCircularAnchor(target, 1));

    // save one for each so that they connections can be accessed in both directions
    connections.add(
        new Connection(
            source.getNode(),
            source.getPortNumber(),
            target.getNode(),
            target.getPortNumber(),
            connection));
    addEdge(connection);
  }

  public void connect(PortInteractor source, PortInteractor target) {
    CustomConnectionWidget connection = new CustomConnectionWidget(this);
    connection.setSourceAnchor(AnchorFactory.createCircularAnchor(source, 1));
    connection.setTargetAnchor(AnchorFactory.createCircularAnchor(target, 1));

    // save one for each so that they connections can be accessed in both directions
    connections.add(
        new Connection(
            source.getNode(),
            source.getPortNumber(),
            target.getNode(),
            target.getPortNumber(),
            connection));
    addEdge(connection);
  }

  public void removeConnections(VisualNode node) {
    Iterator<Connection> i = connections.iterator();
    ArrayList<Connection> toRemove = new ArrayList<Connection>();
    while (i.hasNext()) {
      Connection connection = (Connection) i.next();
      if (connection.getSource() == node || connection.getTarget() == node) {
        toRemove.add(connection);
        this.removeEdge(connection.getConnectionWidget());
      }
    }

    Iterator<Connection> r = toRemove.iterator();
    while (r.hasNext()) {
      Connection removeConnection = r.next();
      connections.remove(removeConnection);
    }
  }

  public void setDataScene(NodeCreator d) {
    dataNodeCreator = d;
  }

  public void setEditMode() {
    this.setActiveTool(EDIT);
  }

  public void setAlignMode() {
    if (this.getActiveTool().equals(EDIT)) {
      setActiveTool(USE);
      removeHoverActions();
    } else {
      setActiveTool(EDIT);
      getActions().addAction(hoverAction);
    }
    printActiveToolMode();
  }

  private void printActiveToolMode() {
    //       System.out.println(this.getActiveTool());
    //       System.out.println(this.getActions());
  }

  private void routeSelectedEdges() {
    Object[] selectedObjects = this.getSelectedObjects().toArray();
    CustomConnectionWidget conn = new CustomConnectionWidget(this.getScene());
    for (int i = 0; i < selectedObjects.length; i++) {
      if (selectedObjects[i].getClass() == conn.getClass()) {
        CustomConnectionWidget connection = (CustomConnectionWidget) selectedObjects[i];
        connection.setRouter(CustomRouterFactory.createOrthogonalSearchRouter(mainLayer));
      }
    }
  }

  private void unrouteSelectedEdges() {
    Object[] selectedObjects = this.getSelectedObjects().toArray();
    CustomConnectionWidget conn = new CustomConnectionWidget(this.getScene());
    for (int i = 0; i < selectedObjects.length; i++) {
      if (selectedObjects[i].getClass() == conn.getClass()) {
        CustomConnectionWidget connection = (CustomConnectionWidget) selectedObjects[i];
        connection.setRouter(CustomRouterFactory.createDirectRouter());
      }
    }
  }

  public void mouseClicked(Widget widget, WidgetMouseEvent wme) {
    this.setLastX(this.getMouseX());
    this.setLastY(this.getMouseY());
    this.setMouseX(wme.getPoint().getX());
    this.setMouseY(wme.getPoint().getY());
  }

  public void mousePressed(Widget widget, WidgetMouseEvent wme) {
    this.setMouseDragged(true);
    this.setLastX(this.getMouseX());
    this.setLastY(this.getMouseY());
    this.setMouseX(wme.getPoint().getX());
    this.setMouseY(wme.getPoint().getY());
  }

  public void mouseReleased(Widget widget, WidgetMouseEvent wme) {
    this.setMouseDragged(false);
  }

  public void mouseEntered(Widget widget, WidgetMouseEvent wme) {
    this.setLastX(this.getMouseX());
    this.setLastY(this.getMouseY());
    this.setMouseX(wme.getPoint().getX());
    this.setMouseY(wme.getPoint().getY());
  }

  public void mouseExited(Widget widget, WidgetMouseEvent wme) {}

  public void mouseDragged(Widget widget, WidgetMouseEvent wme) {
    this.setLastX(this.getMouseX());
    this.setLastY(this.getMouseY());
    this.setMouseX(wme.getPoint().getX());
    this.setMouseY(wme.getPoint().getY());
    this.setxVelocity(this.getLastX() - this.getMouseX());
    this.setyVelocity(this.getMouseY() - this.getLastY());
  }

  void mouseMoved(Widget widget, WidgetMouseEvent wme) {
    this.setLastX(this.getMouseX());
    this.setLastY(this.getMouseY());
    this.setMouseX(wme.getPoint().getX());
    this.setMouseY(wme.getPoint().getY());
    this.setxVelocity(this.getLastX() - this.getMouseX());
    this.setyVelocity(this.getMouseY() - this.getLastY());
  }

  private void removeHoverActions() {
    this.getActions().removeAction(this.getHoverAction());
    Object[] nodes = this.getNodes().toArray();
    for (int i = 0; i < this.getNodes().size() - 1; i++) {
      VisualNode node = (VisualNode) nodes[i];
      node.removeHoverActions();
      //           System.out.println("hover scene off");
    }
  }

  public ArrayList<Connection> getConnections() {
    return connections;
  }

  private void addEdgeSelectionAction() {
    Collection<ConnectionWidget> edges = this.getEdges();
    for (Iterator i = edges.iterator(); i.hasNext(); ) {
      ConnectionWidget connection = (ConnectionWidget) i.next();
      connection.getActions().addAction(createSelectAction());
    }
  }

  private void removeEdgeSelectionAction() {
    Collection<ConnectionWidget> edges = this.getEdges();
    for (Iterator i = edges.iterator(); i.hasNext(); ) {
      ConnectionWidget connection = (ConnectionWidget) i.next();
      connection.getActions().removeAction(createSelectAction());
    }
  }

  private class MyConnectProvider implements ConnectProvider {

    @Override
    public boolean isSourceWidget(Widget source) {
      return source instanceof IconNodeWidget && source != null ? true : false;
    }

    @Override
    public ConnectorState isTargetWidget(Widget src, Widget trg) {
      return src != trg && trg instanceof IconNodeWidget
          ? ConnectorState.ACCEPT
          : ConnectorState.REJECT;
    }

    @Override
    public boolean hasCustomTargetWidgetResolver(Scene arg0) {
      return false;
    }

    @Override
    public Widget resolveTargetWidget(Scene arg0, Point arg1) {
      return null;
    }

    @Override
    public void createConnection(Widget source, Widget target) {
      ConnectionWidget conn = new ConnectionWidget(VisualScene.this);
      conn.setTargetAnchorShape(AnchorShape.TRIANGLE_FILLED);
      conn.setTargetAnchor(AnchorFactory.createRectangularAnchor(target));
      conn.setSourceAnchor(AnchorFactory.createRectangularAnchor(source));
      conn.getActions().addAction(createObjectHoverAction());
      conn.getActions().addAction(createSelectAction());
      connectionLayer.addChild(conn);
    }
  }

  // multi object moving
  private class MultiMoveProvider implements MoveProvider {

    private HashMap<Widget, Point> originals = new HashMap<Widget, Point>();
    private Point original;

    @Override
    public void movementStarted(Widget widget) {
      Object object = findObject(widget);
      if (isNode(object)) {
        for (Object o : getSelectedObjects()) {
          if (isNode(o)) {
            Widget w = findWidget(o);
            if (w != null) {
              originals.put(w, w.getPreferredLocation());
            }
          }
        }
      } else {
        originals.put(widget, widget.getPreferredLocation());
      }
    }

    @Override
    public void movementFinished(Widget widget) {
      originals.clear();
      original = null;
    }

    @Override
    public Point getOriginalLocation(Widget widget) {
      original = widget.getPreferredLocation();
      return original;
    }

    @Override
    public void setNewLocation(Widget widget, Point location) {
      int dx = location.x - original.x;
      int dy = location.y - original.y;
      for (Map.Entry<Widget, Point> entry : originals.entrySet()) {
        Point point = entry.getValue();
        entry.getKey().setPreferredLocation(new Point(point.x + dx, point.y + dy));
      }
    }
  }

  private final class KeyEventAction extends WidgetAction.Adapter {

    @Override
    public State keyPressed(Widget widget, WidgetKeyEvent event) {
      VisualScene scene = (VisualScene) widget;
      if (event.getKeyCode() == KeyEvent.VK_N) { // type 'n' to create a new object
        BorderPortTestNode node = new BorderPortTestNode(scene.getVisualScene());
        node.setPreferredLocation(new Point((int) scene.getMouseX(), (int) scene.getMouseY()));
        scene.setLastPointCreatedAt(new Point((int) scene.getMouseX(), (int) scene.getMouseY()));
        scene.addNode(node);
      }

      if (event.getKeyCode() == KeyEvent.VK_F) { // type 'f' to create a new object
        //                VisualNode node = new FloatDisplay(scene.getScene(),
        // scene.getDataScene());
        //                node.setPreferredLocation(new Point((int) scene.getMouseX(), (int)
        // scene.getMouseY()));
        //                scene.setLastPointCreatedAt(new Point((int) scene.getMouseX(), (int)
        // scene.getMouseY()));
        //                scene.addNode(node);
      }

      if (event.isAltDown()
          && event.getKeyCode() == KeyEvent.VK_A) { // alt + 'a' for align mode toggle
        scene.setAlignMode();
        scene.removeHoverActions();
      }

      if (event.isAltDown()
          && event.getKeyCode() == KeyEvent.VK_E) { // alt + 'e' for edit/use mode toggle
        scene.setEditMode();
      }

      if (event.isAltDown()
          && event.getKeyCode() == KeyEvent.VK_R) { // alt + 'r' route selected selectedObjects
        scene.routeSelectedEdges();
      }

      if (event.isAltDown() && event.isShiftDown() && event.getKeyCode() == KeyEvent.VK_R) {
        scene.unrouteSelectedEdges();
      }

      //// use arrows to move selected objects

      if (getSelectedObjects().size() > 0) {
        Set array = getSelectedObjects();
        if (event.getKeyCode() == KeyEvent.VK_UP) {
          for (Iterator i = array.iterator(); i.hasNext(); ) {
            try {
              BorderPortTestNode node = (BorderPortTestNode) i.next();
              node.setPreferredLocation(
                  new Point(node.getPreferredLocation().x, node.getPreferredLocation().y - 2));
            } catch (java.lang.ClassCastException e) {
            }
          }
        }

        if (event.getKeyCode() == KeyEvent.VK_DOWN) {
          for (Iterator i = array.iterator(); i.hasNext(); ) {
            try {
              BorderPortTestNode node = (BorderPortTestNode) i.next();
              node.setPreferredLocation(
                  new Point(node.getPreferredLocation().x, node.getPreferredLocation().y + 2));
            } catch (java.lang.ClassCastException e) {
            }
          }
        }

        if (event.getKeyCode() == KeyEvent.VK_LEFT) {
          for (Iterator i = array.iterator(); i.hasNext(); ) {
            try {
              BorderPortTestNode node = (BorderPortTestNode) i.next();
              node.setPreferredLocation(
                  new Point(node.getPreferredLocation().x - 2, node.getPreferredLocation().y));
            } catch (java.lang.ClassCastException e) {
            }
          }
        }

        if (event.getKeyCode() == KeyEvent.VK_RIGHT) {
          for (Iterator i = array.iterator(); i.hasNext(); ) {
            try {
              BorderPortTestNode node = (BorderPortTestNode) i.next();
              node.setPreferredLocation(
                  new Point(node.getPreferredLocation().x + 2, node.getPreferredLocation().y));
            } catch (java.lang.ClassCastException e) {
            }
          }
        }
      }

      if (event.getKeyCode() == KeyEvent.VK_E) {
        widget.getScene().setActiveTool(EDGE_CONTROL_MODE);
        scene.addEdgeSelectionAction();

      } else if (event.getKeyCode() == KeyEvent.VK_DELETE) {
        // need to add object and edge deletion
      }

      return State.REJECTED;
    }

    @Override
    public State keyReleased(Widget widget, WidgetKeyEvent event) {
      if (widget.getScene().getActiveTool().equals(EDGE_CONTROL_MODE)) {
        widget.getScene().setActiveTool(EDIT);
      }
      //      System.out.println(widget.getScene().getActiveTool());
      return State.REJECTED;
    }
  }

  private static class MyHoverProvider implements TwoStateHoverProvider {

    private Scene scene;

    public MyHoverProvider(Scene scene) {
      this.scene = scene;
    }

    @Override
    public void unsetHovering(Widget widget) {
      if (widget != null) {
        widget.setBackground(scene.getLookFeel().getBackground(ObjectState.createNormal()));
        widget.setForeground(new Color(0, 0, 0, 0));
        CustomPortInteractor port = (CustomPortInteractor) widget;
        port.setOver(false);
      }
    }

    @Override
    public void setHovering(Widget widget) {
      if (widget != null) {
        ObjectState state = ObjectState.createNormal().deriveSelected(true);
        widget.setBackground(scene.getLookFeel().getBackground(state));
        widget.setForeground(Color.yellow);
        CustomPortInteractor port = (CustomPortInteractor) widget;
        port.setOver(true);
      }
    }
  }

  /** @param ugens - an array of IUGenInfo objects sent from overtone/supercollider */
  public void addIUGenInfo(Collection<IUGenInfo> info) {
    if (info != null) {
      for (Iterator i = info.iterator(); i.hasNext(); ) {
        try {
          IUGenInfo ugenInfo = (IUGenInfo) i.next();
          trie.insert(ugenInfo.getName(), ugenInfo);
        } catch (NonUniqueKeyException ex) {
          Exceptions.printStackTrace(ex);
        } catch (BadKeyException ex) {
          Exceptions.printStackTrace(ex);
        }
      }
    }
  }
}
Beispiel #3
0
/** @author Santhosh Kumar T */
public class RuleScene extends GraphScene<Node, Edge> {
  private final WidgetAction moveAction = ActionFactory.createMoveAction();
  private final WidgetAction nodePopupAction =
      ActionFactory.createPopupMenuAction(new NodePopupProvider(this));
  private final WidgetAction edgePopupAction =
      ActionFactory.createPopupMenuAction(new EdgePopupProvider(this));

  private final WidgetAction hoverAction;
  private final WidgetAction editAction;

  private LayerWidget interactionLayer = new LayerWidget(this);
  private WidgetAction connectAction =
      ActionFactory.createConnectAction(interactionLayer, new NodeConnectProvider());

  private TwoStateHoverProvider hoverProvider;

  public File file;

  public RuleScene(TwoStateHoverProvider hoverProvider, EditProvider edgeEditProvider) {
    addChild(nodes);
    addChild(connections);
    addChild(interactionLayer);

    this.hoverProvider = hoverProvider;
    hoverAction = ActionFactory.createHoverAction(new Highlighter(hoverProvider));
    editAction = ActionFactory.createEditAction(edgeEditProvider);

    getActions().addAction(hoverAction);
    getActions().addAction(ActionFactory.createPopupMenuAction(new ScenePopupProvider(this)));
    getActions().addAction(ActionFactory.createWheelPanAction());
  }

  /*-------------------------------------------------[ Rule ]---------------------------------------------------*/

  private Syntax syntax;
  private Rule rule;
  public final Observable ruleObservable =
      new Observable() {
        @Override
        public void notifyObservers(Object arg) {
          setChanged();
          super.notifyObservers(arg);
        }
      };

  public void setRule(Syntax syntax, Rule rule) {
    this.syntax = syntax;

    if (this.rule != null) {
      for (Node node : getNodes().toArray(new Node[getNodes().size()])) removeNodeWithEdges(node);
    }
    this.rule = rule;
    if (this.rule != null) populate(rule.node);

    layout();
    hoverProvider.unsetHovering(this);
    ruleObservable.notifyObservers(rule);
  }

  public Syntax getSyntax() {
    return syntax;
  }

  public Rule getRule() {
    return rule;
  }

  private void populate(Node node) {
    if (findWidget(node) == null) {
      addNode(node);

      for (Edge edge : node.outgoing) {
        populate(edge.target);
        addEdge(edge);
        setEdgeSource(edge, edge.source);
        setEdgeTarget(edge, edge.target);
      }
    }
  }

  /*-------------------------------------------------[ Layout ]---------------------------------------------------*/

  private final SceneLayout layout =
      LayoutFactory.createSceneGraphLayout(this, new RuleLayout(false));

  public void layout() {
    validate();
    layout.invokeLayoutImmediately();
  }

  public void layout(Object model) {
    findWidget(model).revalidate();
    layout();
  }

  public void refresh() {
    Syntax syntax = this.syntax;
    Rule rule = this.rule;
    setRule(null, null);
    setRule(syntax, rule);
  }

  /*-------------------------------------------------[ Nodes ]---------------------------------------------------*/

  private final LayerWidget nodes = new LayerWidget(this);

  @Override
  protected Widget attachNodeWidget(Node node) {
    NodeWidget widget = new NodeWidget(this);
    widget.setFont(Util.FIXED_WIDTH_FONT);
    widget.highLight(false);

    nodes.addChild(widget);
    //        widget.getActions().addAction(moveAction);
    widget.getActions().addAction(hoverAction);
    widget.getActions().addAction(nodePopupAction);
    widget.getActions().addAction(connectAction);

    return widget;
  }

  /*-------------------------------------------------[ Edges ]---------------------------------------------------*/

  private final LayerWidget connections = new LayerWidget(this);
  private final Router router = new RuleRouter();

  private LabelWidget createEdgeLabel(Edge edge) {
    LabelWidget label = new LabelWidget(this, edge.toString());
    label.setFont(Util.FIXED_WIDTH_FONT);
    label.setBorder(BorderFactory.createEmptyBorder(0, 0, 2, 0));

    label.getActions().addAction(hoverAction);
    label.getActions().addAction(editAction);
    label.getActions().addAction(moveAction);
    label.getActions().addAction(edgePopupAction);
    return label;
  }

  @Override
  protected Widget attachEdgeWidget(Edge edge) {
    EdgeWidget connection = new EdgeWidget(this);
    connection.setStroke(Util.STROKE_2);
    connection.setTargetAnchorShape(AnchorShape.TRIANGLE_FILLED);

    LabelWidget label = createEdgeLabel(edge);
    connection.setConstraint(label, LayoutFactory.ConnectionWidgetLayoutAlignment.TOP_CENTER, 0.5f);

    connection.addChild(label);
    connections.addChild(connection);

    connection.setRouter(router);
    connection.highLight(false);

    connection.getActions().addAction(hoverAction);
    connection.getActions().addAction(editAction);
    connection.getActions().addAction(edgePopupAction);
    return connection;
  }

  @Override
  protected void attachEdgeSourceAnchor(Edge edge, Node oldSource, Node newSource) {
    Widget w = newSource != null ? findWidget(newSource) : null;
    ((ConnectionWidget) findWidget(edge)).setSourceAnchor(AnchorFactory.createRectangularAnchor(w));
  }

  @Override
  protected void attachEdgeTargetAnchor(Edge edge, Node oldTarget, Node newTarget) {
    Widget w = newTarget != null ? findWidget(newTarget) : null;
    ((ConnectionWidget) findWidget(edge)).setTargetAnchor(AnchorFactory.createRectangularAnchor(w));
  }

  /*-------------------------------------------------[ Executing ]---------------------------------------------------*/

  private Widget executionWidget;

  private void executing(Widget widget) {
    if (executionWidget != widget) {
      if (executionWidget != null) {
        ((NBLRWidget) executionWidget).executing(false);
        executionWidget = null;
      }
      if (widget != null) {
        executionWidget = widget;
        ((NBLRWidget) executionWidget).executing(true);
        final Rectangle bounds = Util.bounds(widget);
        SwingUtilities.invokeLater(
            new Runnable() {
              @Override
              public void run() {
                getView().scrollRectToVisible(bounds);
              }
            });
      }
      validate();
    }
  }

  public void executing(Node node) {
    executing(findWidget(node));
  }

  public void executing(Edge edge) {
    executing(findWidget(edge));
  }

  /*-------------------------------------------------[ Antialiasing ]---------------------------------------------------*/

  public void paintChildren() {
    Graphics2D g = getGraphics();
    Object anti = g.getRenderingHint(RenderingHints.KEY_ANTIALIASING);
    Object textAnti = g.getRenderingHint(RenderingHints.KEY_TEXT_ANTIALIASING);
    g.setRenderingHint(RenderingHints.KEY_ANTIALIASING, RenderingHints.VALUE_ANTIALIAS_ON);
    g.setRenderingHint(
        RenderingHints.KEY_TEXT_ANTIALIASING, RenderingHints.VALUE_TEXT_ANTIALIAS_ON);
    super.paintChildren();
    g.setRenderingHint(RenderingHints.KEY_ANTIALIASING, anti);
    g.setRenderingHint(RenderingHints.KEY_TEXT_ANTIALIASING, textAnti);
  }
}