mirror of
https://github.com/genodelabs/genode.git
synced 2025-01-25 22:00:32 +00:00
35bfc34db5
This patch makes nitpicker's geometry utilities available for the use by other programs. Thereby, the 'Point', 'Area', and 'Rect' classes have become templates that take the coordinate type and distance type as arguments.
180 lines
4.3 KiB
C++
180 lines
4.3 KiB
C++
/*
|
|
* \brief Geometric primitives
|
|
* \author Norman Feske
|
|
* \date 2006-08-05
|
|
*/
|
|
|
|
/*
|
|
* Copyright (C) 2006-2013 Genode Labs GmbH
|
|
*
|
|
* This file is part of the Genode OS framework, which is distributed
|
|
* under the terms of the GNU General Public License version 2.
|
|
*/
|
|
|
|
#ifndef _INCLUDE__UTIL__GEOMETRY_H_
|
|
#define _INCLUDE__UTIL__GEOMETRY_H_
|
|
|
|
#include <util/misc_math.h>
|
|
#include <base/stdint.h>
|
|
|
|
namespace Genode {
|
|
template <typename CT = int> class Point;
|
|
template <typename DT = unsigned> class Area;
|
|
template <typename CT = int, typename DT = unsigned> class Rect;
|
|
}
|
|
|
|
|
|
/**
|
|
* \param CT coordinate type
|
|
*/
|
|
template <typename CT>
|
|
class Genode::Point
|
|
{
|
|
private:
|
|
|
|
CT _x, _y;
|
|
|
|
public:
|
|
|
|
Point(CT x, CT y): _x(x), _y(y) { }
|
|
Point(): _x(0), _y(0) { }
|
|
|
|
int x() const { return _x; }
|
|
int y() const { return _y; }
|
|
|
|
/**
|
|
* Operator for adding points
|
|
*/
|
|
Point operator + (Point const &p) const { return Point(_x + p.x(), _y + p.y()); }
|
|
|
|
/**
|
|
* Operator for subtracting points
|
|
*/
|
|
Point operator - (Point const &p) const { return Point(_x - p.x(), _y - p.y()); }
|
|
|
|
/**
|
|
* Operator for testing non-equality of two points
|
|
*/
|
|
bool operator != (Point const &p) const { return p.x() != _x || p.y() != _y; }
|
|
};
|
|
|
|
|
|
/**
|
|
* \param DT distance type
|
|
*/
|
|
template <typename DT>
|
|
class Genode::Area
|
|
{
|
|
private:
|
|
|
|
DT _w, _h;
|
|
|
|
public:
|
|
|
|
Area(DT w, DT h): _w(w), _h(h) { }
|
|
Area(): _w(0), _h(0) { }
|
|
|
|
DT w() const { return _w; }
|
|
DT h() const { return _h; }
|
|
|
|
bool valid() const { return _w > 0 && _h > 0; }
|
|
|
|
size_t count() const { return _w*_h; }
|
|
};
|
|
|
|
|
|
/**
|
|
* Rectangle
|
|
*
|
|
* A valid rectangle consists of two points wheras point 2 has higher or equal
|
|
* coordinates than point 1. All other cases are threated as invalid
|
|
* rectangles.
|
|
*
|
|
* \param CT coordinate type
|
|
* \param DT distance type
|
|
*/
|
|
template <typename CT, typename DT>
|
|
class Genode::Rect
|
|
{
|
|
private:
|
|
|
|
Point<CT> _p1, _p2;
|
|
|
|
public:
|
|
|
|
/**
|
|
* Constructors
|
|
*/
|
|
Rect(Point<CT> p1, Point<CT> p2): _p1(p1), _p2(p2) { }
|
|
|
|
Rect(Point<CT> p, Area<DT> a)
|
|
: _p1(p), _p2(p.x() + a.w() - 1, p.y() + a.h() - 1) { }
|
|
|
|
Rect() { }
|
|
|
|
/**
|
|
* Accessors
|
|
*/
|
|
CT x1() const { return _p1.x(); }
|
|
CT y1() const { return _p1.y(); }
|
|
CT x2() const { return _p2.x(); }
|
|
CT y2() const { return _p2.y(); }
|
|
DT w() const { return _p2.x() - _p1.x() + 1; }
|
|
DT h() const { return _p2.y() - _p1.y() + 1; }
|
|
Point<CT> p1() const { return _p1; }
|
|
Point<CT> p2() const { return _p2; }
|
|
Area<DT> area() const { return Area<DT>(w(), h()); }
|
|
|
|
/**
|
|
* Return true if rectangle area is greater than zero
|
|
*/
|
|
bool valid() const { return _p1.x() <= _p2.x() && _p1.y() <= _p2.y(); }
|
|
|
|
/**
|
|
* Return true if area fits in rectangle
|
|
*/
|
|
bool fits(Area<DT> area) const { return w() >= area.w() && h() >= area.h(); }
|
|
|
|
/**
|
|
* Create new rectangle by intersecting two rectangles
|
|
*/
|
|
static Rect intersect(Rect r1, Rect r2) {
|
|
return Rect(Point<CT>(max(r1.x1(), r2.x1()), max(r1.y1(), r2.y1())),
|
|
Point<CT>(min(r1.x2(), r2.x2()), min(r1.y2(), r2.y2()))); }
|
|
|
|
/**
|
|
* Compute compounding rectangle of two rectangles
|
|
*/
|
|
static Rect compound(Rect r1, Rect r2) {
|
|
return Rect(Point<CT>(min(r1.x1(), r2.x1()), min(r1.y1(), r2.y1())),
|
|
Point<CT>(max(r1.x2(), r2.x2()), max(r1.y2(), r2.y2()))); }
|
|
|
|
/**
|
|
* Cut out rectangle from rectangle
|
|
*
|
|
* \param r rectangle to cut out
|
|
*
|
|
* In the worst case (if we cut a hole into the rectangle) we get
|
|
* four valid resulting rectangles.
|
|
*/
|
|
void cut(Rect r, Rect *top, Rect *left, Rect *right, Rect *bottom) const
|
|
{
|
|
/* limit the cut-out area to the actual rectangle */
|
|
r = intersect(r, *this);
|
|
|
|
*top = Rect(Point<CT>(x1(), y1()), Point<CT>(x2(), r.y1() - 1));
|
|
*left = Rect(Point<CT>(x1(), r.y1()), Point<CT>(r.x1() - 1, r.y2()));
|
|
*right = Rect(Point<CT>(r.x2() + 1, r.y1()), Point<CT>(x2(), r.y2()));
|
|
*bottom = Rect(Point<CT>(x1(), r.y2() + 1), Point<CT>(x2(), y2()));
|
|
}
|
|
|
|
/**
|
|
* Return position of an area when centered within the rectangle
|
|
*/
|
|
Point<CT> center(Area<DT> area) const {
|
|
return Point<CT>(((CT)w() - (CT)area.w())/2,
|
|
((CT)h() - (CT)area.h())/2) + p1(); }
|
|
};
|
|
|
|
#endif /* _INCLUDE__UTIL__GEOMETRY_H_ */
|