/* * 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 2010-2015 Danny Robson */ #ifndef __UTIL_EXTENT_HPP #define __UTIL_EXTENT_HPP #include "coord/base.hpp" #include "vector.hpp" #include "point.hpp" namespace util { /** * A pure two-dimensional size, without positioning */ template struct extent : public coord::base { using coord::base::base; extent () = default; explicit extent (vector); constexpr T area (void) const; constexpr T diameter (void) const; template constexpr U aspect (void) const; template bool includes (util::point) const; extent expanded (vector) const; extent expanded (T) const; extent contracted (vector) const; extent contracted (T) const; bool empty (void) const; static constexpr extent max (void); static constexpr extent min (void); }; template struct extent_range { public: struct iterator : public std::iterator, size_t> { public: iterator (extent, util::point); point operator* () const; iterator& operator++ (void); bool operator!= (const iterator &rhs) const; bool operator== (const iterator &rhs) const; private: point m_cursor; extent m_target; }; explicit extent_range (extent target); iterator begin (void) const; iterator end (void) const; private: extent m_target; }; /// create an extent from supplied arguments, optionally specifying the /// underlying type. /// /// much like experimental::make_array we use a void type to signal we /// need to deduce the underlying type. template < typename _T = void, typename ...Args > auto make_extent (Args &&...args) { using T = std::conditional_t< std::is_void_v<_T>, std::common_type_t, _T >; return extent { std::forward (args)... }; } /////////////////////////////////////////////////////////////////////////// template extent_range make_range (extent e) { return extent_range {e}; } /////////////////////////////////////////////////////////////////////////// // convenience typedefs template using extent2 = extent<2,T>; template using extent3 = extent<3,T>; template using extentu = extent; template using extenti = extent; template using extentf = extent; template using extentd = extent; typedef extent2 extent2i; typedef extent2 extent2u; typedef extent2 extent2f; typedef extent2 extent2d; typedef extent3 extent3u; typedef extent3 extent3f; //------------------------------------------------------------------------- template using extent_range2 = extent_range<2,T>; template using extent_range3 = extent_range<3,T>; using extent_range2u = extent_range2; using extent_range2i = extent_range2; using extent_range3u = extent_range2; } #include "extent.ipp" #endif