chromium/third_party/mediapipe/src/mediapipe/framework/deps/rectangle.h

// Copyright 2019 The MediaPipe Authors.
//
// Licensed under the Apache License, Version 2.0 (the "License");
// you may not use this file except in compliance with the License.
// You may obtain a copy of the License at
//
//      http://www.apache.org/licenses/LICENSE-2.0
//
// Unless required by applicable law or agreed to in writing, software
// distributed under the License is distributed on an "AS IS" BASIS,
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
// See the License for the specific language governing permissions and
// limitations under the License.
//
// Class for axis-aligned rectangles represented as two corner points
// (min_x, min_y) and (max_x, max_y).  The methods such as Contain, Intersect
// and IsEmpty() assume that the points in region include the 4 boundary edges.
// The default box is initialized so that IsEmpty() is true.  Note that the
// use of corner points supports both right-handed (Cartesian) and left-
// handed (image) coordinate systems.

#ifndef MEDIAPIPE_DEPS_RECTANGLE_H_
#define MEDIAPIPE_DEPS_RECTANGLE_H_

#include <algorithm>
#include <cmath>
#include <cstdint>
#include <iosfwd>
#include <limits>
#include <ostream>

#include "mediapipe/framework/deps/point2.h"

template <typename T>
class Rectangle;

template <typename T>
std::ostream& operator<<(std::ostream&, const Rectangle<T>&);

template <typename T>
class Rectangle {
 public:
  typedef Rectangle<T> Self;

  // Default constructed rectangle which is empty.
  Rectangle() { SetEmpty(); }

  // Creates a rectangle from the minimum point and the dimensions.
  Rectangle(const T& x, const T& y, const T& width, const T& height);

  // Creates a rectangle given two points.  The resulting rectangle will
  // have non-negative width and height.
  Rectangle(const Point2<T>& p0, const Point2<T>& p1);

  // Same as above but using vectors as input.
  Rectangle(const Vector2<T>& p0, const Vector2<T>& p1);

  // Sets min to be very large numbers and max to be very large negative numbers
  // so that points can be used to correctly extend the rectangle.
  void SetEmpty();

  // A rectangle is empty if there are no points inside of it.  A degenerate
  // rectangle where the corners are coincident has zero area but is not empty.
  bool IsEmpty() const { return min_.x() > max_.x() || min_.y() > max_.y(); }

  bool operator==(const Rectangle&) const;
  bool operator!=(const Rectangle&) const;

  // Width and height are both max - min, which may be negative if SetEmpty()
  // was called or the user explicity set the min and max points.
  T Width() const { return max_.x() - min_.x(); }
  T Height() const { return max_.y() - min_.y(); }

  // Computes the area, which is negative if the width xor height is negative.
  // The value is undefined if SetEmpty() is called.
  // Watch out for large integer rectangles because the area may overflow.
  T Area() const { return Width() * Height(); }

  // Accessors are provided for both points and sides.
  const T& xmin() const { return min_.x(); }
  const T& xmax() const { return max_.x(); }
  const T& ymin() const { return min_.y(); }
  const T& ymax() const { return max_.y(); }

  // Returns the min and max corner points.
  const Point2<T>& min_xy() const { return min_; }
  const Point2<T>& max_xy() const { return max_; }

  // Sets the geometry of the rectangle given two points.
  // The resulting rectangle will have non-negative width and height.
  void Set(const Point2<T>& p0, const Point2<T>& p1);

  // Same as above using vectors as input.
  void Set(const Vector2<T>& p0, const Vector2<T>& p1);

  // Sets the geometry of the rectangle given a minimum point and dimensions.
  void Set(const T& x, const T& y, const T& width, const T& height);

  // Sets the min and max values, and min greater than max is allowable,
  // but the user has to be aware of the consequences such as negative width
  // and height.  Both point and side accessors are provided.
  void set_xmin(const T& x) { min_.set_x(x); }
  void set_xmax(const T& x) { max_.set_x(x); }
  void set_ymin(const T& y) { min_.set_y(y); }
  void set_ymax(const T& y) { max_.set_y(y); }

  void set_min_xy(const Point2<T>& p) { min_.Set(p.x(), p.y()); }
  void set_max_xy(const Point2<T>& p) { max_.Set(p.x(), p.y()); }

  // Expands a rectangle to contain a point or vector.
  void Expand(const T& x, const T& y);
  void Expand(const Point2<T>& p);
  void Expand(const Vector2<T>& p);

  // Expands a rectangle to contain another rectangle.
  void Expand(const Rectangle& other);

  // Returns the union of this rectangle with another rectangle, which
  // is the smallest rectangle that contains both rectangles.
  Rectangle Union(const Rectangle& other) const;

  // Returns the intersection of this rectangle with another rectangle.
  // If the intersection is empty, returns a rectangle initialized by
  // SetEmpty().
  Rectangle Intersect(const Rectangle& other) const;

  // Tests if this rectangle has a non-empty intersection with another rectangle
  // including the boundary.
  bool Intersects(const Rectangle& other) const;

  // Tests if a point is inside or on any of the 4 edges of the rectangle.
  bool Contains(const T& x, const T& y) const;
  bool Contains(const Point2<T>& pt) const;
  bool Contains(const Vector2<T>& pt) const;

  // Tests if a rectangle is inside or on any of the 4 edges of the rectangle.
  bool Contains(const Rectangle& other) const;

  // Translates this rectangle by a vector.
  void Translate(const Vector2<T>& v);

  // Adds a border around the rectangle by subtracting the border size from the
  // min point and adding it to the max point.  The border size can be
  // negative.
  void AddBorder(const T& border_size);

  // Debug printing.
  friend std::ostream& operator<< <T>(std::ostream&, const Rectangle&);

