view dwtx/draw2d/Polyline.d @ 103:2d6540440fe6

Replace static ctors with lazy init.
author Frank Benoit <benoit@tionex.de>
date Sun, 03 Aug 2008 17:01:51 +0200
parents 95307ad235d9
children
line wrap: on
line source

/*******************************************************************************
 * Copyright (c) 2000, 2007 IBM Corporation and others.
 * All rights reserved. This program and the accompanying materials
 * are made available under the terms of the Eclipse Public License v1.0
 * which accompanies this distribution, and is available at
 * http://www.eclipse.org/legal/epl-v10.html
 *
 * Contributors:
 *     IBM Corporation - initial API and implementation
 * Port to the D programming language:
 *     Frank Benoit <benoit@tionex.de>
 *******************************************************************************/
module dwtx.draw2d.Polyline;

import dwt.dwthelper.utils;
import dwtx.dwtxhelper.Collection;

import dwtx.draw2d.geometry.Point;
import dwtx.draw2d.geometry.PointList;
import dwtx.draw2d.geometry.Rectangle;
import dwtx.draw2d.Shape;
import dwtx.draw2d.Graphics;
import dwtx.draw2d.IFigure;
import dwtx.draw2d.Connection;

/**
 * Renders a {@link PointList} as a series of line segments.  A Polyline figure should be
 * positioned by manipulating its points, <EM>NOT</EM> by calling
 * {@link Figure#setBounds(Rectangle)}.
 * <P>
 * A polyline's bounds will be calculated automatically based on its PointList.  The
 * bounds will be the smallest Rectangle large enough to render the line properly.
 * Children should not be added to a Polyline and will not affect the bounds calculation.
 */
