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

impl.org.controlsfx.tools.rectangle.CoordinatePositions Maven / Gradle / Ivy

/**
 * Copyright (c) 2014, ControlsFX
 * All rights reserved.
 *
 * Redistribution and use in source and binary forms, with or without
 * modification, are permitted provided that the following conditions are met:
 *     * Redistributions of source code must retain the above copyright
 * notice, this list of conditions and the following disclaimer.
 *     * Redistributions in binary form must reproduce the above copyright
 * notice, this list of conditions and the following disclaimer in the
 * documentation and/or other materials provided with the distribution.
 *     * Neither the name of ControlsFX, any associated website, nor the
 * names of its contributors may be used to endorse or promote products
 * derived from this software without specific prior written permission.
 *
 * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS" AND
 * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED
 * WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE
 * DISCLAIMED. IN NO EVENT SHALL CONTROLSFX BE LIABLE FOR ANY
 * DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES
 * (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES;
 * LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND
 * ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
 * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS
 * SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
 */
package impl.org.controlsfx.tools.rectangle;

import java.util.EnumSet;

import javafx.geometry.Point2D;
import javafx.geometry.Rectangle2D;

/**
 * Computes coordinate positions relative to a rectangle.
 */
public class CoordinatePositions {

    /**
     * Returns all positions the specified point has regarding the specified rectangle and its edges (using the
     * specified tolerance).
     * 
     * @param rectangle
     *            the rectangle relative to which the point will be checked
     * @param point
     *            the checked point
     * @param edgeTolerance
     *            the tolerance in pixels used to determine whether the coordinates are on some edge
     * @return a set of those positions the coordinates have regarding the specified rectangle
     */
    public static EnumSet onRectangleAndEdges(
            Rectangle2D rectangle, Point2D point, double edgeTolerance) {

        EnumSet positions = EnumSet.noneOf(CoordinatePosition.class);
        positions.add(inRectangle(rectangle, point));
        positions.add(onEdges(rectangle, point, edgeTolerance));
        return positions;
    }

    /*
     * RECTANGLE
     */

    /**
     * Returns the position the specified coordinates have regarding the specified rectangle. Edges are not checked.
     * 
     * @param rectangle
     *            the rectangle relative to which the point will be checked
     * @param point
     *            the checked point
     * @return depending on the point either {@link CoordinatePosition#IN_RECTANGLE IN_RECTANGLE} or
     *         {@link CoordinatePosition#OUT_OF_RECTANGLE OUT_OF_RECTANGLE}
     */
    public static CoordinatePosition inRectangle(Rectangle2D rectangle, Point2D point) {
        if (rectangle.contains(point)) {
            return CoordinatePosition.IN_RECTANGLE;
        } else {
            return CoordinatePosition.OUT_OF_RECTANGLE;
        }
    }

    /*
     * EDGES
     */

    /**
     * Returns the position the specified coordinates have regarding the specified rectangle's edges using the specified
     * tolerance.
     * 
     * @param rectangle
     *            the rectangle relative to which the point will be checked
     * @param point
     *            the checked point
     * @param edgeTolerance
     *            the tolerance in pixels used to determine whether the coordinates are on some edge
     * @return the edge position the coordinates have regarding the specified rectangle; the value will be null if the
     *         point is not near any edge
     */
    public static CoordinatePosition onEdges(Rectangle2D rectangle, Point2D point,
            double edgeTolerance) {

        CoordinatePosition vertical = closeToVertical(rectangle, point, edgeTolerance);
        CoordinatePosition horizontal = closeToHorizontal(rectangle, point, edgeTolerance);

        return extractSingleCardinalDirection(vertical, horizontal);
    }