 private:
  Point2<T> min_;
  Point2<T> max_;
};

//
// Inline method definitions.  These are not placed in the definition of the
// class to keep the class interface more readable.
//

template <typename T>
Rectangle<T>::Rectangle(const Point2<T>& p0, const Point2<T>& p1) {
  Set(p0, p1);
}

template <typename T>
Rectangle<T>::Rectangle(const Vector2<T>& p0, const Vector2<T>& p1) {
  Set(p0, p1);
}

template <typename T>
Rectangle<T>::Rectangle(const T& x, const T& y, const T& width,
                        const T& height) {
  Set(x, y, width, height);
}

// The general version works only when T models Integer (there are more
// integer classes than float classes).
template <typename T>
void Rectangle<T>::SetEmpty() {
  T min_value = std::numeric_limits<T>::min();
  T max_value = std::numeric_limits<T>::max();
  min_.Set(max_value, max_value);
  max_.Set(min_value, min_value);
}

template <>
inline void Rectangle<float>::SetEmpty() {
  float max_value = std::numeric_limits<float>::max();
  min_.Set(max_value, max_value);
  max_.Set(-max_value, -max_value);
}

template <>
inline void Rectangle<double>::SetEmpty() {
  double max_value = std::numeric_limits<double>::max();
  min_.Set(max_value, max_value);
  max_.Set(-max_value, -max_value);
}

template <typename T>
bool Rectangle<T>::operator==(const Rectangle<T>& other) const {
  return min_ == other.min_ && max_ == other.max_;
}

template <typename T>
bool Rectangle<T>::operator!=(const Rectangle<T>& other) const {
  return !(*this == other);
}

template <typename T>
void Rectangle<T>::Set(const Vector2<T>& p0, const Vector2<T>& p1) {
  if (p0[0] <= p1[0])
    min_.set_x(p0[0]), max_.set_x(p1[0]);
  else
    max_.set_x(p0[0]), min_.set_x(p1[0]);

  if (p0[1] <= p1[1])
    min_.set_y(p0[1]), max_.set_y(p1[1]);
  else
    max_.set_y(p0[1]), min_.set_y(p1[1]);
}

template <typename T>
void Rectangle<T>::Set(const Point2<T>& p0, const Point2<T>& p1) {
  Set(p0.ToVector(), p1.ToVector());
}

template <typename T>
void Rectangle<T>::Set(const T& x, const T& y, const T& width,
                       const T& height) {
  min_.Set(x, y);
  max_.Set(x + width, y + height);
}

template <typename T>
void Rectangle<T>::Expand(const T& x, const T& y) {
  min_.Set(std::min(x, xmin()), std::min(y, ymin()));
  max_.Set(std::max(x, xmax()), std::max(y, ymax()));
}

template <typename T>
void Rectangle<T>::Expand(const Point2<T>& p) {
  Expand(p.x(), p.y());
}

template <typename T>
void Rectangle<T>::Expand(const Vector2<T>& v) {
  Expand(v[0], v[1]);
}

template <typename T>
void Rectangle<T>::Expand(const Rectangle<T>& other) {
  Expand(other.min_);
  Expand(other.max_);
}

template <typename T>
void Rectangle<T>::Translate(const Vector2<T>& v) {
  min_ += v;
  max_ += v;
}

template <typename T>
bool Rectangle<T>::Contains(const T& x, const T& y) const {
  return x >= xmin() && x <= xmax() && y >= ymin() && y <= ymax();
}

template <typename T>
bool Rectangle<T>::Contains(const Point2<T>& p) const {
  return Contains(p.x(), p.y());
}

template <typename T>
bool Rectangle<T>::Contains(const Vector2<T>& v) const {
  return Contains(v[0], v[1]);
}

template <typename T>
bool Rectangle<T>::Contains(const Rectangle<T>& r) const {
  return Contains(r.min_) && Contains(r.max_);
}

template <typename T>
Rectangle<T> Rectangle<T>::Union(const Rectangle<T>& r) const {
  return Rectangle<T>(
      Point2<T>(std::min(xmin(), r.xmin()), std::min(ymin(), r.ymin())),
      Point2<T>(std::max(xmax(), r.xmax()), std::max(ymax(), r.ymax())));
}

template <typename T>
Rectangle<T> Rectangle<T>::Intersect(const Rectangle<T>& r) const {
  Point2<T> pmin(std::max(xmin(), r.xmin()), std::max(ymin(), r.ymin()));
  Point2<T> pmax(std::min(xmax(), r.xmax()), std::min(ymax(), r.ymax()));

  if (pmin.x() > pmax.x() || pmin.y() > pmax.y())
    return Rectangle<T>();
  else
    return Rectangle<T>(pmin, pmax);
}

template <typename T>
bool Rectangle<T>::Intersects(const Rectangle<T>& r) const {
  return !(IsEmpty() || r.IsEmpty() || r.xmax() < xmin() || xmax() < r.xmin() ||
           r.ymax() < ymin() || ymax() < r.ymin());
}

template <typename T>
void Rectangle<T>::AddBorder(const T& border_size) {
  min_.Set(xmin() - border_size, ymin() - border_size);
  max_.Set(xmax() + border_size, ymax() + border_size);
}

template <typename T>
std::ostream& operator<<(std::ostream& out, const Rectangle<T>& r) {
  out << "[(" << r.xmin() << ", " << r.ymin() << "), (" << r.xmax() << ", "
      << r.ymax() << ")]";
  return out;
}

template <typename T>
class Rectangle;

typedef Rectangle<uint8_t> Rectangle_b;
typedef Rectangle<int> Rectangle_i;
typedef Rectangle<float> Rectangle_f;
typedef Rectangle<double> Rectangle_d;

#endif  // MEDIAPIPE_DEPS_RECTANGLE_H_