Alternative approach to search intersection in layout.
--HG-- branch : develop
This commit is contained in:
parent
2565f88038
commit
aa1d30bdb8
|
@ -296,18 +296,6 @@ const QPointF &VContour::at(int i) const
|
||||||
return d->globalContour.at(i);
|
return d->globalContour.at(i);
|
||||||
}
|
}
|
||||||
|
|
||||||
//---------------------------------------------------------------------------------------------------------------------
|
|
||||||
QRectF VContour::BoundingRect() const
|
|
||||||
{
|
|
||||||
return d->m_boundingRect;
|
|
||||||
}
|
|
||||||
|
|
||||||
//---------------------------------------------------------------------------------------------------------------------
|
|
||||||
QPainterPath VContour::ContourPath() const
|
|
||||||
{
|
|
||||||
return d->m_contourPath;
|
|
||||||
}
|
|
||||||
|
|
||||||
//---------------------------------------------------------------------------------------------------------------------
|
//---------------------------------------------------------------------------------------------------------------------
|
||||||
void VContour::AppendWhole(QVector<QPointF> &contour, const VLayoutPiece &detail, int detJ) const
|
void VContour::AppendWhole(QVector<QPointF> &contour, const VLayoutPiece &detail, int detJ) const
|
||||||
{
|
{
|
||||||
|
@ -360,35 +348,6 @@ void VContour::ResetAttributes()
|
||||||
{
|
{
|
||||||
if (not d->globalContour.isEmpty())
|
if (not d->globalContour.isEmpty())
|
||||||
{
|
{
|
||||||
// Bounding rect
|
|
||||||
{
|
|
||||||
QVector<QPointF> points = d->globalContour;
|
|
||||||
if (points.isEmpty())
|
|
||||||
{
|
|
||||||
d->m_boundingRect = QRectF();
|
|
||||||
}
|
|
||||||
else
|
|
||||||
{
|
|
||||||
points.append(points.first());
|
|
||||||
d->m_boundingRect = QPolygonF(points).boundingRect();
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
{
|
|
||||||
// Countour path
|
|
||||||
QPainterPath path;
|
|
||||||
path.setFillRule(Qt::WindingFill);
|
|
||||||
const QVector<QPointF> points = d->globalContour;
|
|
||||||
path.moveTo(points.at(0));
|
|
||||||
for (qint32 i = 1; i < points.count(); ++i)
|
|
||||||
{
|
|
||||||
path.lineTo(points.at(i));
|
|
||||||
}
|
|
||||||
path.lineTo(points.at(0));
|
|
||||||
|
|
||||||
d->m_contourPath = path;
|
|
||||||
}
|
|
||||||
|
|
||||||
d->m_emptySheetEdgesCount = d->globalContour.count(); // Edges count
|
d->m_emptySheetEdgesCount = d->globalContour.count(); // Edges count
|
||||||
}
|
}
|
||||||
else
|
else
|
||||||
|
|
|
@ -89,10 +89,6 @@ public:
|
||||||
|
|
||||||
const QPointF & at(int i) const;
|
const QPointF & at(int i) const;
|
||||||
|
|
||||||
QRectF BoundingRect() const;
|
|
||||||
|
|
||||||
QPainterPath ContourPath() const;
|
|
||||||
|
|
||||||
private:
|
private:
|
||||||
QSharedDataPointer<VContourData> d;
|
QSharedDataPointer<VContourData> d;
|
||||||
|
|
||||||
|
|
|
@ -60,8 +60,6 @@ public:
|
||||||
paperWidth(contour.paperWidth),
|
paperWidth(contour.paperWidth),
|
||||||
shift(contour.shift),
|
shift(contour.shift),
|
||||||
layoutWidth(contour.layoutWidth),
|
layoutWidth(contour.layoutWidth),
|
||||||
m_boundingRect(contour.m_boundingRect),
|
|
||||||
m_contourPath(contour.m_contourPath),
|
|
||||||
m_emptySheetEdgesCount(contour.m_emptySheetEdgesCount)
|
m_emptySheetEdgesCount(contour.m_emptySheetEdgesCount)
|
||||||
{}
|
{}
|
||||||
|
|
||||||
|
@ -80,9 +78,6 @@ public:
|
||||||
|
|
||||||
qreal layoutWidth{0};
|
qreal layoutWidth{0};
|
||||||
|
|
||||||
QRectF m_boundingRect{};
|
|
||||||
QPainterPath m_contourPath{};
|
|
||||||
|
|
||||||
int m_emptySheetEdgesCount{0};
|
int m_emptySheetEdgesCount{0};
|
||||||
|
|
||||||
private:
|
private:
|
||||||
|
|
|
@ -65,6 +65,12 @@ struct VBestSquareResData
|
||||||
qreal depthPosition{INT_MAX};
|
qreal depthPosition{INT_MAX};
|
||||||
};
|
};
|
||||||
|
|
||||||
|
struct VCachedPositions
|
||||||
|
{
|
||||||
|
QRectF boundingRect{};
|
||||||
|
QPainterPath layoutAllowancePath{};
|
||||||
|
};
|
||||||
|
|
||||||
/* Warning! Debugging doesn't work stable in debug mode. If you need big allocation use release mode. Or disable
|
/* Warning! Debugging doesn't work stable in debug mode. If you need big allocation use release mode. Or disable
|
||||||
* Address Sanitizer. See page https://bitbucket.org/dismine/valentina/wiki/developers/Address_Sanitizer
|
* Address Sanitizer. See page https://bitbucket.org/dismine/valentina/wiki/developers/Address_Sanitizer
|
||||||
*/
|
*/
|
||||||
|
|
|
@ -252,6 +252,7 @@ bool VLayoutPaper::AddToSheet(const VLayoutPiece &detail, std::atomic_bool &stop
|
||||||
data.rotate = d->localRotate;
|
data.rotate = d->localRotate;
|
||||||
data.rotationNumber = d->localRotationNumber;
|
data.rotationNumber = d->localRotationNumber;
|
||||||
data.followGrainline = d->followGrainline;
|
data.followGrainline = d->followGrainline;
|
||||||
|
data.positionsCache = d->positionsCache;
|
||||||
|
|
||||||
auto *thread = new VPosition(data, &stop, d->saveLength);
|
auto *thread = new VPosition(data, &stop, d->saveLength);
|
||||||
//Info for debug
|
//Info for debug
|
||||||
|
@ -322,6 +323,12 @@ bool VLayoutPaper::SaveResult(const VBestSquare &bestResult, const VLayoutPiece
|
||||||
d->details.append(workDetail);
|
d->details.append(workDetail);
|
||||||
d->globalContour.SetContour(newGContour);
|
d->globalContour.SetContour(newGContour);
|
||||||
|
|
||||||
|
VCachedPositions positionChache;
|
||||||
|
QVector<QPointF> layoutPoints = workDetail.GetLayoutAllowancePoints();
|
||||||
|
positionChache.boundingRect = VLayoutPiece::BoundingRect(layoutPoints);
|
||||||
|
positionChache.layoutAllowancePath = VLayoutPiece::PainterPath(layoutPoints);
|
||||||
|
d->positionsCache.append(positionChache);
|
||||||
|
|
||||||
#ifdef LAYOUT_DEBUG
|
#ifdef LAYOUT_DEBUG
|
||||||
# ifdef SHOW_BEST
|
# ifdef SHOW_BEST
|
||||||
VPosition::DrawDebug(d->globalContour, workDetail, UINT_MAX, d->paperIndex, d->details.count(), d->details);
|
VPosition::DrawDebug(d->globalContour, workDetail, UINT_MAX, d->paperIndex, d->details.count(), d->details);
|
||||||
|
|
|
@ -54,6 +54,7 @@ public:
|
||||||
VLayoutPaperData(const VLayoutPaperData &paper)
|
VLayoutPaperData(const VLayoutPaperData &paper)
|
||||||
: QSharedData(paper),
|
: QSharedData(paper),
|
||||||
details(paper.details),
|
details(paper.details),
|
||||||
|
positionsCache(paper.positionsCache),
|
||||||
globalContour(paper.globalContour),
|
globalContour(paper.globalContour),
|
||||||
paperIndex(paper.paperIndex),
|
paperIndex(paper.paperIndex),
|
||||||
frame(paper.frame),
|
frame(paper.frame),
|
||||||
|
@ -71,6 +72,8 @@ public:
|
||||||
/** @brief details list of arranged details. */
|
/** @brief details list of arranged details. */
|
||||||
QVector<VLayoutPiece> details{};
|
QVector<VLayoutPiece> details{};
|
||||||
|
|
||||||
|
QVector<VCachedPositions> positionsCache{};
|
||||||
|
|
||||||
/** @brief globalContour list of global points contour. */
|
/** @brief globalContour list of global points contour. */
|
||||||
VContour globalContour{};
|
VContour globalContour{};
|
||||||
|
|
||||||
|
|
|
@ -681,26 +681,14 @@ int VLayoutPiece::LayoutEdgeByPoint(const QPointF &p1) const
|
||||||
//---------------------------------------------------------------------------------------------------------------------
|
//---------------------------------------------------------------------------------------------------------------------
|
||||||
QRectF VLayoutPiece::DetailBoundingRect() const
|
QRectF VLayoutPiece::DetailBoundingRect() const
|
||||||
{
|
{
|
||||||
QVector<QPointF> points;
|
return IsSeamAllowance() && not IsSeamAllowanceBuiltIn() ? BoundingRect(GetSeamAllowancePoints()) :
|
||||||
if (IsSeamAllowance() && not IsSeamAllowanceBuiltIn())
|
BoundingRect(GetContourPoints());
|
||||||
{
|
|
||||||
points = GetSeamAllowancePoints();
|
|
||||||
}
|
|
||||||
else
|
|
||||||
{
|
|
||||||
points = GetContourPoints();
|
|
||||||
}
|
|
||||||
|
|
||||||
points.append(points.first());
|
|
||||||
return QPolygonF(points).boundingRect();
|
|
||||||
}
|
}
|
||||||
|
|
||||||
//---------------------------------------------------------------------------------------------------------------------
|
//---------------------------------------------------------------------------------------------------------------------
|
||||||
QRectF VLayoutPiece::LayoutBoundingRect() const
|
QRectF VLayoutPiece::LayoutBoundingRect() const
|
||||||
{
|
{
|
||||||
QVector<QPointF> points = GetLayoutAllowancePoints();
|
return BoundingRect(GetLayoutAllowancePoints());
|
||||||
points.append(points.first());
|
|
||||||
return QPolygonF(points).boundingRect();
|
|
||||||
}
|
}
|
||||||
|
|
||||||
//---------------------------------------------------------------------------------------------------------------------
|
//---------------------------------------------------------------------------------------------------------------------
|
||||||
|
@ -832,16 +820,9 @@ QPainterPath VLayoutPiece::ContourPath() const
|
||||||
QPainterPath path;
|
QPainterPath path;
|
||||||
|
|
||||||
// contour
|
// contour
|
||||||
QVector<QPointF> points = GetContourPoints();
|
|
||||||
|
|
||||||
if (not IsHideMainPath() || not IsSeamAllowance() || IsSeamAllowanceBuiltIn())
|
if (not IsHideMainPath() || not IsSeamAllowance() || IsSeamAllowanceBuiltIn())
|
||||||
{
|
{
|
||||||
path.moveTo(points.at(0));
|
path = PainterPath(GetContourPoints());
|
||||||
for (qint32 i = 1; i < points.count(); ++i)
|
|
||||||
{
|
|
||||||
path.lineTo(points.at(i));
|
|
||||||
}
|
|
||||||
path.lineTo(points.at(0));
|
|
||||||
}
|
}
|
||||||
|
|
||||||
// seam allowance
|
// seam allowance
|
||||||
|
@ -885,11 +866,16 @@ QPainterPath VLayoutPiece::ContourPath() const
|
||||||
|
|
||||||
//---------------------------------------------------------------------------------------------------------------------
|
//---------------------------------------------------------------------------------------------------------------------
|
||||||
QPainterPath VLayoutPiece::LayoutAllowancePath() const
|
QPainterPath VLayoutPiece::LayoutAllowancePath() const
|
||||||
|
{
|
||||||
|
return PainterPath(GetLayoutAllowancePoints());
|
||||||
|
}
|
||||||
|
|
||||||
|
//---------------------------------------------------------------------------------------------------------------------
|
||||||
|
QPainterPath VLayoutPiece::PainterPath(const QVector<QPointF> &points)
|
||||||
{
|
{
|
||||||
QPainterPath path;
|
QPainterPath path;
|
||||||
path.setFillRule(Qt::WindingFill);
|
path.setFillRule(Qt::WindingFill);
|
||||||
|
|
||||||
const QVector<QPointF> points = GetLayoutAllowancePoints();
|
|
||||||
path.moveTo(points.at(0));
|
path.moveTo(points.at(0));
|
||||||
for (qint32 i = 1; i < points.count(); ++i)
|
for (qint32 i = 1; i < points.count(); ++i)
|
||||||
{
|
{
|
||||||
|
@ -957,6 +943,13 @@ qreal VLayoutPiece::BiggestEdge() const
|
||||||
return edge;
|
return edge;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
//---------------------------------------------------------------------------------------------------------------------
|
||||||
|
QRectF VLayoutPiece::BoundingRect(QVector<QPointF> points)
|
||||||
|
{
|
||||||
|
points.append(points.first());
|
||||||
|
return QPolygonF(points).boundingRect();
|
||||||
|
}
|
||||||
|
|
||||||
//---------------------------------------------------------------------------------------------------------------------
|
//---------------------------------------------------------------------------------------------------------------------
|
||||||
void VLayoutPiece::CreateLabelStrings(QGraphicsItem *parent, const QVector<QPointF> &labelShape,
|
void VLayoutPiece::CreateLabelStrings(QGraphicsItem *parent, const QVector<QPointF> &labelShape,
|
||||||
const VTextManager &tm, bool textAsPaths) const
|
const VTextManager &tm, bool textAsPaths) const
|
||||||
|
|
|
@ -135,11 +135,16 @@ public:
|
||||||
QRectF LayoutBoundingRect() const;
|
QRectF LayoutBoundingRect() const;
|
||||||
qreal Diagonal() const;
|
qreal Diagonal() const;
|
||||||
|
|
||||||
|
static QRectF BoundingRect(QVector<QPointF> points);
|
||||||
|
|
||||||
bool isNull() const;
|
bool isNull() const;
|
||||||
qint64 Square() const;
|
qint64 Square() const;
|
||||||
QPainterPath ContourPath() const;
|
|
||||||
|
|
||||||
|
QPainterPath ContourPath() const;
|
||||||
QPainterPath LayoutAllowancePath() const;
|
QPainterPath LayoutAllowancePath() const;
|
||||||
|
|
||||||
|
static QPainterPath PainterPath(const QVector<QPointF> &points);
|
||||||
|
|
||||||
Q_REQUIRED_RESULT QGraphicsItem *GetItem(bool textAsPaths) const;
|
Q_REQUIRED_RESULT QGraphicsItem *GetItem(bool textAsPaths) const;
|
||||||
|
|
||||||
bool IsLayoutAllowanceValid() const;
|
bool IsLayoutAllowanceValid() const;
|
||||||
|
|
|
@ -484,24 +484,33 @@ void VPosition::RotateOnAngle(qreal angle)
|
||||||
//---------------------------------------------------------------------------------------------------------------------
|
//---------------------------------------------------------------------------------------------------------------------
|
||||||
VPosition::CrossingType VPosition::Crossing(const VLayoutPiece &detail) const
|
VPosition::CrossingType VPosition::Crossing(const VLayoutPiece &detail) const
|
||||||
{
|
{
|
||||||
const QRectF gRect = m_data.gContour.BoundingRect();
|
if (m_data.positionsCache.isEmpty())
|
||||||
if (not gRect.intersects(detail.LayoutBoundingRect()) && not gRect.contains(detail.DetailBoundingRect()))
|
|
||||||
{
|
{
|
||||||
// This we can determine efficiently.
|
|
||||||
return CrossingType::NoIntersection;
|
return CrossingType::NoIntersection;
|
||||||
}
|
}
|
||||||
|
|
||||||
const QPainterPath gPath = m_data.gContour.ContourPath();
|
const QVector<QPointF> layoutPoints = detail.GetLayoutAllowancePoints();
|
||||||
CrossingType crossing = CrossingType::EdgeError;
|
const QRectF layoutBoundingRect = VLayoutPiece::BoundingRect(layoutPoints);
|
||||||
if (not gPath.contains(detail.ContourPath()) && not gPath.intersects(detail.LayoutAllowancePath()))
|
const QPainterPath layoutAllowancePath = VLayoutPiece::PainterPath(layoutPoints);
|
||||||
|
|
||||||
|
const QVector<QPointF> contourPoints = detail.IsSeamAllowance() && not detail.IsSeamAllowanceBuiltIn() ?
|
||||||
|
detail.GetSeamAllowancePoints() : detail.GetContourPoints();
|
||||||
|
const QRectF detailBoundingRect = VLayoutPiece::BoundingRect(contourPoints);
|
||||||
|
const QPainterPath contourPath = VLayoutPiece::PainterPath(contourPoints);
|
||||||
|
|
||||||
|
for(auto &position : m_data.positionsCache)
|
||||||
{
|
{
|
||||||
crossing = CrossingType::NoIntersection;
|
if (position.boundingRect.intersects(layoutBoundingRect) || position.boundingRect.contains(detailBoundingRect))
|
||||||
}
|
|
||||||
else
|
|
||||||
{
|
{
|
||||||
crossing = CrossingType::Intersection;
|
if (position.layoutAllowancePath.contains(contourPath) ||
|
||||||
|
position.layoutAllowancePath.intersects(layoutAllowancePath))
|
||||||
|
{
|
||||||
|
return CrossingType::Intersection;
|
||||||
}
|
}
|
||||||
return crossing;
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return CrossingType::NoIntersection;
|
||||||
}
|
}
|
||||||
|
|
||||||
//---------------------------------------------------------------------------------------------------------------------
|
//---------------------------------------------------------------------------------------------------------------------
|
||||||
|
|
|
@ -49,6 +49,7 @@ struct VPositionData
|
||||||
bool rotate{false};
|
bool rotate{false};
|
||||||
int rotationNumber{0};
|
int rotationNumber{0};
|
||||||
bool followGrainline{false};
|
bool followGrainline{false};
|
||||||
|
QVector<VCachedPositions> positionsCache{};
|
||||||
};
|
};
|
||||||
|
|
||||||
class VPosition : public QRunnable
|
class VPosition : public QRunnable
|
||||||
|
|
Loading…
Reference in New Issue
Block a user