2020-01-16 16:34:08 +01:00
|
|
|
/************************************************************************
|
|
|
|
**
|
|
|
|
** @file compatibility.h
|
|
|
|
** @author Roman Telezhynskyi <dismine(at)gmail.com>
|
|
|
|
** @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
|
2020-01-31 07:00:05 +01:00
|
|
|
** <https://gitlab.com/smart-pattern/valentina> All Rights Reserved.
|
2020-01-16 16:34:08 +01:00
|
|
|
**
|
|
|
|
** 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 <http://www.gnu.org/licenses/>.
|
|
|
|
**
|
|
|
|
*************************************************************************/
|
|
|
|
#ifndef COMPATIBILITY_H
|
|
|
|
#define COMPATIBILITY_H
|
|
|
|
|
|
|
|
#include <QtGlobal>
|
|
|
|
#include <QStringList>
|
|
|
|
#include <QSet>
|
|
|
|
|
|
|
|
class QPointF;
|
|
|
|
|
|
|
|
// Contains helpful methods to hide version dependent code. It can be deprecation of method or change in API
|
|
|
|
|
|
|
|
//---------------------------------------------------------------------------------------------------------------------
|
|
|
|
template <typename T, template <typename> class Cont>
|
|
|
|
inline const T& ConstFirst (const Cont<T> &container)
|
|
|
|
{
|
|
|
|
#if QT_VERSION >= QT_VERSION_CHECK(5, 6, 0)
|
|
|
|
return container.constFirst();
|
|
|
|
#else
|
|
|
|
return container.first(); // clazy:exclude=detaching-temporary
|
|
|
|
#endif
|
|
|
|
}
|
|
|
|
|
|
|
|
//---------------------------------------------------------------------------------------------------------------------
|
|
|
|
template <typename T, typename C>
|
|
|
|
inline const T& ConstFirst (const C &container)
|
|
|
|
{
|
|
|
|
#if QT_VERSION >= QT_VERSION_CHECK(5, 6, 0)
|
|
|
|
return container.constFirst();
|
|
|
|
#else
|
|
|
|
return container.first(); // clazy:exclude=detaching-temporary
|
|
|
|
#endif
|
|
|
|
}
|
|
|
|
|
2020-01-17 14:14:00 +01:00
|
|
|
//---------------------------------------------------------------------------------------------------------------------
|
|
|
|
template <typename T, template <typename> class Cont>
|
|
|
|
inline const T& ConstLast (const Cont<T> &container)
|
|
|
|
{
|
|
|
|
#if QT_VERSION >= QT_VERSION_CHECK(5, 6, 0)
|
|
|
|
return container.constLast();
|
|
|
|
#else
|
|
|
|
return container.last(); // clazy:exclude=detaching-temporary
|
|
|
|
#endif
|
|
|
|
}
|
|
|
|
|
|
|
|
//---------------------------------------------------------------------------------------------------------------------
|
|
|
|
template <typename T, typename C>
|
|
|
|
inline const T& ConstLast (const C &container)
|
|
|
|
{
|
|
|
|
#if QT_VERSION >= QT_VERSION_CHECK(5, 6, 0)
|
|
|
|
return container.constLast();
|
|
|
|
#else
|
|
|
|
return container.last(); // clazy:exclude=detaching-temporary
|
|
|
|
#endif
|
|
|
|
}
|
|
|
|
|
2020-01-16 16:34:08 +01:00
|
|
|
//---------------------------------------------------------------------------------------------------------------------
|
|
|
|
template <typename T>
|
|
|
|
inline typename T::IntersectType Intersects(const T &l1, const T &l2, QPointF *intersectionPoint)
|
|
|
|
{
|
|
|
|
#if QT_VERSION >= QT_VERSION_CHECK(5, 14, 0)
|
|
|
|
return l1.intersects(l2, intersectionPoint);
|
|
|
|
#else
|
|
|
|
return l1.intersect(l2, intersectionPoint);
|
|
|
|
#endif
|
|
|
|
}
|
|
|
|
|
2020-01-16 18:09:23 +01:00
|
|
|
//---------------------------------------------------------------------------------------------------------------------
|
|
|
|
template <typename T, template <typename> class C>
|
|
|
|
inline QList<T> ConvertToList(const C<T> &container)
|
|
|
|
{
|
|
|
|
#if QT_VERSION >= QT_VERSION_CHECK(5, 14, 0)
|
2020-01-17 13:44:25 +01:00
|
|
|
return QList<T>(container.begin(), container.end());
|
2020-01-16 18:09:23 +01:00
|
|
|
#else
|
|
|
|
return container.toList();
|
|
|
|
#endif
|
|
|
|
}
|
|
|
|
|
|
|
|
//---------------------------------------------------------------------------------------------------------------------
|
|
|
|
template <typename T, template <typename> class C>
|
|
|
|
inline QSet<T> ConvertToSet(const C<T> &container)
|
|
|
|
{
|
|
|
|
#if QT_VERSION >= QT_VERSION_CHECK(5, 14, 0)
|
2020-01-17 13:44:25 +01:00
|
|
|
return QSet<T>(container.begin(), container.end());
|
2020-01-16 18:09:23 +01:00
|
|
|
#else
|
|
|
|
return container.toSet();
|
|
|
|
#endif
|
|
|
|
}
|
|
|
|
|
|
|
|
//---------------------------------------------------------------------------------------------------------------------
|
|
|
|
template <typename T, typename C>
|
|
|
|
inline QSet<T> ConvertToSet(const C &container)
|
|
|
|
{
|
|
|
|
#if QT_VERSION >= QT_VERSION_CHECK(5, 14, 0)
|
2020-01-17 13:44:25 +01:00
|
|
|
return QSet<T>(container.begin(), container.end());
|
2020-01-16 18:09:23 +01:00
|
|
|
#else
|
|
|
|
return container.toSet();
|
|
|
|
#endif
|
|
|
|
}
|
|
|
|
|
2020-01-16 18:20:55 +01:00
|
|
|
//---------------------------------------------------------------------------------------------------------------------
|
|
|
|
template <typename T, template <typename> class C>
|
|
|
|
inline QVector<T> ConvertToVector(const C<T> &container)
|
|
|
|
{
|
|
|
|
#if QT_VERSION >= QT_VERSION_CHECK(5, 14, 0)
|
|
|
|
return QVector<T>(container.begin(), container.end());
|
|
|
|
#else
|
|
|
|
return container.toVector();
|
|
|
|
#endif
|
|
|
|
}
|
|
|
|
|
2020-01-16 18:46:05 +01:00
|
|
|
//---------------------------------------------------------------------------------------------------------------------
|
|
|
|
template <typename T>
|
|
|
|
inline QVector<T> ConvertToVector(const QSet<T> &container)
|
|
|
|
{
|
|
|
|
#if QT_VERSION >= QT_VERSION_CHECK(5, 14, 0)
|
|
|
|
return QVector<T>(container.begin(), container.end());
|
|
|
|
#else
|
|
|
|
return container.toList().toVector();
|
|
|
|
#endif
|
|
|
|
}
|
|
|
|
|
2020-01-16 18:09:23 +01:00
|
|
|
//---------------------------------------------------------------------------------------------------------------------
|
|
|
|
template <typename T>
|
|
|
|
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
|
|
|
|
}
|
|
|
|
|
2020-01-17 14:14:00 +01:00
|
|
|
//---------------------------------------------------------------------------------------------------------------------
|
|
|
|
template <typename T>
|
|
|
|
inline void Move(T &vector, int from, int to)
|
|
|
|
{
|
2020-01-31 16:49:06 +01:00
|
|
|
QT_WARNING_PUSH
|
|
|
|
QT_WARNING_DISABLE_GCC("-Wstrict-overflow")
|
|
|
|
|
2020-01-17 14:14:00 +01:00
|
|
|
#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;
|
|
|
|
}
|
2020-01-20 13:03:57 +01:00
|
|
|
typename T::iterator b = vector.begin();
|
2020-01-17 14:14:00 +01:00
|
|
|
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)
|
2020-01-31 16:49:06 +01:00
|
|
|
|
|
|
|
QT_WARNING_POP
|
2020-01-17 14:14:00 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
//---------------------------------------------------------------------------------------------------------------------
|
|
|
|
template <typename Cont, typename Input>
|
|
|
|
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)
|
|
|
|
}
|
|
|
|
|
2020-03-24 16:31:32 +01:00
|
|
|
//---------------------------------------------------------------------------------------------------------------------
|
|
|
|
template <typename T>
|
|
|
|
inline bool SetIntersects(const QSet<T> &set1, const QSet<T> &set2)
|
|
|
|
{
|
|
|
|
#if QT_VERSION >= QT_VERSION_CHECK(5, 6, 0)
|
|
|
|
return set1.intersects(set2);
|
|
|
|
#else
|
|
|
|
const bool otherIsBigger = set2.size() > set1.size();
|
|
|
|
const QSet<T> &smallestSet = otherIsBigger ? set1 : set2;
|
|
|
|
const QSet<T> &biggestSet = otherIsBigger ? set2 : set1;
|
|
|
|
typename QSet<T>::const_iterator i = smallestSet.cbegin();
|
|
|
|
typename QSet<T>::const_iterator e = smallestSet.cend();
|
|
|
|
while (i != e)
|
|
|
|
{
|
|
|
|
if (biggestSet.contains(*i))
|
|
|
|
return true;
|
|
|
|
++i;
|
|
|
|
}
|
|
|
|
return false;
|
|
|
|
|
|
|
|
#endif
|
|
|
|
}
|
|
|
|
|
2020-01-16 16:34:08 +01:00
|
|
|
#endif // COMPATIBILITY_H
|