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

org.netbeans.modules.visual.action.ReconnectAction Maven / Gradle / Ivy

There is a newer version: RELEASE230
Show newest version
/*
 * Licensed to the Apache Software Foundation (ASF) under one
 * or more contributor license agreements.  See the NOTICE file
 * distributed with this work for additional information
 * regarding copyright ownership.  The ASF licenses this file
 * to you under the Apache License, Version 2.0 (the
 * "License"); you may not use this file except in compliance
 * with the License.  You may obtain a copy of the License at
 *
 *   http://www.apache.org/licenses/LICENSE-2.0
 *
 * Unless required by applicable law or agreed to in writing,
 * software distributed under the License is distributed on an
 * "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY
 * KIND, either express or implied.  See the License for the
 * specific language governing permissions and limitations
 * under the License.
 */
package org.netbeans.modules.visual.action;

import org.netbeans.api.visual.anchor.Anchor;
import org.netbeans.api.visual.widget.ConnectionWidget;
import org.netbeans.api.visual.widget.Scene;
import org.netbeans.api.visual.widget.Widget;
import org.netbeans.api.visual.action.*;

import java.awt.*;
import java.awt.event.MouseEvent;
import java.awt.event.KeyEvent;
import java.util.List;

/**
 * Override at least isReplacementWidget and setConnectionAnchor methods. reconnectingFinished is always called before setConnectionAnchor.
 *
 * @author David Kaspar
 */
public final class ReconnectAction extends WidgetAction.LockedAdapter {

    private static final int MIN_DIFFERENCE = 5;

    private ReconnectDecorator decorator;
    private ReconnectProvider provider;

    private ConnectionWidget connectionWidget = null;
    private boolean reconnectingSource = false;
    private Point floatPoint = null;
    private Widget replacementWidget = null;
    private Anchor originalAnchor = null;

    public ReconnectAction (ReconnectDecorator decorator, ReconnectProvider provider) {
        this.decorator = decorator;
        this.provider = provider;
    }

    protected boolean isLocked () {
        return connectionWidget != null;
    }

    public State mousePressed (Widget widget, WidgetMouseEvent event) {
        if (isLocked ())
            return State.createLocked (widget, this);
        if (event.getButton () == MouseEvent.BUTTON1  &&  event.getClickCount () == 1) {
            if (widget instanceof ConnectionWidget) {
                ConnectionWidget conn = (ConnectionWidget) widget;
                int index = conn.getControlPointHitAt (event.getPoint ());
                List controlPoints = conn.getControlPoints ();
                if (index == 0  &&  provider.isSourceReconnectable (conn)) {
                    reconnectingSource = true;
                } else if (controlPoints != null  &&  index == controlPoints.size () - 1  && provider.isTargetReconnectable (conn)) {
                    reconnectingSource = false;
                } else {
                    return State.REJECTED;
                }

                floatPoint = new Point (event.getPoint ());
                replacementWidget = null;
                connectionWidget = conn;
                provider.reconnectingStarted (conn, reconnectingSource);
                if (reconnectingSource)
                    originalAnchor = connectionWidget.getSourceAnchor ();
                else
                    originalAnchor = connectionWidget.getTargetAnchor ();
                return State.createLocked (widget, this);
            }
        }
        return State.REJECTED;
    }

    public State mouseReleased (Widget widget, WidgetMouseEvent event) {
        Point point = event.getPoint ();
        boolean state = move (widget, point);
        if (state) {
            cancel (point);
        }
        return state ? State.CONSUMED : State.REJECTED;
    }

    private void cancel (Point point) {
        if (reconnectingSource)
            connectionWidget.setSourceAnchor (originalAnchor);
        else
            connectionWidget.setTargetAnchor (originalAnchor);
        provider.reconnectingFinished (connectionWidget, reconnectingSource);
        if (point != null)
            if (Math.abs (floatPoint.x - point.x) >= ReconnectAction.MIN_DIFFERENCE  ||  Math.abs (floatPoint.y - point.y) >= ReconnectAction.MIN_DIFFERENCE)
                provider.reconnect (connectionWidget, replacementWidget, reconnectingSource);
        replacementWidget = null;
        floatPoint = null;
        connectionWidget = null;
    }

    public State mouseDragged (Widget widget, WidgetMouseEvent event) {
        return move (widget, event.getPoint ()) ? State.createLocked (widget, this) : State.REJECTED;
    }

    private boolean move (Widget widget, Point point) {
        if (connectionWidget != widget)
            return false;

        Point replacementSceneLocation = widget.convertLocalToScene (point);
        replacementWidget = resolveReplacementWidgetCore (connectionWidget.getScene (), replacementSceneLocation);
        Anchor replacementAnchor = null;
        if (replacementWidget != null)
            replacementAnchor = decorator.createReplacementWidgetAnchor (replacementWidget);
        if (replacementAnchor == null)
            replacementAnchor = decorator.createFloatAnchor (replacementSceneLocation);

        if (reconnectingSource)
            connectionWidget.setSourceAnchor (replacementAnchor);
        else
            connectionWidget.setTargetAnchor (replacementAnchor);

        return true;
    }

    protected Widget resolveReplacementWidgetCore (Scene scene, Point sceneLocation) {
        if (provider != null)
            if (provider.hasCustomReplacementWidgetResolver (scene))
                return provider.resolveReplacementWidget (scene, sceneLocation);
        Point sceneOrigin = scene.getLocation ();
        sceneLocation = new Point (sceneLocation.x + sceneOrigin.x, sceneLocation.y + sceneOrigin.y);
        Widget[] result = new Widget[]{null};
        resolveReplacementWidgetCoreDive (result, scene, sceneLocation);
        return result[0];
    }

    private boolean resolveReplacementWidgetCoreDive (Widget[] result, Widget widget, Point parentLocation) {
        if (widget == connectionWidget)
            return false;

        Point widgetLocation = widget.getLocation ();
        Point location = new Point (parentLocation.x - widgetLocation.x, parentLocation.y - widgetLocation.y);

        if (! widget.getBounds ().contains (location))
            return false;

        java.util.List children = widget.getChildren ();
        for (int i = children.size () - 1; i >= 0; i --) {
            if (resolveReplacementWidgetCoreDive (result, children.get (i), location))
                return true;
        }

        if (! widget.isHitAt (location))
            return false;

        ConnectorState state = provider.isReplacementWidget (connectionWidget, widget, reconnectingSource);
        if (state == ConnectorState.REJECT)
            return false;
        if (state == ConnectorState.ACCEPT)
            result[0] = widget;
        return true;
    }


    public State keyPressed (Widget widget, WidgetKeyEvent event) {
        if (isLocked ()  &&  event.getKeyCode () == KeyEvent.VK_ESCAPE) {
            cancel (null);
            return State.CONSUMED;
        }
        return super.keyPressed (widget, event);
    }

}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy