scummvm/graphics/rect2d.cpp

191 lines
5.1 KiB
C++

/* Residual - A 3D game interpreter
*
* Residual is the legal property of its developers, whose names
* are too numerous to list here. Please refer to the COPYRIGHT
* file distributed with this source distribution.
*
* This library is free software; you can redistribute it and/or
* modify it under the terms of the GNU Lesser General Public
* License as published by the Free Software Foundation; either
* version 2.1 of the License, or (at your option) any later version.
*
* This library 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
* Lesser General Public License for more details.
*
* You should have received a copy of the GNU Lesser General Public
* License along with this library; if not, write to the Free Software
* Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA
*
* $URL$
* $Id$
*/
#include "common/scummsys.h"
#include "graphics/rect2d.h"
namespace Graphics {
Rect2d::Rect2d() {
}
Rect2d::Rect2d(const Vector2d &topLeft, const Vector2d &bottomRight) {
float left = (topLeft.getX() <= bottomRight.getX() ? topLeft.getX() : bottomRight.getX());
float right = (topLeft.getX() <= bottomRight.getX() ? bottomRight.getX() : topLeft.getX());
float top = (topLeft.getY() <= bottomRight.getY() ? topLeft.getY() : bottomRight.getY());
float bottom = (topLeft.getY() <= bottomRight.getY() ? bottomRight.getY() : topLeft.getY());
_topLeft = Vector2d(left, top);
_topRight = Vector2d(right, top);
_bottomLeft = Vector2d(left, bottom);
_bottomRight = Vector2d(right, bottom);
}
Rect2d::Rect2d(const Vector2d &topLeft, const Vector2d &topRight,
const Vector2d &bottomLeft, const Vector2d &bottomRight) :
_topLeft(topLeft), _topRight(topRight),
_bottomLeft(bottomLeft), _bottomRight(bottomRight) {
}
void Rect2d::rotateAround(const Vector2d &point, float angle) {
_topLeft.rotateAround(point, angle);
_topRight.rotateAround(point, angle);
_bottomLeft.rotateAround(point, angle);
_bottomRight.rotateAround(point, angle);
}
void Rect2d::rotateAroundCenter(float angle) {
Vector2d center = getCenter();
rotateAround(center, angle);
}
bool Rect2d::intersectsRect(const Rect2d &rect) const {
}
bool Rect2d::intersectsCircle(const Vector2d &center, float radius) const {
Vector2d c = getCenter();
float w = getWidth();
float h = getHeight();
float angle = (_topRight - _topLeft).getAngle();
if (angle < 0.1 && angle > -0.1) {
Vector2d circleDistance(fabs(center.getX() - c.getX()), fabs(center.getY() - c.getY()));
if (circleDistance.getX() > (w / 2.f + radius)) {
return false;
}
if (circleDistance.getY() > (h / 2.f + radius)) {
return false;
}
if (circleDistance.getX() <= (w / 2.f)) {
return true;
}
if (circleDistance.getY() <= (h / 2.f)) {
return true;
}
float cornerDistance_sq = pow(circleDistance.getX() - w / 2.f, 2.f) +
pow(circleDistance.getY() - h / 2.f, 2.f);
return (cornerDistance_sq <= radius * radius);
} else { //The rectangle was rotated
Rect2d r(_topLeft, _topRight, _bottomLeft, _bottomRight);
r.rotateAroundCenter(-angle);
Vector2d circle(center);
circle.rotateAround(r.getCenter(), -angle);
return r.intersectsCircle(circle, radius);
}
}
bool Rect2d::containsPoint(const Vector2d &point) const {
return (point.getX() >= _topLeft.getX() && point.getX() <= _bottomRight.getX() &&
point.getY() >= _topLeft.getY() && point.getY() <= _bottomRight.getY());
}
Vector2d Rect2d::getCenter() const {
Vector2d sum = _topLeft + _topRight + _bottomLeft + _bottomRight;
sum /= 4;
return sum;
}
Vector2d Rect2d::getTopLeft() const {
return _topLeft;
}
Vector2d Rect2d::getTopRight() const {
return _topRight;
}
Vector2d Rect2d::getBottomLeft() const {
return _bottomLeft;
}
Vector2d Rect2d::getBottomRight() const {
return _bottomRight;
}
float Rect2d::getWidth() const {
float x = _topRight.getX() - _topLeft.getX();
float y = _topRight.getY() - _topLeft.getY();
return sqrt(x * x + y * y);
}
float Rect2d::getHeight() const {
float x = _bottomLeft.getX() - _topLeft.getX();
float y = _bottomLeft.getY() - _topLeft.getY();
return sqrt(x * x + y * y);
}
Vector2d Rect2d::getIntersection(const Vector2d &start, const Vector2d &dir, Segment2d *edge) const {
float w = getWidth();
float h = getHeight();
float d = sqrt(w * w + h * h);
Segment2d line(start, start + dir.getNormalized() * 2*d);
Vector2d intersection;
Segment2d l(_topLeft, _topRight);
if (line.intersectsSegment(l, &intersection)) {
if (edge) {
*edge = l;
}
return intersection;
}
l = Segment2d(_topRight, _bottomRight);
if (line.intersectsSegment(l, &intersection)) {
if (edge) {
*edge = l;
}
return intersection;
}
l = Segment2d(_bottomRight, _bottomLeft);
if (line.intersectsSegment(l, &intersection)) {
if (edge) {
*edge = l;
}
return intersection;
}
l = Segment2d(_bottomLeft, _topLeft);
if (line.intersectsSegment(l, &intersection)) {
if (edge) {
*edge = l;
}
return intersection;
}
return intersection;
}
}