Add clear, reserve and capacity methods to QPainterPath

This allows anticipating and reusing internal allocations of
QPainterPathElements instead of using the common `m_myPath = QPainterPath{}` pattern.

[ChangeLog][QtGui][QPainterPath] Added clear(), reserve(), capacity().
clear() removes allocated QPainterPath elements but preserves allocated memory, which can be
useful for application with complex paths that are often recreated. reserve() and capacity()
follow QVector semantics.

Change-Id: I763461e2a421feda9053d3eb512af2fcf07ade2b
Reviewed-by: Allan Sandfeld Jensen <allan.jensen@qt.io>
This commit is contained in:
Jean-Michaël Celerier 2017-07-09 19:27:35 +02:00 committed by Jean-Michaël Celerier
parent 4e6a42cdd0
commit e6a7b61d27
4 changed files with 129 additions and 6 deletions

View File

@ -628,6 +628,55 @@ QPainterPath::~QPainterPath()
{
}
/*!
Clears the path elements stored.
This allows the path to reuse previous memory allocations.
\sa reserve(), capacity()
\since 5.13
*/
void QPainterPath::clear()
{
if (!d_ptr)
return;
detach();
d_func()->clear();
}
/*!
Reserves a given amount of elements in QPainterPath's internal memory.
Attempts to allocate memory for at least \a size elements.
\sa clear(), capacity(), QVector::reserve()
\since 5.13
*/
void QPainterPath::reserve(int size)
{
Q_D(QPainterPath);
if ((!d && size > 0) || (d && d->elements.capacity() < size)) {
detach();
d->elements.reserve(size);
}
}
/*!
Returns the number of elements allocated by the QPainterPath.
\sa clear(), reserve()
\since 5.13
*/
int QPainterPath::capacity() const
{
Q_D(QPainterPath);
if (d)
return d->elements.capacity();
return 0;
}
/*!
Closes the current subpath by drawing a line to the beginning of
the subpath, automatically starting a new path. The current point
@ -2271,13 +2320,19 @@ static inline bool epsilonCompare(const QPointF &a, const QPointF &b, const QSiz
bool QPainterPath::operator==(const QPainterPath &path) const
{
QPainterPathData *d = reinterpret_cast<QPainterPathData *>(d_func());
if (path.d_func() == d)
QPainterPathData *other_d = path.d_func();
if (other_d == d)
return true;
else if (!d || !path.d_func())
else if (!d || !other_d) {
if (!d && other_d->elements.empty() && other_d->fillRule == Qt::OddEvenFill)
return true;
if (!other_d && d && d->elements.empty() && d->fillRule == Qt::OddEvenFill)
return true;
return false;
else if (d->fillRule != path.d_func()->fillRule)
}
else if (d->fillRule != other_d->fillRule)
return false;
else if (d->elements.size() != path.d_func()->elements.size())
else if (d->elements.size() != other_d->elements.size())
return false;
const qreal qt_epsilon = sizeof(qreal) == sizeof(double) ? 1e-12 : qreal(1e-5);
@ -2287,8 +2342,8 @@ bool QPainterPath::operator==(const QPainterPath &path) const
epsilon.rheight() *= qt_epsilon;
for (int i = 0; i < d->elements.size(); ++i)
if (d->elements.at(i).type != path.d_func()->elements.at(i).type
|| !epsilonCompare(d->elements.at(i), path.d_func()->elements.at(i), epsilon))
if (d->elements.at(i).type != other_d->elements.at(i).type
|| !epsilonCompare(d->elements.at(i), other_d->elements.at(i), epsilon))
return false;
return true;

View File

@ -97,8 +97,13 @@ public:
{ qSwap(d_ptr, other.d_ptr); return *this; }
#endif
~QPainterPath();
inline void swap(QPainterPath &other) Q_DECL_NOEXCEPT { d_ptr.swap(other.d_ptr); }
void clear();
void reserve(int size);
int capacity() const;
void closeSubpath();
void moveTo(const QPointF &p);

View File

@ -194,6 +194,7 @@ public:
inline bool isClosed() const;
inline void close();
inline void maybeMoveTo();
inline void clear();
const QVectorPath &vectorPath() {
if (!pathConverter)
@ -290,6 +291,25 @@ inline void QPainterPathData::maybeMoveTo()
}
}
inline void QPainterPathData::clear()
{
Q_ASSERT(ref.load() == 1);
elements.clear();
cStart = 0;
bounds = {};
controlBounds = {};
require_moveTo = false;
dirtyBounds = false;
dirtyControlBounds = false;
convex = false;
delete pathConverter;
pathConverter = nullptr;
}
#define KAPPA qreal(0.5522847498)

View File

@ -43,6 +43,8 @@ public slots:
void cleanupTestCase();
private slots:
void getSetCheck();
void clear();
void reserveAndCapacity();
void swap();
void contains_QPointF_data();
@ -148,6 +150,47 @@ void tst_QPainterPath::swap()
QCOMPARE(p2.boundingRect().toRect(), QRect( 0, 0,10,10));
}
void tst_QPainterPath::clear()
{
QPainterPath p1;
QPainterPath p2;
p1.clear();
QCOMPARE(p1, p2);
p1.addRect(0, 0, 10, 10);
p1.clear();
QCOMPARE(p1, p2);
QCOMPARE(p1.fillRule(), Qt::OddEvenFill);
p1.setFillRule(Qt::WindingFill);
p1.clear();
QCOMPARE(p1.fillRule(), Qt::WindingFill);
}
void tst_QPainterPath::reserveAndCapacity()
{
QPainterPath p;
QVERIFY(p.capacity() == 0);
p.addRect(0, 0, 10, 10);
QVERIFY(p.capacity() > 0);
p.clear();
QVERIFY(p.capacity() > 0);
p = QPainterPath{};
QVERIFY(p.capacity() == 0);
p.moveTo(100, 100);
QVERIFY(p.capacity() > 1);
p.reserve(1000);
QVERIFY(p.capacity() >= 1000);
p.reserve(0);
QVERIFY(p.capacity() >= 1000);
}
Q_DECLARE_METATYPE(QPainterPath)
void tst_QPainterPath::currentPosition()