Wed, 08 Jun 2022 19:33:00 +0300
Concentrate model editing into one coroutine inside main()
/* * LDForge: LDraw parts authoring CAD * Copyright (C) 2013 - 2020 Teemu Piippo * * This program is free software: you can redistribute it and/or modify * it under the terms of the GNU General Public License as published by * the Free Software Foundation, either version 3 of the License, or * (at your option) any later version. * * This program is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the * GNU General Public License for more details. * * You should have received a copy of the GNU General Public License * along with this program. If not, see <http://www.gnu.org/licenses/>. */ #pragma once #include <algorithm> #include <cstdio> #include <cstdlib> #include <cstring> #include <cmath> #include <optional> #include <type_traits> #include <QMatrix4x4> #include <QObject> #include <QPointF> #include <QSet> #include <QString> #include <QStringList> #include <QVariant> #include <QVector> #include <QVector3D> #include <glm/glm.hpp> using GLRotationMatrix = QMatrix4x4; enum Axis { X = 0, Y = 1, Z = 2 }; enum Result { Success = 0, PartialSuccess, Failure }; constexpr bool failed(Result r) { return r == Failure; } enum Winding { NoWinding, Anticlockwise, Clockwise, }; /* * Special operator definition that implements the XOR operator for windings. * However, if either winding is NoWinding, then this function returns NoWinding. */ inline Winding operator^(Winding one, Winding other) { if (one == NoWinding or other == NoWinding) return NoWinding; else return static_cast<Winding>(static_cast<int>(one) ^ static_cast<int>(other)); } inline Winding& operator^=(Winding& one, Winding other) { one = one ^ other; return one; } template<typename T, int N> constexpr int countof(T const (&)[N]) { return N; } /** * @brief casts @c x to a suitable unsigned integer */ template<typename T> constexpr auto unsigned_cast(T x) -> std::enable_if_t<std::is_integral_v<T>, std::make_unsigned_t<T>> { return static_cast<std::make_unsigned_t<T>>(x); } /** * @brief casts @c x to a suitable signed integer */ template<typename T> constexpr auto signed_cast(T x) -> std::enable_if_t<std::is_integral_v<T>, std::make_signed_t<T>> { return static_cast<std::make_signed_t<T>>(x); } /** * @brief casts floating point values to float, converting non-floating point values causes an error * @param[in] x floating point value to cast * @returns float */ template<typename T> auto toFloat(T x) -> std::enable_if_t<std::is_floating_point_v<T>, float> { return static_cast<float>(x); } /** * @brief casts floating point values to double, converting non-floating point values causes an error * @param[in] x floating point value to cast * @returns double */ template<typename T> auto toDouble(T x) -> std::enable_if_t<std::is_floating_point_v<T>, double> { return static_cast<double>(x); } /** * @brief casts floating point values to qreal, converting non-floating point values causes an error * @param[in] x floating point value to cast * @returns qreal */ template<typename T> auto toQreal(T x) -> std::enable_if_t<std::is_floating_point_v<T>, qreal> { return static_cast<qreal>(x); } template<int N, typename T, glm::qualifier Q> inline QPoint toQPoint(const glm::vec<N, T, Q>& vec) { return {static_cast<int>(vec.x), static_cast<int>(vec.y)}; } template<int N, typename T, glm::qualifier Q> inline QPointF toQPointF(const glm::vec<N, T, Q>& vec) { return {toDouble(vec.x), toDouble(vec.y)}; } inline glm::vec2 toVec2(const QPoint& point) { return {point.x(), point.y()}; } inline glm::vec2 toVec2(const QPointF& point) { return {point.x(), point.y()}; } /* * coalesce(arg1, arg2, ..., argn) * Returns the first of the given arguments that evaluates to true. */ template<typename T> T coalesce(T&& arg) { // recursion base: 1 argument return arg; } template<typename T, typename... Rest> std::common_type_t<T, Rest...> coalesce(T&& arg, Rest&&... rest) { // general case: n arguments return arg ? arg : coalesce(rest...); } /** * @brief Finds an element in a map and possibly returns a reference to it if find * @param map * @param key * @returns the value or nullptr if not found */ template<typename T, typename R, typename K> const R* findInMap(const std::map<T, R>& map, K&& key) { auto pair = map.find(key); if (pair != map.end()) { return &pair->second; } else { return nullptr; } } /** * @brief Finds an element in a map and possibly returns a reference to it if find * @param map * @param key * @returns the value or no value if not found */ template<typename T, typename R, typename K> R* findInMap(std::map<T, R>& map, K&& key) { auto pair = map.find(key); if (pair != map.end()) { return &pair->second; } else { return nullptr; } } template<typename T = float> constexpr std::enable_if_t<std::is_floating_point_v<T>, T> pi = static_cast<T>(M_PIl); constexpr double infinity = std::numeric_limits<double>::infinity(); Q_DECLARE_METATYPE(glm::vec3) Q_DECLARE_METATYPE(glm::mat4)