/* * 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 */ #include "extent.hpp" #include "debug.hpp" #include "maths.hpp" #include using util::extent; /////////////////////////////////////////////////////////////////////////////// template extent::extent (vector _v) { std::transform (std::begin (_v), std::end (_v), this->begin (), [] (auto i) { // using std::abs gives unsigned abs warnings under clang. this tricks // it sufficiently to quiet the warnings. return i < 0 ? -i : i; }); } /////////////////////////////////////////////////////////////////////////////// template T extent::diameter (void) const { return static_cast ( std::sqrt ( std::accumulate (std::begin (this->data), std::end (this->data), T {0}, [] (auto a, auto b) { return a + b * b; }) ) ); } //----------------------------------------------------------------------------- template T extent::area (void) const { return std::accumulate (std::begin (this->data), std::end (this->data), T {1}, std::multiplies ()); } /////////////////////////////////////////////////////////////////////////////// template extent extent::expanded (util::vector mag) const { return *this + mag; } //----------------------------------------------------------------------------- template extent extent::expanded (T t) const { return *this + util::vector {t}; } //----------------------------------------------------------------------------- template extent extent::contracted (vector t) const { return *this - t; } //----------------------------------------------------------------------------- template extent extent::contracted (T t) const { return *this - vector {t}; } /////////////////////////////////////////////////////////////////////////////// template bool extent::empty (void) const { return almost_equal (area(), 0); } /////////////////////////////////////////////////////////////////////////////// template const extent extent::MIN { 0 }; //----------------------------------------------------------------------------- template const extent extent::MAX { std::numeric_limits::max () }; /////////////////////////////////////////////////////////////////////////////// namespace debug { template struct validator { static bool is_valid (const extent &e) { return std::all_of (std::begin (e.data), std::end (e.data), [] (auto i) { return i >= 0; }); } }; } template bool debug::valid (const extent<2,float>&); template bool debug::valid (const extent<2,double>&); template bool debug::valid (const extent<2,uint16_t>&); template bool debug::valid (const extent<2,uint32_t>&); template bool debug::valid (const extent<2,uint64_t>&); /////////////////////////////////////////////////////////////////////////////// template std::ostream& util::operator<< (std::ostream &os, extent e) { os << "["; std::copy (std::begin (e.data), std::end (e.data), std::ostream_iterator (os, ", ")); os << "]"; return os; } //----------------------------------------------------------------------------- namespace util { #define INSTANTIATE_S_T(S,T) \ template struct extent; \ template std::ostream& operator<< (std::ostream&, extent); #define INSTANTIATE(T) \ INSTANTIATE_S_T(2,T) \ INSTANTIATE_S_T(3,T) INSTANTIATE(uint16_t) INSTANTIATE(uint32_t) INSTANTIATE(uint64_t) INSTANTIATE(float) INSTANTIATE(double) }