Blob Blame Raw
#pragma once

#ifndef TCG_RECT_H
#define TCG_RECT_H

#include "point.h"
#include "size.h"

// STD includes
#include <limits>

namespace tcg {

//**********************************************************************************
//    Bidimensional Rect  class
//**********************************************************************************

template <typename T>
struct RectT {
  T x0, y0, x1, y1;

public:
  RectT() : x0((std::numeric_limits<T>::max)()), y0(x0), x1(-x0), y1(x1) {}
  RectT(T x0_, T y0_, T x1_, T y1_) : x0(x0_), y0(y0_), x1(x1_), y1(y1_) {}
  RectT(const PointT<T> &p0, const PointT<T> &p1)
      : x0(p0.x), y0(p0.y), x1(p1.x), y1(p1.y) {}
  RectT(const PointT<T> &p0, const SizeT<T> &size)
      : x0(p0.x), y0(p0.y), x1(p0.x + size.w), y1(p0.y + size.h) {}

  bool empty() const { return (x1 <= x0) || (y1 <= y0); }

  PointT<T> p0() const { return PointT<T>(x0, y0); }
  PointT<T> p1() const { return PointT<T>(x1, y1); }
  PointT<T> center() const { return PointT<T>((x0 + x1) / 2, (y0 + y1) / 2); }

  T width() const { return x1 - x0; }
  T height() const { return y1 - y0; }
  SizeT<T> size() const { return SizeT<T>(width(), height()); }

  bool operator==(const RectT &other) const {
    return x0 == other.x0 && y0 == other.y0 && x1 == other.x1 && y1 == other.y1;
  }
  bool operator!=(const RectT &other) const { return !operator==(other); }

  RectT &operator+=(const PointT<T> &p) {
    x0 += p.x, y0 += p.y, x1 += p.x, y1 += p.y;
    return *this;
  }
  RectT &operator-=(const PointT<T> &p) {
    x0 -= p.x, y0 -= p.y, x1 -= p.x, y1 -= p.y;
    return *this;
  }

  friend RectT<T> operator+(const RectT<T> &r, const tcg::PointT<T> &p) {
    return RectT<T>(r.x0 + p.x, r.y0 + p.y, r.x1 + p.x, r.y1 + p.y);
  }
  friend RectT<T> operator-(const RectT<T> &r, const tcg::PointT<T> &p) {
    return RectT<T>(r.x0 - p.x, r.y0 - p.y, r.x1 - p.x, r.y1 - p.y);
  }
  friend RectT<T> operator+(const tcg::PointT<T> &p, const RectT<T> &r) {
    return RectT<T>(p.x + r.x0, p.y + r.y0, p.x + r.x1, p.y + r.y1);
  }
  friend RectT<T> operator-(const tcg::PointT<T> &p, const RectT<T> &r) {
    return RectT<T>(p.x - r.x0, p.y - r.y0, p.x - r.x1, p.y - r.y1);
  }

  template <typename K>
  RectT &operator*=(K k) {
    x0 *= k, y0 *= k, x1 *= k, y1 *= k;
    return *this;
  }

  template <typename K>
  friend RectT<T> operator*(const RectT<T> &r, K k) {
    return RectT<T>(r.x0 * k, r.y0 * k, r.x1 * k, r.y1 * k);
  }
  template <typename K>
  friend RectT<T> operator*(K k, const RectT<T> &r) {
    return RectT<T>(k * r.x0, k * r.y0, k * r.x1, k * r.y1);
  }

  RectT &operator|=(const RectT &other) {
    if (x0 > other.x0) x0 = other.x0;
    if (y0 > other.y0) y0 = other.y0;
    if (x1 < other.x1) x1 = other.x1;
    if (y1 < other.y1) y1 = other.y1;
    return *this;
  }

  RectT &operator&=(const RectT &other) {
    if (x0 < other.x0) x0 = other.x0;
    if (y0 < other.y0) y0 = other.y0;
    if (x1 > other.x1) x1 = other.x1;
    if (y1 > other.y1) y1 = other.y1;
    return *this;
  }

