/* ========================================================================= * * * * OpenMesh * * Copyright (c) 2001-2019, RWTH-Aachen University * * Department of Computer Graphics and Multimedia * * All rights reserved. * * www.openmesh.org * * * *---------------------------------------------------------------------------* * This file is part of OpenMesh. * *---------------------------------------------------------------------------* * * * Redistribution and use in source and binary forms, with or without * * modification, are permitted provided that the following conditions * * are met: * * * * 1. Redistributions of source code must retain the above copyright notice, * * this list of conditions and the following disclaimer. * * * * 2. Redistributions in binary form must reproduce the above copyright * * notice, this list of conditions and the following disclaimer in the * * documentation and/or other materials provided with the distribution. * * * * 3. Neither the name of the copyright holder nor the names of its * * contributors may be used to endorse or promote products derived from * * this software without specific prior written permission. * * * * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS * * "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED * * TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A * * PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER * * OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, * * EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, * * PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR * * PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF * * LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING * * NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS * * SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. * * * * ========================================================================= */ #pragma once #include #include #include //== NAMESPACES =============================================================== namespace OpenMesh { //== FORWARD DECLARATION ====================================================== //== CLASS DEFINITION ========================================================= namespace { struct Identity { template T operator()(const T& _t) const { return _t; } }; } /// Base class for all smart range types template struct SmartRangeT { // TODO: Someone with better c++ knowledge may improve the code below. template auto sum(Functor&& f) -> decltype (f(std::declval())+f(std::declval())) { auto range = static_cast(this); auto begin = range->begin(); auto end = range->end(); assert(begin != end); decltype (f(*begin) + f(*begin)) sum = f(*begin); auto it = begin; ++it; for (; it != end; ++it) sum += f(*it); return sum; } template auto avg(Functor&& f) -> decltype (1.0 * (f(std::declval())+f(std::declval()))) { auto range = static_cast(this); auto begin = range->begin(); auto end = range->end(); assert(begin != end); decltype (f(*begin) + f(*begin)) sum = f(*begin); auto it = begin; ++it; int n_elements = 1; for (; it != end; ++it) { sum += f(*it); ++n_elements; } return (1.0 / n_elements) * sum; } template auto to_array(Functor&& f = {}) -> std::array()))>::type, n> { auto range = static_cast(this); std::array()))>::type, n> res; auto it = range->begin(); auto end = range->end(); int i = 0; while (i < n && it != end) res[i++] = f(*(it++)); return res; } template auto to_vector(Functor&& f = {}) -> std::vector()))>::type> { auto range = static_cast(this); std::vector()))>::type> res; for (auto e : *range) res.push_back(f(e)); return res; } template auto min(Functor&& f) -> typename std::remove_reference()))>::type { using std::min; auto range = static_cast(this); auto it = range->begin(); auto end = range->end(); assert(it != end); typename std::remove_reference()))>::type res = f(*it); ++it; for (; it != end; ++it) res = min(res, f(*it)); return res; } template auto max(Functor&& f) -> typename std::remove_reference()))>::type { using std::max; auto range = static_cast(this); auto it = range->begin(); auto end = range->end(); assert(it != end); typename std::remove_reference()))>::type res = f(*it); ++it; for (; it != end; ++it) res = max(res, f(*it)); return res; } template auto minmax(Functor&& f) -> std::pair()))>::type, typename std::remove_reference()))>::type> { return std::make_pair(this->min(f), this->max(f)); } }; //============================================================================= } // namespace OpenMesh //============================================================================= //=============================================================================