/* * 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. * * 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; } ::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