  RectT &operator|=(const PointT<T> &p) {
    return operator|=(RectT(p.x, p.y, p.x, p.y));
  }
  RectT &operator&=(const PointT<T> &p) {
    return operator&=(RectT(p.x, p.y, p.x, p.y));
  }

  friend RectT<T> operator|(const RectT<T> &a, const RectT<T> &b) {
    return RectT<T>(a) |= b;
  }
  friend RectT<T> operator&(const RectT<T> &a, const RectT<T> &b) {
    return RectT<T>(a) &= b;
  }

  friend RectT<T> operator|(const RectT<T> &r, const PointT<T> &p) {
    return RectT<T>(r) |= p;
  }
  friend RectT<T> operator&(const RectT<T> &r, const PointT<T> &p) {
    return RectT<T>(r) &= p;
  }
  friend RectT<T> operator|(const PointT<T> &p, const RectT<T> &r) {
    return RectT<T>(r) |= p;
  }
  friend RectT<T> operator&(const PointT<T> &p, const RectT<T> &r) {
    return RectT<T>(r) &= p;
  }
};

//------------------------------------------------------------------------

typedef RectT<double> RectD;
typedef RectT<int> RectI;
typedef RectI Rect;

//**********************************************************************************
//    Tridimensional Rect  class
//**********************************************************************************

template <typename T>
struct Rect3T {
  T x0, y0, z0, x1, y1, z1;

public:
  Rect3T()
      : x0((std::numeric_limits<T>::max)())
      , y0(x0)
      , z0(x0)
      , x1(-x0)
      , y1(x1)
      , z1(x1) {}
  Rect3T(T x0_, T y0_, T z0_, T x1_, T y1_, T z1_)
      : x0(x0_), y0(y0_), z0(z0_), x1(x1_), y1(y1_), z1(z1_) {}
  Rect3T(const Point3T<T> &p0, const Point3T<T> &p1)
      : x0(p0.x), y0(p0.y), z0(p0.z) x1(p1.x), y1(p1.y), z1(p1.z) {}
  Rect3T(const Point3T<T> &p0, const Size3T<T> &size)
      : x0(p0.x)
      , y0(p0.y)
      , z0(p0.z)
      , x1(p0.x + size.w)
      , y1(p0.y + size.h)
      , z1(p0.z + size.d) {}

  bool empty() const { return (x1 <= x0) || (y1 <= y0) || (z1 <= z0); }

  Point3T<T> p0() const { return Point3T<T>(x0, y0, z0); }
  Point3T<T> p1() const { return Point3T<T>(x1, y1, z1); }
  Point3T<T> center() const {
    return Point3T<T>((x0 + x1) / 2, (y0 + y1) / 2, (z0 + z1) / 2);
  }

  T width() const { return x1 - x0; }
  T height() const { return y1 - y0; }
  T depth() const { return z1 - z0; }
  Size3T<T> size() const { return Size3T<T>(width(), height(), depth()); }

  bool operator==(const Rect3T &other) const {
    return x0 == other.x0 && y0 == other.y0 && z0 == other.z0 &&
           x1 == other.x1 && y1 == other.y1 && z1 == other.z1;
  }
  bool operator!=(const Rect3T &other) const { return !operator==(other); }

  Rect3T &operator+=(const Point3T<T> &p) {
    x0 += p.x, y0 += p.y, z0 += p.z, x1 += p.x, y1 += p.y, z1 += p.z;
    return *this;
  }
  Rect3T &operator-=(const Point3T<T> &p) {
    x0 -= p.x, y0 -= p.y, z0 -= p.z, x1 -= p.x, y1 -= p.y, z1 -= p.z;
    return *this;
  }