public class Polyline
    : Shape
{

PointList points;
private int tolerance = 2;

private static Rectangle LINEBOUNDS_;
private static Rectangle LINEBOUNDS(){
    if( LINEBOUNDS_ is null ){
        synchronized( Polyline.classinfo ){
            if( LINEBOUNDS_ is null ){
                LINEBOUNDS_ = Rectangle.SINGLETON;
                assert( LINEBOUNDS !is null );
            }
        }
    }
    return LINEBOUNDS_;
}

this(){
    points = new PointList();
    setFill(false);
    bounds  = null;
}

/**
 * Adds the passed point to the Polyline.
 *
 * @param pt the Point to be added to the Polyline
 * @since 2.0
 */
public void addPoint(Point pt) {
    points.addPoint(pt);
    bounds = null;
    repaint();
}

/**
 * @see dwtx.draw2d.IFigure#containsPoint(int, int)
 */
public bool containsPoint(int x, int y) {
    int tolerance = Math.max(lineWidth / 2, this.tolerance);
    LINEBOUNDS.setBounds(getBounds());
    LINEBOUNDS.expand(tolerance, tolerance);
    if (!LINEBOUNDS.contains(x, y))
        return false;
    int ints[] = points.toIntArray();
    for (int index = 0; index < ints.length - 3; index  += 2) {
        if (lineContainsPoint(ints[index], ints[index + 1],
            ints[index + 2], ints[index + 3], x, y, tolerance))
            return true;
    }
    List children = getChildren();
    for (int i = 0; i < children.size(); i++) {
        if ((cast(IFigure)children.get(i)).containsPoint(x, y))
            return true;
    }
    return false;
}

private bool lineContainsPoint(
    int x1, int y1,
    int x2, int y2,
    int px, int py,
    int tolerance) {
    LINEBOUNDS.setSize(0, 0);
    LINEBOUNDS.setLocation(x1, y1);
    LINEBOUNDS.union_(x2, y2);
    LINEBOUNDS.expand(tolerance, tolerance);
    if (!LINEBOUNDS.contains(px, py))
        return false;

    int v1x, v1y, v2x, v2y;
    int numerator, denominator;
    int result = 0;

    /**
     * calculates the length squared of the cross product of two vectors, v1 & v2.
     */
    if (x1 !is x2 && y1 !is y2) {
        v1x = x2 - x1;
        v1y = y2 - y1;
        v2x = px - x1;
        v2y = py - y1;

        numerator = v2x * v1y - v1x * v2y;

        denominator = v1x * v1x + v1y * v1y;

        result = cast(int)(cast(long)numerator * numerator / denominator);
    }

    // if it is the same point, and it passes the bounding box test,
    // the result is always true.
    return result <= tolerance * tolerance;

}

/**
 * Null implementation for a line.
 * @see dwtx.draw2d.Shape#fillShape(Graphics)
 */
protected void fillShape(Graphics g) { }

/**
 * @see dwtx.draw2d.IFigure#getBounds()
 */
public Rectangle getBounds() {
    if (bounds is null) {
        bounds = getPoints()
            .getBounds()
            .getExpanded(lineWidth / 2, lineWidth / 2);
    }
    return bounds;
}

/**
 * Returns the last point in the Polyline.
 * @since 2.0
 * @return the last point
 */
public Point getEnd() {
    return points.getLastPoint();
}

/**
 * Returns the points in this Polyline <B>by reference</B>. If the returned list is
 * modified, this Polyline must be informed by calling {@link #setPoints(PointList)}.
 * Failure to do so will result in layout and paint problems.
 *
 * @return this Polyline's points
 * @since 2.0
 */
public PointList getPoints() {
    return points;
}

/**
 * @return the first point in the Polyline
 * @since 2.0
 */
public Point getStart() {
    return points.getFirstPoint();
}

/**
 * Inserts a given point at a specified index in the Polyline.
 *
 * @param pt the point to be added
 * @param index the position in the Polyline where the point is to be added
 *
 * @since 2.0
 */
public void insertPoint(Point pt, int index) {
    bounds = null;
    points.insertPoint(pt, index);
    repaint();
}

/**
 * @return <code>false</code> because Polyline's aren't filled
 */
public bool isOpaque() {
    return false;
}

/**
 * @see Shape#outlineShape(Graphics)
 */
protected void outlineShape(Graphics g) {
    g.drawPolyline(points);
}

/**
 * @see Figure#primTranslate(int, int)
 */
public void primTranslate(int x, int y) { }

/**
 * Erases the Polyline and removes all of its {@link Point Points}.
 *
 * @since 2.0
 */
public void removeAllPoints() {
    erase();
    bounds = null;
    points.removeAllPoints();
}

/**
 * Removes a point from the Polyline.
 *
 * @param index the position of the point to be removed
 * @since 2.0
 */
public void removePoint(int index) {
    erase();
    bounds = null;
    points.removePoint(index);
}

/**
 * Sets the end point of the Polyline
 *
 * @param end the point that will become the last point in the Polyline
 * @since 2.0
 */
public void setEnd(Point end) {
    if (points.size() < 2)
        addPoint(end);
    else
        setPoint(end, points.size() - 1);
}

/**
 * Sets the points at both extremes of the Polyline
 *
 * @param start the point to become the first point in the Polyline
 * @param end the point to become the last point in the Polyline
 * @since 2.0
 */
public void setEndpoints(Point start, Point end) {
    setStart(start);
    setEnd(end);
}

/**
 * @see dwtx.draw2d.Shape#setLineWidth(int)
 */
public void setLineWidth(int w) {
    if (lineWidth is w)
        return;
    if (w < lineWidth) //The bounds will become smaller, so erase must occur first.
        erase();
    bounds = null;
    super.setLineWidth(w);
}

/**
 * Sets the point at <code>index</code> to the Point <code>pt</code>.  Calling this method
 * results in a recalculation of the polyline's bounding box.  If you're going to set
 * multiple Points, use {@link #setPoints(PointList)}.
 * @param pt the point
 * @param index the index
 */
public void setPoint(Point pt, int index) {
    erase();
    points.setPoint(pt, index);
    bounds = null;
    repaint();
}

/**
 * Sets the list of points to be used by this polyline connection. Removes any previously
 * existing points. The polyline will hold onto the given list by reference.
 *
 * @param points new set of points
 * @since 2.0
 */
public void setPoints(PointList points) {
    erase();
    this.points = points;
    bounds = null;
    firePropertyChange(Connection.PROPERTY_POINTS, null, points);
    repaint();
}

/**
 * Sets the start point of the Polyline
 *
 * @param start the point that will become the first point in the Polyline
 * @since 2.0
 */
public void setStart(Point start) {
    if (points.size() is 0)
        addPoint(start);
    else
        setPoint(start, 0);
}

/**
  * Sets the tolerance
  *
  * @param tolerance the new tolerance value of the Polyline
  */
public void setTolerance(int tolerance) {
       this.tolerance = tolerance;
}
}