/************************************************************************ ** ** @file compatibility.h ** @author Roman Telezhynskyi ** @date 16 1, 2020 ** ** @brief ** @copyright ** This source code is part of the Valentina project, a pattern making ** program, whose allow create and modeling patterns of clothing. ** Copyright (C) 2020 Valentina project ** All Rights Reserved. ** ** Valentina 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. ** ** Valentina 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 Valentina. If not, see . ** *************************************************************************/ #ifndef COMPATIBILITY_H #define COMPATIBILITY_H #include #include #include #include #include #include #include #include "defglobal.h" #if QT_VERSION >= QT_VERSION_CHECK(6, 0, 0) // WARNING:QVariant::load: unknown user type with name QMarginsF. // QVariant::value() fails to convert unless QVariant::fromValue() has been called previously. // https://stackoverflow.com/questions/70974383/qvariantvaluet-fails-to-convert-unless-qvariantfromvaluet-has-been-c #if QT_VERSION >= QT_VERSION_CHECK(6, 1, 0) #define REGISTER_META_TYPE_STREAM_OPERATORS(TYPE) \ QMetaType::fromType().hasRegisteredDataStreamOperators(); // Dummy call #else #define REGISTER_META_TYPE_STREAM_OPERATORS(TYPE) QVariant::fromValue(TYPE{}); // Dummy call #endif // QT_VERSION >= QT_VERSION_CHECK(6, 1, 0) #else #define REGISTER_META_TYPE_STREAM_OPERATORS(TYPE) qRegisterMetaTypeStreamOperators(#TYPE); #endif // QT_VERSION >= QT_VERSION_CHECK(6, 0, 0) #if QT_VERSION < QT_VERSION_CHECK(5, 5, 0) #include "diagnostic.h" #endif // QT_VERSION < QT_VERSION_CHECK(5, 5, 0) class QPointF; // Contains helpful methods to hide version dependent code. It can be deprecation of method or change in API //--------------------------------------------------------------------------------------------------------------------- template class Cont> inline auto ConstFirst(const Cont &container) -> const T & { #if QT_VERSION >= QT_VERSION_CHECK(5, 6, 0) return container.constFirst(); #else return container.first(); // clazy:exclude=detaching-temporary #endif } //--------------------------------------------------------------------------------------------------------------------- template inline auto ConstFirst(const C &container) -> const T & { #if QT_VERSION >= QT_VERSION_CHECK(5, 6, 0) return container.constFirst(); #else return container.first(); // clazy:exclude=detaching-temporary #endif } //--------------------------------------------------------------------------------------------------------------------- template class Cont> inline auto ConstLast(const Cont &container) -> const T & { #if QT_VERSION >= QT_VERSION_CHECK(5, 6, 0) return container.constLast(); #else return container.last(); // clazy:exclude=detaching-temporary #endif } //--------------------------------------------------------------------------------------------------------------------- template inline auto ConstLast(const C &container) -> const T & { #if QT_VERSION >= QT_VERSION_CHECK(5, 6, 0) return container.constLast(); #else return container.last(); // clazy:exclude=detaching-temporary #endif } //--------------------------------------------------------------------------------------------------------------------- template inline auto Intersects(const T &l1, const T &l2, QPointF *intersectionPoint) -> typename T::IntersectType { #if QT_VERSION >= QT_VERSION_CHECK(5, 14, 0) return l1.intersects(l2, intersectionPoint); #else return l1.intersect(l2, intersectionPoint); #endif } //--------------------------------------------------------------------------------------------------------------------- template class C> inline auto ConvertToList(const C &container) -> QList { #if QT_VERSION >= QT_VERSION_CHECK(5, 14, 0) return QList(container.begin(), container.end()); #else return container.toList(); #endif } //--------------------------------------------------------------------------------------------------------------------- template class C> inline auto ConvertToStringList(const C &container) -> QStringList { return {ConvertToList(container)}; } //--------------------------------------------------------------------------------------------------------------------- template class C> inline auto ConvertToSet(const C &container) -> QSet { #if QT_VERSION >= QT_VERSION_CHECK(5, 14, 0) return QSet(container.begin(), container.end()); #else return container.toSet(); #endif } //--------------------------------------------------------------------------------------------------------------------- template inline auto ConvertToSet(const C &container) -> QSet { #if QT_VERSION >= QT_VERSION_CHECK(5, 14, 0) return QSet(container.begin(), container.end()); #else return container.toSet(); #endif } //--------------------------------------------------------------------------------------------------------------------- template class C> inline auto ConvertToVector(const C &container) -> QVector { #if QT_VERSION >= QT_VERSION_CHECK(5, 14, 0) return QVector(container.begin(), container.end()); #else return container.toVector(); #endif } //--------------------------------------------------------------------------------------------------------------------- template inline auto ConvertToVector(const QSet &container) -> QVector { #if QT_VERSION >= QT_VERSION_CHECK(5, 14, 0) return QVector(container.begin(), container.end()); #else return container.toList().toVector(); #endif } //--------------------------------------------------------------------------------------------------------------------- // NOTE: Delete if not necessary anymore // template // inline void SwapItemsAt(T &container, int i, int j) //{ // #if QT_VERSION >= QT_VERSION_CHECK(5, 13, 0) // container.swapItemsAt(i, j); // #else // container.swap(i, j); // #endif //} //--------------------------------------------------------------------------------------------------------------------- template inline void Move(T &vector, int from, int to) { QT_WARNING_PUSH // cppcheck-suppress unknownMacro QT_WARNING_DISABLE_GCC("-Wstrict-overflow") #if QT_VERSION < QT_VERSION_CHECK(5, 6, 0) Q_ASSERT_X(from >= 0 && from < vector.size(), "QVector::move(int,int)", "'from' is out-of-range"); Q_ASSERT_X(to >= 0 && to < vector.size(), "QVector::move(int,int)", "'to' is out-of-range"); if (from == to) // don't detach when no-op { return; } typename T::iterator b = vector.begin(); from < to ? std::rotate(b + from, b + from + 1, b + to + 1) : std::rotate(b + to, b + from, b + from + 1); #else vector.move(from, to); #endif // QT_VERSION < QT_VERSION_CHECK(5, 6, 0) QT_WARNING_POP } //--------------------------------------------------------------------------------------------------------------------- template inline auto Reverse(const QVector &container) -> QVector { if (container.isEmpty()) { return container; } QVector reversed(container.size()); vsizetype j = 0; for (vsizetype i = container.size() - 1; i >= 0; --i) { reversed.replace(j, container.at(i)); ++j; } return reversed; } template class C> //--------------------------------------------------------------------------------------------------------------------- inline auto Reverse(const C &container) -> C { return ConvertToList(Reverse(ConvertToVector(container))); } //--------------------------------------------------------------------------------------------------------------------- template ::value, T>::type * = nullptr> inline auto Reverse(const T &container) -> T { return Reverse(container); } //--------------------------------------------------------------------------------------------------------------------- template inline void AppendTo(Cont &container, const Input &input) { #if QT_VERSION >= QT_VERSION_CHECK(5, 5, 0) container.append(input); #else for (auto &item : input) { container.append(item); } #endif // QT_VERSION >= QT_VERSION_CHECK(5, 5, 0) } //--------------------------------------------------------------------------------------------------------------------- template inline auto SetIntersects(const QSet &set1, const QSet &set2) -> bool { #if QT_VERSION >= QT_VERSION_CHECK(5, 6, 0) return set1.intersects(set2); #else const bool otherIsBigger = set2.size() > set1.size(); const QSet &smallestSet = otherIsBigger ? set1 : set2; const QSet &biggestSet = otherIsBigger ? set2 : set1; typename QSet::const_iterator i = smallestSet.cbegin(); typename QSet::const_iterator e = smallestSet.cend(); while (i != e) { if (biggestSet.contains(*i)) { return true; } ++i; } return false; #endif } //--------------------------------------------------------------------------------------------------------------------- inline auto TextWidth(const QFontMetrics &fm, const QString &text, int len = -1) -> int { #if QT_VERSION >= QT_VERSION_CHECK(5, 11, 0) return fm.horizontalAdvance(text, len); #else return fm.width(text, len); #endif } //--------------------------------------------------------------------------------------------------------------------- inline auto TextWidthF(const QFontMetricsF &fm, const QString &text) -> qreal { #if QT_VERSION >= QT_VERSION_CHECK(5, 11, 0) return fm.horizontalAdvance(text); #else return fm.width(text); #endif } //--------------------------------------------------------------------------------------------------------------------- template inline auto Insert(QMap &map1, const QMap &map2) -> void { #if QT_VERSION >= QT_VERSION_CHECK(5, 15, 0) map1.insert(map2); #else auto i = map2.constBegin(); while (i != map2.constEnd()) { map1.insert(i.key(), i.value()); ++i; } #endif } //--------------------------------------------------------------------------------------------------------------------- inline auto VLocaleCharacter(const QString &character) -> QChar { Q_ASSERT(character.size() == 1); #if QT_VERSION >= QT_VERSION_CHECK(5, 10, 0) return character.front(); #else return character.at(0); #endif } //--------------------------------------------------------------------------------------------------------------------- inline auto VLocaleCharacter(const QChar &character) -> QChar { return character; } //--------------------------------------------------------------------------------------------------------------------- template inline auto DropEventPos(const T *event) -> QPoint { #if QT_VERSION >= QT_VERSION_CHECK(6, 0, 0) return event->position().toPoint(); #else return event->pos(); #endif } //--------------------------------------------------------------------------------------------------------------------- template inline auto QLibraryPath(T loc) -> QString { #if QT_VERSION >= QT_VERSION_CHECK(6, 0, 0) return QLibraryInfo::path(loc); #else return QLibraryInfo::location(loc); #endif } //--------------------------------------------------------------------------------------------------------------------- inline auto LineCenter(const QLineF &line) -> QPointF { #if QT_VERSION >= QT_VERSION_CHECK(5, 8, 0) return line.center(); #else return {0.5 * line.p1().x() + 0.5 * line.p2().x(), 0.5 * line.p1().y() + 0.5 * line.p2().y()}; #endif } //--------------------------------------------------------------------------------------------------------------------- template inline auto First(const T &list, N n) -> T { #if QT_VERSION >= QT_VERSION_CHECK(6, 0, 0) return list.first(n); #else Q_ASSERT(n <= list.size()); T result; result.reserve(n); std::copy(list.begin(), list.begin() + n, std::back_inserter(result)); return result; #endif } //--------------------------------------------------------------------------------------------------------------------- template inline auto Last(const T &list, N n) -> T { #if QT_VERSION >= QT_VERSION_CHECK(6, 0, 0) return list.last(n); #else Q_ASSERT(n <= list.size()); T result; result.reserve(n); std::copy(list.end() - n, list.end(), std::back_inserter(result)); return result; #endif } //--------------------------------------------------------------------------------------------------------------------- template inline auto Sliced(const T &list, N pos) -> T { #if QT_VERSION >= QT_VERSION_CHECK(6, 0, 0) return list.sliced(pos); #else Q_ASSERT(pos <= list.size()); T result; result.reserve(list.size() - pos); std::copy(list.begin() + pos, list.end(), std::back_inserter(result)); return result; #endif } //--------------------------------------------------------------------------------------------------------------------- template inline auto Sliced(const T &list, N pos, N n) -> T { #if QT_VERSION >= QT_VERSION_CHECK(6, 0, 0) return list.sliced(pos, n); #else Q_ASSERT(pos <= list.size()); Q_ASSERT(n >= 0); Q_ASSERT(pos + n <= list.size()); T result; result.reserve(n); std::copy(list.begin() + pos, list.begin() + pos + n, std::back_inserter(result)); return result; #endif } //--------------------------------------------------------------------------------------------------------------------- inline auto Back(const QString &str) -> QChar { #if QT_VERSION >= QT_VERSION_CHECK(5, 10, 0) return str.back(); #else return str.at(str.size() - 1); #endif } //--------------------------------------------------------------------------------------------------------------------- inline auto Front(const QString &str) -> QChar { #if QT_VERSION >= QT_VERSION_CHECK(5, 10, 0) return str.front(); #else return str.at(0); #endif } #endif // COMPATIBILITY_H