/* * This Source Code Form is subject to the terms of the Mozilla Public * License, v. 2.0. If a copy of the MPL was not distributed with this * file, You can obtain one at http://mozilla.org/MPL/2.0/. * * Copyright 2015-2017 Danny Robson */ #ifndef CRUFT_UTIL_GEOM_AABB_HPP #define CRUFT_UTIL_GEOM_AABB_HPP #include "../debug.hpp" #include "../extent.hpp" #include "../point.hpp" #include namespace util::geom { /////////////////////////////////////////////////////////////////////////// /// represents an axis-aligned bounding-box through two opposing corners. /// /// `lo' must be less-than-or-equal to `hi'. equality is allowed so that /// we can represent zero sized bounding-boxes. template struct aabb { aabb () = default; aabb (point _lo, point _hi): lo (_lo), hi (_hi) { CHECK (all (lo <= hi)); } /////////////////////////////////////////////////////////////////////// extent magnitude (void) const { return (hi - lo).template as (); } //--------------------------------------------------------------------- T diameter (void) const { return magnitude ().diameter (); } //--------------------------------------------------------------------- /// tests whether a point lies within the region, inclusive of borders constexpr bool inclusive (point p) const noexcept { return all (lo <= p && hi >= p); } //--------------------------------------------------------------------- point closest (point query) const { return clamp (query, lo, hi); } //--------------------------------------------------------------------- void cover (point p) { lo = min (p, lo); hi = max (p, hi); } //--------------------------------------------------------------------- aabb operator+ (vector v) const { return { lo + v, hi + v }; } aabb operator- (vector v) const { return { lo - v, hi - v }; } /// returns an aabb that covers the supplied point in addition to the /// current aabb area. auto operator| [[nodiscard]] (point p) const noexcept { return aabb ( util::min (lo, p), util::max (hi, p) ); } auto& operator|= (point p) noexcept { return *this = *this | p; } aabb operator| [[nodiscard]] (aabb rhs) const noexcept { return { min (lo, rhs.lo), max (hi, rhs.hi) }; } std::array,util::pow(2,S)> vertices (void) const noexcept; ::util::point lo; ::util::point hi; }; /////////////////////////////////////////////////////////////////////////// template constexpr bool operator== (const aabb &a, const aabb &b) noexcept { return a.lo == b.lo && a.hi == b.hi; } /////////////////////////////////////////////////////////////////////////// typedef aabb<2,float> aabb2f; typedef aabb<2,unsigned> aabb2u; typedef aabb<2,int> aabb2i; typedef aabb<3,float> aabb3f; typedef aabb<3,unsigned> aabb3u; typedef aabb<3,int> aabb3i; } /////////////////////////////////////////////////////////////////////////////// #include "./sample.hpp" #include namespace util::geom { template struct sampler { static point fn (aabb b, G &g) { std::uniform_real_distribution d; point p; std::generate (p.begin (), p.end (), [&] (void) { return d (g); }); return p * (b.hi - b.lo) + b.lo.template as (); } }; } #endif