  friend Rect3T<T> operator+(const Rect3T<T> &r, const tcg::Point3T<T> &p) {
    return Rect3T<T>(r.x0 + p.x, r.y0 + p.y, r.z0 + p.z, r.x1 + p.x, r.y1 + p.y,
                     r.z1 + p.z);
  }
  friend Rect3T<T> operator-(const Rect3T<T> &r, const tcg::Point3T<T> &p) {
    return Rect3T<T>(r.x0 - p.x, r.y0 - p.y, r.z0 - p.z, r.x1 - p.x, r.y1 - p.y,
                     r.z1 - p.z);
  }
  friend Rect3T<T> operator+(const tcg::Point3T<T> &p, const Rect3T<T> &r) {
    return Rect3T<T>(p.x + r.x0, p.x + r.y0, p.x + r.z0, p.x + r.x1 + p.x, r.y1,
                     p.x + r.z1);
  }
  friend Rect3T<T> operator-(const tcg::Point3T<T> &p, const Rect3T<T> &r) {
    return Rect3T<T>(p.x - r.x0, p.y - r.y0, p.z - r.z0, p.x - r.x1, p.y - r.y1,
                     p.z - r.z1);
  }

  template <typename K>
  Rect3T &operator*=(K k) {
    x0 *= k, y0 *= k, z0 *= k, x1 *= k, y1 *= k, z1 *= k;
    return *this;
  }

  template <typename K>
  friend Rect3T<T> operator*(const Rect3T<T> &r, K k) {
    return RectT<T>(r.x0 * k, r.y0 * k, r.z0 * k, r.x1 * k, r.y1 * k, r.z1 * k);
  }
  template <typename K>
  friend Rect3T<T> operator*(K k, const Rect3T<T> &r) {
    return RectT<T>(k * r.x0, k * r.y0, k * r.z0, k * r.x1, k * r.y1, k * r.z1);
  }

  Rect3T &operator|=(const Rect3T &other) {
    if (x0 > other.x0) x0 = other.x0;
    if (y0 > other.y0) y0 = other.y0;
    if (z0 > other.z0) z0 = other.z0;
    if (x1 < other.x1) x1 = other.x1;
    if (y1 < other.y1) y1 = other.y1;
    if (z1 < other.z1) z1 = other.z1;
    return *this;
  }

  Rect3T &operator&=(const Rect3T &other) {
    if (x0 < other.x0) x0 = other.x0;
    if (y0 < other.y0) y0 = other.y0;
    if (z0 < other.z0) z0 = other.z0;
    if (x1 > other.x1) x1 = other.x1;
    if (y1 > other.y1) y1 = other.y1;
    if (z1 > other.z1) z1 = other.z1;
    return *this;
  }

  Rect3T &operator|=(const Point3T<T> &p) {
    return operator|=(RectT(p.x, p.y, p.z, p.x, p.y, p.z));
  }
  Rect3T &operator&=(const Point3T<T> &p) {
    return operator&=(RectT(p.x, p.y, p.z, p.x, p.y, p.z));
  }

  friend Rect3T<T> operator|(const Rect3T<T> &a, const Rect3T<T> &b) {
    return Rect3T<T>(a) |= b;
  }
  friend Rect3T<T> operator&(const Rect3T<T> &a, const Rect3T<T> &b) {
    return Rect3T<T>(a) &= b;
  }

  friend Rect3T<T> operator|(const Rect3T<T> &r, const Point3T<T> &p) {
    return Rect3T<T>(r) |= p;
  }
  friend Rect3T<T> operator&(const Rect3T<T> &r, const Point3T<T> &p) {
    return Rect3T<T>(r) &= p;
  }
  friend Rect3T<T> operator|(const Point3T<T> &p, const Rect3T<T> &r) {
    return Rect3T<T>(r) |= p;
  }
  friend Rect3T<T> operator&(const Point3T<T> &p, const Rect3T<T> &r) {
    return Rect3T<T>(r) &= p;
  }
};

//------------------------------------------------------------------------

typedef Rect3T<double> Rect3D;
typedef Rect3T<int> Rect3I;
typedef RectI Rect;

}  // namespace tcg

#endif  // TCG_RECT_H