/*
* This file is part of aion-emu <aion-emu.com>.
*
* aion-emu is free software: you can redistribute it and/or modify
* it under the terms of the GNU General Public License as published by
* the Free Software Foundation, either version 3 of the License, or
* (at your option) any later version.
*
* aion-emu 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 General Public License for more details.
*
* You should have received a copy of the GNU General Public License
* along with aion-emu. If not, see <http://www.gnu.org/licenses/>.
*/
package com.aionemu.gameserver.model.geometry;
import java.awt.Point;
import java.awt.Polygon;
import java.util.Collection;
import com.aionemu.gameserver.utils.MathUtil;
/**
* Area of free form
*
* @author SoulKeeper
*/
public class PolyArea extends AbstractArea
{
/**
* Collection of x points
*/
private final int[] xPoints;
/**
* Collection of y points
*/
private final int[] yPoints;
/**
* Polygon used to calculate isInside()
*/
private final Polygon poly;
/**
* Creates new area from given points
*
* @param points
* list of points
* @param zMin
* minimal z
* @param zMax
* maximal z
*/
public PolyArea(Collection<Point> points, int zMin, int zMax)
{
this(points.toArray(new Point[points.size()]), zMin, zMax);
}
/**
* Creates new area from given points
*
* @param points
* list of points
* @param zMin
* minimal z
* @param zMax
* maximal z
*/
public PolyArea(Point[] points, int zMin, int zMax)
{
super(zMin, zMax);
if(points.length < 3)
{
throw new IllegalArgumentException("Not enough points, needed at least 3 but got " + points.length);
}
this.xPoints = new int[points.length];
this.yPoints = new int[points.length];
Polygon polygon = new Polygon();
for(int i = 0, n = points.length; i < n; i++)
{
Point p = points[i];
polygon.addPoint(p.x, p.y);
xPoints[i] = p.x;
yPoints[i] = p.y;
}
this.poly = polygon;
}
/**
* {@inheritDoc}
*/
@Override
public boolean isInside2D(int x, int y)
{
return poly.contains(x, y);
}
/**
* {@inheritDoc}
*/
@Override
public double getDistance2D(int x, int y)
{
if(isInside2D(x, y))
{
return 0;
}
else
{
Point cp = getClosestPoint(x, y);
return MathUtil.getDistance(cp.x, cp.y, x, y);
}
}
/**
* {@inheritDoc}
*/
@Override
public double getDistance3D(int x, int y, int z)
{
if(isInside3D(x, y, z))
{
return 0;
}
else if(isInsideZ(z))
{
return getDistance2D(x, y);
}
else
{
Point3D cp = getClosestPoint(x, y, z);
return MathUtil.getDistance(cp.getX(), cp.getY(), cp.getZ(), x, y, z);
}
}
/**
* {@inheritDoc}
*/
@Override
public Point getClosestPoint(int x, int y)
{
Point closestPoint = null;
double closestDistance = 0;
for(int i = 0; i < xPoints.length; i++)
{
int nextIndex = i + 1;
if(nextIndex == xPoints.length)
{
nextIndex = 0;
}
int p1x = xPoints[i];
int p1y = yPoints[i];
int p2x = xPoints[nextIndex];
int p2y = yPoints[nextIndex];
Point point = MathUtil.getClosestPointOnSegment(p1x, p1y, p2x, p2y, x, y);
if(closestPoint == null)
{
closestPoint = point;
closestDistance = MathUtil.getDistance(closestPoint.x, closestPoint.y, x, y);
}
else
{
double newDistance = MathUtil.getDistance(point.x, point.y, x, y);
if(newDistance < closestDistance)
{
closestPoint = point;
closestDistance = newDistance;
}
}
}
return closestPoint;
}
}