/************************************************************************ ** ** @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 #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) class QPointF; #if QT_VERSION < QT_VERSION_CHECK(6, 4, 0) namespace Qt { inline namespace Literals { inline namespace StringLiterals { QT_WARNING_PUSH QT_WARNING_DISABLE_CLANG("-Wreserved-identifier") //--------------------------------------------------------------------------------------------------------------------- Q_DECL_CONSTEVAL inline auto operator""_L1(char ch)Q_DECL_NOEXCEPT->QLatin1Char { return QLatin1Char(ch); } //--------------------------------------------------------------------------------------------------------------------- Q_DECL_CONSTEVAL inline auto operator""_L1(const char *str, size_t size)Q_DECL_NOEXCEPT->QLatin1String { return QLatin1String(str, static_cast(size)); } QT_WARNING_POP //--------------------------------------------------------------------------------------------------------------------- inline auto operator""_ba(const char *str, size_t size)Q_DECL_NOEXCEPT->QByteArray { return {str, static_cast(size)}; } //--------------------------------------------------------------------------------------------------------------------- inline auto operator""_s(const char16_t *str, size_t size)Q_DECL_NOEXCEPT->QString { return QString::fromUtf16(str, static_cast(size)); } } // namespace StringLiterals } // namespace Literals } // namespace Qt #endif // Contains helpful methods to hide version dependent code. It can be deprecation of method or change in API //--------------------------------------------------------------------------------------------------------------------- template class C> inline auto ConvertToList(const C &container) -> QList { return QList(std::begin(container), std::end(container)); } //--------------------------------------------------------------------------------------------------------------------- template class C> inline auto ConvertToStringList(const C &container) -> QStringList { return {ConvertToList(container)}; } //--------------------------------------------------------------------------------------------------------------------- template class C> inline auto ConvertToSet(const C &container) -> QSet { return QSet(std::begin(container), std::end(container)); } //--------------------------------------------------------------------------------------------------------------------- template inline auto ConvertToSet(const C &container) -> QSet { return QSet(std::begin(container), std::end(container)); } //--------------------------------------------------------------------------------------------------------------------- template class C> inline auto ConvertToVector(const C &container) -> QVector { return QVector(std::begin(container), std::end(container)); } //--------------------------------------------------------------------------------------------------------------------- template inline auto ConvertToVector(const QSet &container) -> QVector { return QVector(std::begin(container), std::end(container)); } //--------------------------------------------------------------------------------------------------------------------- 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 , T> * = nullptr> inline auto Reverse(const T &container) -> T { return Reverse(container); } //--------------------------------------------------------------------------------------------------------------------- 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 } //--------------------------------------------------------------------------------------------------------------------- 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 FontFromString(const QString &descrip) -> QFont { using namespace Qt::Literals::StringLiterals; QFont font; if (!descrip.isEmpty()) { // Qt 6's QFont::toString returns a value with 17 fields, e.g. // Ubuntu,11,-1,5,400,0,0,0,0,0,0,0,0,0,0,1 // Qt 5's QFont::fromString expects a value with 11 fields, e.g. // Ubuntu,10,-1,5,50,0,0,0,0,0 #if (QT_VERSION < QT_VERSION_CHECK(6, 0, 0)) const auto l = descrip.split(','_L1); // Qt5's QFont::fromString() isn't compatible with Qt6's QFont::toString(). // If we were built with Qt5, don't try to process a font preference that // was created by Qt6. if (l.count() <= 11) { font.fromString(descrip); } #else font.fromString(descrip); #endif } return font; } //--------------------------------------------------------------------------------------------------------------------- template inline auto TerritoryToString(const T &territory) -> QString { using namespace Qt::Literals::StringLiterals; // Since Qt 5.12 country names have spaces #if QT_VERSION < QT_VERSION_CHECK(6, 2, 0) return QLocale::countryToString(territory).remove(' '_L1); #else return QLocale::territoryToString(territory).remove(' '_L1); #endif } //--------------------------------------------------------------------------------------------------------------------- // NOLINTNEXTLINE(readability-inconsistent-declaration-parameter-name) template <> inline auto TerritoryToString(const QLocale &loc) -> QString { using namespace Qt::Literals::StringLiterals; // Since Qt 5.12 country names have spaces #if QT_VERSION < QT_VERSION_CHECK(6, 2, 0) return QLocale::countryToString(loc.country()).remove(' '_L1); #else return QLocale::territoryToString(loc.territory()).remove(' '_L1); #endif } //--------------------------------------------------------------------------------------------------------------------- template inline void SetTextAlignment(T *item, Qt::Alignment alignment) { #if QT_VERSION >= QT_VERSION_CHECK(6, 4, 0) item->setTextAlignment(alignment); #else item->setTextAlignment(static_cast(alignment)); #endif } //--------------------------------------------------------------------------------------------------------------------- inline void RemoveLast(QString &str) { #if QT_VERSION >= QT_VERSION_CHECK(6, 5, 0) str.removeLast(); #else if (!str.isEmpty()) { str.remove(str.size() - 1, 1); } #endif } #endif // COMPATIBILITY_H