    /**
     * Returns the vertical bound the specified coordinates are closest to, if the distance is smaller than the edge
     * tolerance. Otherwise, null is returned.
     * 
     * @param rectangle
     *            the rectangle relative to which the point will be checked
     * @param point
     *            the checked point
     * @param edgeTolerance
     *            the tolerance in pixels used to determine whether the coordinates are on some edge
     * @return EAST_EDGE, WEST_EDGE or null
     */
    private static CoordinatePosition closeToVertical(Rectangle2D rectangle, Point2D point, double edgeTolerance) {

        double xDistanceToLeft = Math.abs(point.getX() - rectangle.getMinX());
        double xDistanceToRight = Math.abs(point.getX() - rectangle.getMaxX());
        boolean xCloseToLeft = xDistanceToLeft < edgeTolerance && xDistanceToLeft < xDistanceToRight;
        boolean xCloseToRight = xDistanceToRight < edgeTolerance && xDistanceToRight < xDistanceToLeft;

        if (!xCloseToLeft && !xCloseToRight) {
            return null;
        }

        boolean yCloseToVertical = rectangle.getMinY() - edgeTolerance < point.getY()
                && point.getY() < rectangle.getMaxY() + edgeTolerance;
        if (yCloseToVertical) {
            if (xCloseToLeft) {
                return CoordinatePosition.WEST_EDGE;
            }
            if (xCloseToRight) {
                return CoordinatePosition.EAST_EDGE;
            }
        }

        return null;
    }

    /**
     * Returns the horizontal bound the specified coordinates are closest to, if the distance is smaller than the edge
     * tolerance. Otherwise, null is returned.
     * 
     * @param rectangle
     *            the rectangle relative to which the point will be checked
     * @param point
     *            the checked point
     * @param edgeTolerance
     *            the tolerance in pixels used to determine whether the coordinates are on some edge
     * @return NORTH_EDGE, SOUTH_EDGE or null
     */
    private static CoordinatePosition closeToHorizontal(Rectangle2D rectangle, Point2D point, double edgeTolerance) {

        double yDistanceToUpper = Math.abs(point.getY() - rectangle.getMinY());
        double yDistanceToLower = Math.abs(point.getY() - rectangle.getMaxY());
        boolean yCloseToUpper = yDistanceToUpper < edgeTolerance && yDistanceToUpper < yDistanceToLower;
        boolean yCloseToLower = yDistanceToLower < edgeTolerance && yDistanceToLower < yDistanceToUpper;

        if (!yCloseToUpper && !yCloseToLower) {
            return null;
        }

        boolean xCloseToHorizontal = rectangle.getMinX() - edgeTolerance < point.getX()
                && point.getX() < rectangle.getMaxX() + edgeTolerance;
        if (xCloseToHorizontal) {
            if (yCloseToUpper) {
                return CoordinatePosition.NORTH_EDGE;
            }
            if (yCloseToLower) {
                return CoordinatePosition.SOUTH_EDGE;
            }
        }

        return null;
    }

    /**
     * Extracts a single cardinal direction from the two specified positions. The conditions stated below are not
     * checked!
     * 
     * @param vertical
     *            a vertical edge (EAST or WEST) or null
     * @param horizontal
     *            a horizontal edge (NORTH OR SOUTH) or null
     * @return the single coordinate position which matches the specified positions 
* (e.g. NORTH for (null, NORTH) and SOUTHWEST for (WEST, SOUTH)) */ private static CoordinatePosition extractSingleCardinalDirection(CoordinatePosition vertical, CoordinatePosition horizontal) { if (vertical == null) { return horizontal; } if (horizontal == null) { return vertical; } // north if (horizontal == CoordinatePosition.NORTH_EDGE && vertical == CoordinatePosition.EAST_EDGE) { return CoordinatePosition.NORTHEAST_EDGE; } if (horizontal == CoordinatePosition.NORTH_EDGE && vertical == CoordinatePosition.WEST_EDGE) { return CoordinatePosition.NORTHWEST_EDGE; } // south if (horizontal == CoordinatePosition.SOUTH_EDGE && vertical == CoordinatePosition.EAST_EDGE) { return CoordinatePosition.SOUTHEAST_EDGE; } if (horizontal == CoordinatePosition.SOUTH_EDGE && vertical == CoordinatePosition.WEST_EDGE) { return CoordinatePosition.SOUTHWEST_EDGE; } throw new IllegalArgumentException(); } }




© 2015 - 2025 Weber Informatics LLC | Privacy Policy