finalcut/src/frect.cpp

402 lines
10 KiB
C++
Raw Normal View History

2017-11-04 07:03:53 +01:00
/***********************************************************************
* frect.cpp - Rectangle with position and size *
* *
* This file is part of the FINAL CUT widget toolkit *
2017-11-04 07:03:53 +01:00
* *
* Copyright 2014-2020 Markus Gans *
2017-11-04 07:03:53 +01:00
* *
* FINAL CUT 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 3 of *
2017-11-04 07:03:53 +01:00
* the License, or (at your option) any later version. *
* *
* FINAL CUT is distributed in the hope that it will be useful, but *
* WITHOUT ANY WARRANTY; without even the implied warranty of *
2017-11-04 07:03:53 +01:00
* 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 program. If not, see *
* <http://www.gnu.org/licenses/>. *
***********************************************************************/
2015-05-23 13:35:12 +02:00
2017-09-11 03:06:02 +02:00
#include <algorithm>
2020-08-23 00:32:41 +02:00
#include <utility>
2017-09-11 03:06:02 +02:00
#include "final/fpoint.h"
#include "final/frect.h"
#include "final/fsize.h"
2015-05-23 13:35:12 +02:00
namespace finalcut
{
2015-05-23 13:35:12 +02:00
//----------------------------------------------------------------------
// class FRect
//----------------------------------------------------------------------
// constructor and destructor
2019-01-16 16:00:15 +01:00
//----------------------------------------------------------------------
FRect::FRect (const FPoint& p, const FSize& s)
: X1{p.getX()}
, Y1{p.getY()}
, X2{p.getX() + int(s.getWidth()) - 1}
, Y2{p.getY() + int(s.getHeight()) - 1}
2019-01-16 16:00:15 +01:00
{ }
2015-05-23 13:35:12 +02:00
//----------------------------------------------------------------------
FRect::FRect (const FPoint& p1, const FPoint& p2)
: X1{p1.getX()}
, Y1{p1.getY()}
, X2{p2.getX()}
, Y2{p2.getY()}
2015-09-20 05:44:50 +02:00
{ }
2015-05-23 13:35:12 +02:00
//----------------------------------------------------------------------
FRect::~FRect() // destructor
2015-09-22 04:18:20 +02:00
{ }
2015-05-23 13:35:12 +02:00
2017-09-11 03:06:02 +02:00
2015-05-23 13:35:12 +02:00
// public methods of FRect
2020-04-14 23:46:42 +02:00
//----------------------------------------------------------------------
FRect& FRect::operator = (const FRect& r)
{
X1 = r.X1;
Y1 = r.Y1;
X2 = r.X2;
Y2 = r.Y2;
return *this;
}
//----------------------------------------------------------------------
FRect& FRect::operator = (FRect&& r) noexcept
{
2020-04-19 20:38:52 +02:00
X1 = std::move(r.X1);
Y1 = std::move(r.Y1);
X2 = std::move(r.X2);
Y2 = std::move(r.Y2);
2020-04-14 23:46:42 +02:00
return *this;
}
2015-05-23 13:35:12 +02:00
//----------------------------------------------------------------------
2019-01-16 16:00:15 +01:00
bool FRect::isEmpty() const
2015-05-23 13:35:12 +02:00
{
return X2 == X1 - 1 && Y2 == Y1 - 1;
2015-05-23 13:35:12 +02:00
}
//----------------------------------------------------------------------
FPoint FRect::getPos() const
{
return { X1, Y1 };
}
//----------------------------------------------------------------------
FPoint FRect::getUpperLeftPos() const
{
return { X1, Y1 };
}
//----------------------------------------------------------------------
FPoint FRect::getUpperRightPos() const
{
return { X2, Y1 };
}
//----------------------------------------------------------------------
FPoint FRect::getLowerLeftPos() const
{
return { X1, Y2 };
}
//----------------------------------------------------------------------
FPoint FRect::getLowerRightPos() const
{
return { X2, Y2 };
}
//----------------------------------------------------------------------
FSize FRect::getSize() const
{
return { getWidth(), getHeight() };
}
2015-05-23 13:35:12 +02:00
//----------------------------------------------------------------------
void FRect::setX1 (int n)
{
X1 = n;
2015-05-23 13:35:12 +02:00
}
//----------------------------------------------------------------------
void FRect::setY1 (int n)
{
Y1 = n;
2015-05-23 13:35:12 +02:00
}
//----------------------------------------------------------------------
void FRect::setX2 (int n)
{
X2 = n;
2015-05-23 13:35:12 +02:00
}
//----------------------------------------------------------------------
void FRect::setY2 (int n)
{
Y2 = n;
2015-05-23 13:35:12 +02:00
}
//----------------------------------------------------------------------
void FRect::setX (int n)
{
const int dX = X2 - X1;
X1 = n;
X2 = X1 + dX;
2015-05-23 13:35:12 +02:00
}
//----------------------------------------------------------------------
void FRect::setY (int n)
{
const int dY = Y2 - Y1;
Y1 = n;
Y2 = Y1 + dY;
2015-05-23 13:35:12 +02:00
}
//----------------------------------------------------------------------
void FRect::setPos (int x, int y)
{
const int dX = X2 - X1;
const int dY = Y2 - Y1;
X1 = x;
Y1 = y;
X2 = X1 + dX;
Y2 = Y1 + dY;
}
//----------------------------------------------------------------------
void FRect::setPos (const FPoint& p)
{
const int dX = X2 - X1;
const int dY = Y2 - Y1;
X1 = p.getX();
Y1 = p.getY();
X2 = X1 + dX;
Y2 = Y1 + dY;
}
2015-05-23 13:35:12 +02:00
//----------------------------------------------------------------------
void FRect::setWidth (std::size_t w)
2015-05-23 13:35:12 +02:00
{
X2 = X1 + int(w) - 1;
2015-05-23 13:35:12 +02:00
}
//----------------------------------------------------------------------
void FRect::setHeight (std::size_t h)
2015-05-23 13:35:12 +02:00
{
Y2 = Y1 + int(h) - 1;
2015-05-23 13:35:12 +02:00
}
//----------------------------------------------------------------------
2019-08-11 18:15:57 +02:00
void FRect::setSize (std::size_t width, std::size_t height)
{
2019-08-11 18:15:57 +02:00
X2 = X1 + int(width) - 1;
Y2 = Y1 + int(height) - 1;
}
2019-01-16 16:00:15 +01:00
//----------------------------------------------------------------------
void FRect::setSize (const FSize& s)
{
X2 = X1 + int(s.getWidth()) - 1;
Y2 = Y1 + int(s.getHeight()) - 1;
}
2015-05-23 13:35:12 +02:00
//----------------------------------------------------------------------
void FRect::setRect (const FRect& r)
{
X1 = r.X1;
Y1 = r.Y1;
X2 = r.X2;
Y2 = r.Y2;
}
//----------------------------------------------------------------------
void FRect::setRect (const FPoint& p, const FSize& s)
{
X1 = p.getX();
Y1 = p.getY();
X2 = p.getX() + int(s.getWidth()) - 1;
Y2 = p.getY() + int(s.getHeight()) - 1;
2015-05-23 13:35:12 +02:00
}
//----------------------------------------------------------------------
void FRect::setRect (int x, int y, std::size_t width, std::size_t height)
2015-05-23 13:35:12 +02:00
{
X1 = x;
Y1 = y;
X2 = x + int(width) - 1;
Y2 = y + int(height) - 1;
2015-05-23 13:35:12 +02:00
}
//----------------------------------------------------------------------
void FRect::setCoordinates (const FPoint& p1, const FPoint& p2)
{
setCoordinates (p1.getX(), p1.getY(), p2.getX(), p2.getY());
}
//----------------------------------------------------------------------
void FRect::setCoordinates (int x1, int y1, int x2, int y2)
{
X1 = x1;
Y1 = y1;
X2 = x2;
Y2 = y2;
}
2015-05-23 13:35:12 +02:00
//----------------------------------------------------------------------
void FRect::move (int dx, int dy)
{
X1 += dx;
Y1 += dy;
X2 += dx;
Y2 += dy;
2015-05-23 13:35:12 +02:00
}
//----------------------------------------------------------------------
void FRect::move (const FPoint& d)
2015-05-23 13:35:12 +02:00
{
X1 += d.getX();
Y1 += d.getY();
X2 += d.getX();
Y2 += d.getY();
2015-05-23 13:35:12 +02:00
}
2019-08-11 18:15:57 +02:00
//----------------------------------------------------------------------
void FRect::scaleBy (int dx, int dy)
{
X2 += dx;
Y2 += dy;
}
//----------------------------------------------------------------------
void FRect::scaleBy (const FPoint& d)
{
X2 += d.getX();
Y2 += d.getY();
}
2015-05-23 13:35:12 +02:00
//----------------------------------------------------------------------
bool FRect::contains (int x, int y) const
{
return x >= X1 && x <= X2
&& y >= Y1 && y <= Y2;
}
//----------------------------------------------------------------------
bool FRect::contains (const FPoint& p) const
{
return p.getX() >= X1 && p.getX() <= X2
&& p.getY() >= Y1 && p.getY() <= Y2;
}
//----------------------------------------------------------------------
bool FRect::contains (const FRect& r) const
{
return r.X1 >= X1 && r.X2 <= X2
&& r.Y1 >= Y1 && r.Y2 <= Y2;
}
//----------------------------------------------------------------------
bool FRect::overlap (const FRect &r) const
{
return ( std::max(X1, r.X1) <= std::min(X2, r.X2)
2017-11-26 22:37:18 +01:00
&& std::max(Y1, r.Y1) <= std::min(Y2, r.Y2) );
2015-05-23 13:35:12 +02:00
}
//----------------------------------------------------------------------
FRect FRect::intersect (const FRect& r) const
{
// intersection: this ∩ r
int _X1 = std::max(X1, r.X1);
int _Y1 = std::max(Y1, r.Y1);
int _X2 = std::min(X2, r.X2);
int _Y2 = std::min(Y2, r.Y2);
const FPoint p1{ _X1, _Y1 };
const FPoint p2{ _X2, _Y2 };
return { p1, p2 };
2015-05-23 13:35:12 +02:00
}
2017-03-12 00:29:56 +01:00
//----------------------------------------------------------------------
FRect FRect::combined (const FRect& r) const
{
// Union: this r
int _X1 = std::min(X1, r.X1);
int _Y1 = std::min(Y1, r.Y1);
int _X2 = std::max(X2, r.X2);
int _Y2 = std::max(Y2, r.Y2);
const FPoint p1{ _X1, _Y1 };
const FPoint p2{ _X2, _Y2 };
return { p1, p2 };
2017-03-12 00:29:56 +01:00
}
2019-07-29 02:34:58 +02:00
// FRect non-member operators
2015-05-23 13:35:12 +02:00
//----------------------------------------------------------------------
FRect operator + (const FRect& r, const FSize& s)
2015-05-23 13:35:12 +02:00
{
return { r.X1
, r.Y1
, std::size_t(r.X2 - r.X1) + 1 + s.getWidth()
, std::size_t(r.Y2 - r.Y1) + 1 + s.getHeight() };
2015-05-23 13:35:12 +02:00
}
//----------------------------------------------------------------------
FRect operator - (const FRect& r, const FSize& s)
2015-05-23 13:35:12 +02:00
{
return { r.X1
, r.Y1
, std::size_t(r.X2 - r.X1) + 1 - s.getWidth()
, std::size_t(r.Y2 - r.Y1) + 1 - s.getHeight() };
2015-05-23 13:35:12 +02:00
}
//----------------------------------------------------------------------
bool operator == (const FRect& r1, const FRect& r2)
{
return r1.X1 == r2.X1
&& r1.Y1 == r2.Y1
&& r1.X2 == r2.X2
&& r1.Y2 == r2.Y2;
}
//----------------------------------------------------------------------
bool operator != (const FRect& r1, const FRect& r2)
{
return r1.X1 != r2.X1
|| r1.Y1 != r2.Y1
|| r1.X2 != r2.X2
|| r1.Y2 != r2.Y2;
}
2018-04-16 02:24:37 +02:00
//----------------------------------------------------------------------
std::ostream& operator << (std::ostream& outstr, const FRect& r)
{
outstr << r.X1 << " "
<< r.Y1 << " "
<< r.X2 << " "
<< r.Y2;
2018-04-16 02:24:37 +02:00
return outstr;
}
//----------------------------------------------------------------------
std::istream& operator >> (std::istream& instr, FRect& r)
{
2020-04-13 12:40:11 +02:00
int x1{};
int y1{};
int x2{};
int y2{};
2018-04-16 02:24:37 +02:00
instr >> x1;
instr >> y1;
instr >> x2;
instr >> y2;
r.setCoordinates (x1, y1, x2, y2);
return instr;
}
} // namespace finalcut