1059 lines
34 KiB
C++
1059 lines
34 KiB
C++
/************************************************************************
|
|
**
|
|
** @file vpsheet.cpp
|
|
** @author Ronan Le Tiec
|
|
** @date 23 5, 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
|
|
** <https://gitlab.com/smart-pattern/valentina> 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 <http://www.gnu.org/licenses/>.
|
|
**
|
|
*************************************************************************/
|
|
#include "vpsheet.h"
|
|
|
|
#include "../scene/vpgraphicspiece.h"
|
|
#include "../scene/vpgraphicspiececontrols.h"
|
|
#include "../scene/vpgraphicssheet.h"
|
|
#include "../scene/vpgraphicstilegrid.h"
|
|
#include "../vpapplication.h"
|
|
#include "../vwidgets/vmaingraphicsscene.h"
|
|
#include "theme/vscenestylesheet.h"
|
|
#include "theme/vstylesheetstyle.h"
|
|
#include "vplayout.h"
|
|
#include "vppiece.h"
|
|
|
|
#include <QtConcurrent>
|
|
|
|
namespace
|
|
{
|
|
struct VSheetPiece
|
|
{
|
|
QString m_id{};
|
|
bool m_showFullPiece{false};
|
|
QLineF m_seamMirrorLine{};
|
|
QLineF m_seamAllowanceMirrorLine{};
|
|
QVector<QPointF> m_externalContourPoints{};
|
|
};
|
|
|
|
struct VPiecesValidationData
|
|
{
|
|
bool m_warnPiecesOutOfBound{false};
|
|
bool m_warnSuperpositionOfPieces{false};
|
|
bool m_warnPieceGapePosition{false};
|
|
bool m_cutOnFold{false};
|
|
QRectF m_sheetRect{};
|
|
qreal m_pieceGap{0};
|
|
QVector<VSheetPiece> m_pieces{};
|
|
};
|
|
|
|
//---------------------------------------------------------------------------------------------------------------------
|
|
void ValidatePiecesOutOfBound(const VPiecesValidationData &data, QHash<QString, VPiecePositionValidity> &validations)
|
|
{
|
|
|
|
for (const auto &piece : data.m_pieces)
|
|
{
|
|
VPiecePositionValidity validation = validations.value(piece.m_id);
|
|
|
|
if (data.m_cutOnFold && not piece.m_showFullPiece && !piece.m_seamMirrorLine.isNull())
|
|
{
|
|
QLineF const foldLine = data.m_sheetRect.width() >= data.m_sheetRect.height()
|
|
? QLineF(data.m_sheetRect.topLeft(), data.m_sheetRect.topRight())
|
|
: QLineF(data.m_sheetRect.topRight(), data.m_sheetRect.bottomRight());
|
|
|
|
validation.m_outOfBound =
|
|
not VGObject::IsLineSegmentOnLineSegment(foldLine, piece.m_seamAllowanceMirrorLine, MmToPixel(0.5));
|
|
}
|
|
else
|
|
{
|
|
QRectF const pieceRect = VLayoutPiece::BoundingRect(piece.m_externalContourPoints);
|
|
validation.m_outOfBound = not data.m_sheetRect.contains(pieceRect);
|
|
}
|
|
|
|
validations.insert(piece.m_id, validation);
|
|
}
|
|
}
|
|
|
|
//---------------------------------------------------------------------------------------------------------------------
|
|
void ValidateSuperpositionOfPieces(const VPiecesValidationData &data,
|
|
QHash<QString, VPiecePositionValidity> &validations)
|
|
{
|
|
QSet<QString> invalidPieces;
|
|
invalidPieces.reserve(data.m_pieces.size());
|
|
|
|
for (const auto &piece : data.m_pieces)
|
|
{
|
|
if (invalidPieces.contains(piece.m_id))
|
|
{
|
|
continue;
|
|
}
|
|
|
|
bool hasSuperposition = false;
|
|
VSheetPiece invalidPiece;
|
|
|
|
for (const auto &p : data.m_pieces)
|
|
{
|
|
if (piece.m_id == p.m_id)
|
|
{
|
|
continue;
|
|
}
|
|
|
|
if (VPPiece::PathsSuperposition(piece.m_externalContourPoints, p.m_externalContourPoints))
|
|
{
|
|
hasSuperposition = true;
|
|
invalidPiece = p;
|
|
break;
|
|
}
|
|
}
|
|
|
|
auto UpdateValidity = [&validations, hasSuperposition, &invalidPieces](const QString &id)
|
|
{
|
|
VPiecePositionValidity validation = validations.value(id);
|
|
validation.m_superposition = hasSuperposition;
|
|
validations.insert(id, validation);
|
|
if (hasSuperposition)
|
|
{
|
|
invalidPieces.insert(id);
|
|
}
|
|
};
|
|
|
|
UpdateValidity(piece.m_id);
|
|
|
|
if (!invalidPiece.m_id.isEmpty())
|
|
{
|
|
UpdateValidity(invalidPiece.m_id);
|
|
}
|
|
}
|
|
}
|
|
|
|
//---------------------------------------------------------------------------------------------------------------------
|
|
void ValidatePiecesGapePosition(const VPiecesValidationData &data, QHash<QString, VPiecePositionValidity> &validations)
|
|
{
|
|
if (data.m_pieceGap <= 0)
|
|
{
|
|
return;
|
|
}
|
|
|
|
QSet<QString> invalidPieces;
|
|
invalidPieces.reserve(data.m_pieces.size());
|
|
|
|
for (const auto &piece : data.m_pieces)
|
|
{
|
|
if (invalidPieces.contains(piece.m_id))
|
|
{
|
|
continue;
|
|
}
|
|
|
|
QRectF const path1Rect = VLayoutPiece::BoundingRect(piece.m_externalContourPoints)
|
|
.adjusted(-data.m_pieceGap, -data.m_pieceGap, data.m_pieceGap, data.m_pieceGap);
|
|
QVector<QPointF> const path1 = VPPiece::PrepareStickyPath(piece.m_externalContourPoints);
|
|
bool hasInvalidPieceGapPosition = false;
|
|
VSheetPiece invalidPiece;
|
|
|
|
for (const auto &p : data.m_pieces)
|
|
{
|
|
if (piece.m_id == p.m_id)
|
|
{
|
|
continue;
|
|
}
|
|
|
|
QRectF const path2Rect =
|
|
VLayoutPiece::BoundingRect(p.m_externalContourPoints)
|
|
.adjusted(-data.m_pieceGap, -data.m_pieceGap, data.m_pieceGap, data.m_pieceGap);
|
|
|
|
if (!path1Rect.intersects(path2Rect) && !path1Rect.contains(path2Rect) && !path2Rect.contains(path1Rect))
|
|
{
|
|
continue;
|
|
}
|
|
|
|
QVector<QPointF> const path2 = VPPiece::PrepareStickyPath(p.m_externalContourPoints);
|
|
|
|
QLineF const distance = VPPiece::ClosestDistance(path1, path2);
|
|
|
|
if (distance.length() < data.m_pieceGap - accuracyPointOnLine)
|
|
{
|
|
hasInvalidPieceGapPosition = true;
|
|
invalidPiece = p;
|
|
break;
|
|
}
|
|
}
|
|
|
|
auto UpdateValidity = [&validations, hasInvalidPieceGapPosition, &invalidPieces](const QString &id)
|
|
{
|
|
VPiecePositionValidity validation = validations.value(id);
|
|
validation.m_gap = hasInvalidPieceGapPosition;
|
|
validations.insert(id, validation);
|
|
if (hasInvalidPieceGapPosition)
|
|
{
|
|
invalidPieces.insert(id);
|
|
}
|
|
};
|
|
|
|
UpdateValidity(piece.m_id);
|
|
|
|
if (!invalidPiece.m_id.isEmpty())
|
|
{
|
|
UpdateValidity(invalidPiece.m_id);
|
|
}
|
|
}
|
|
}
|
|
|
|
//---------------------------------------------------------------------------------------------------------------------
|
|
auto ValidatePiecesPositions(const VPiecesValidationData &data) -> QHash<QString, VPiecePositionValidity>
|
|
{
|
|
QHash<QString, VPiecePositionValidity> validations;
|
|
|
|
if (data.m_warnPiecesOutOfBound)
|
|
{
|
|
ValidatePiecesOutOfBound(data, validations);
|
|
}
|
|
|
|
if (data.m_warnSuperpositionOfPieces)
|
|
{
|
|
ValidateSuperpositionOfPieces(data, validations);
|
|
}
|
|
|
|
if (data.m_warnPieceGapePosition)
|
|
{
|
|
ValidatePiecesGapePosition(data, validations);
|
|
}
|
|
|
|
return validations;
|
|
}
|
|
} // namespace
|
|
|
|
// VPSheetSceneData
|
|
//---------------------------------------------------------------------------------------------------------------------
|
|
VPSheetSceneData::VPSheetSceneData(const VPLayoutPtr &layout, const QUuid &sheetUuid)
|
|
: m_layout(layout),
|
|
m_scene(QSharedPointer<VMainGraphicsScene>::create()),
|
|
m_sheetUuid(sheetUuid)
|
|
{
|
|
SCASSERT(not layout.isNull())
|
|
|
|
m_graphicsSheet = new VPGraphicsSheet(layout);
|
|
m_graphicsSheet->setPos(0, 0);
|
|
m_scene->addItem(m_graphicsSheet);
|
|
|
|
m_graphicsTileGrid = new VPGraphicsTileGrid(layout, m_sheetUuid);
|
|
m_scene->addItem(m_graphicsTileGrid);
|
|
|
|
m_rotationControls = new VPGraphicsPieceControls(layout);
|
|
m_scene->addItem(m_rotationControls);
|
|
|
|
m_rotationOrigin = new VPGraphicsTransformationOrigin(layout);
|
|
m_rotationOrigin->setVisible(false);
|
|
m_scene->addItem(m_rotationOrigin);
|
|
|
|
QObject::connect(m_rotationControls, &VPGraphicsPieceControls::ShowOrigin, m_rotationOrigin,
|
|
&VPGraphicsTransformationOrigin::on_ShowOrigin);
|
|
QObject::connect(m_rotationControls, &VPGraphicsPieceControls::TransformationOriginChanged, m_rotationOrigin,
|
|
&VPGraphicsTransformationOrigin::SetTransformationOrigin);
|
|
}
|
|
|
|
//---------------------------------------------------------------------------------------------------------------------
|
|
VPSheetSceneData::~VPSheetSceneData() = default;
|
|
|
|
//---------------------------------------------------------------------------------------------------------------------
|
|
auto VPSheetSceneData::Scene() const -> QSharedPointer<VMainGraphicsScene>
|
|
{
|
|
return m_scene;
|
|
}
|
|
|
|
//---------------------------------------------------------------------------------------------------------------------
|
|
void VPSheetSceneData::RefreshLayout()
|
|
{
|
|
m_graphicsSheet->update();
|
|
|
|
m_graphicsTileGrid->update();
|
|
|
|
m_scene->update();
|
|
}
|
|
|
|
//---------------------------------------------------------------------------------------------------------------------
|
|
void VPSheetSceneData::RefreshPieces()
|
|
{
|
|
qDeleteAll(m_graphicsPieces);
|
|
m_graphicsPieces.clear();
|
|
|
|
VPLayoutPtr const layout = m_layout.toStrongRef();
|
|
if (layout.isNull())
|
|
{
|
|
return;
|
|
}
|
|
|
|
VPSheetPtr const sheet = layout->GetSheet(m_sheetUuid);
|
|
if (sheet.isNull())
|
|
{
|
|
return;
|
|
}
|
|
|
|
QList<VPPiecePtr> const pieces = sheet->GetPieces();
|
|
m_graphicsPieces.reserve(pieces.size());
|
|
|
|
for (const auto &piece : pieces)
|
|
{
|
|
if (piece.isNull())
|
|
{
|
|
continue;
|
|
}
|
|
|
|
auto *graphicsPiece = new VPGraphicsPiece(piece);
|
|
m_graphicsPieces.append(graphicsPiece);
|
|
m_scene->addItem(graphicsPiece);
|
|
|
|
// Restore selection state
|
|
if (piece->IsSelected())
|
|
{
|
|
graphicsPiece->setSelected(piece->IsSelected());
|
|
}
|
|
|
|
ConnectPiece(graphicsPiece);
|
|
}
|
|
}
|
|
|
|
//---------------------------------------------------------------------------------------------------------------------
|
|
void VPSheetSceneData::PrepareForExport()
|
|
{
|
|
VStylesheetStyle::SetExportColorScheme(ExportColorScheme::BackAndWhite);
|
|
VSceneStylesheet::ResetStyles();
|
|
|
|
m_graphicsSheet->SetShowBorder(false);
|
|
m_graphicsSheet->SetShowMargin(false);
|
|
|
|
m_rotationControls->setVisible(false);
|
|
m_rotationOrigin->setVisible(false);
|
|
|
|
if (VPLayoutPtr const layout = m_layout.toStrongRef(); not layout.isNull())
|
|
{
|
|
m_showGridTmp = layout->LayoutSettings().GetShowGrid();
|
|
layout->LayoutSettings().SetShowGrid(false);
|
|
|
|
m_showTilesTmp = layout->LayoutSettings().GetShowTiles();
|
|
layout->LayoutSettings().SetShowTiles(false);
|
|
|
|
VPSheetPtr const sheet = layout->GetSheet(m_sheetUuid);
|
|
m_slectedPiecesTmp = sheet->GetSelectedPieces();
|
|
|
|
for (const auto &piece : qAsConst(m_slectedPiecesTmp))
|
|
{
|
|
if (not piece.isNull())
|
|
{
|
|
piece->SetSelected(false);
|
|
}
|
|
}
|
|
|
|
m_outOfBoundTmp = layout->LayoutSettings().GetWarningPiecesOutOfBound();
|
|
layout->LayoutSettings().SetWarningPiecesOutOfBound(false);
|
|
|
|
m_pieceSuperpositionTmp = layout->LayoutSettings().GetWarningSuperpositionOfPieces();
|
|
layout->LayoutSettings().SetWarningSuperpositionOfPieces(false);
|
|
|
|
m_pieceGapePositionTmp = layout->LayoutSettings().GetWarningPieceGapePosition();
|
|
layout->LayoutSettings().SetWarningPieceGapePosition(false);
|
|
}
|
|
|
|
RefreshLayout();
|
|
RefreshPieces();
|
|
}
|
|
|
|
//---------------------------------------------------------------------------------------------------------------------
|
|
void VPSheetSceneData::CleanAfterExport()
|
|
{
|
|
VStylesheetStyle::SetExportColorScheme(ExportColorScheme::Default);
|
|
VSceneStylesheet::ResetStyles();
|
|
|
|
m_graphicsSheet->SetShowBorder(true);
|
|
m_graphicsSheet->SetShowMargin(true);
|
|
|
|
m_rotationControls->setVisible(true);
|
|
|
|
if (VPLayoutPtr const layout = m_layout.toStrongRef(); not layout.isNull())
|
|
{
|
|
layout->LayoutSettings().SetShowGrid(m_showGridTmp);
|
|
layout->LayoutSettings().SetShowTiles(m_showTilesTmp);
|
|
|
|
for (const auto &piece : qAsConst(m_slectedPiecesTmp))
|
|
{
|
|
if (not piece.isNull())
|
|
{
|
|
piece->SetSelected(true);
|
|
emit layout->PieceSelectionChanged(piece);
|
|
}
|
|
}
|
|
|
|
layout->LayoutSettings().SetWarningPiecesOutOfBound(m_outOfBoundTmp);
|
|
layout->LayoutSettings().SetWarningSuperpositionOfPieces(m_pieceSuperpositionTmp);
|
|
layout->LayoutSettings().SetWarningPieceGapePosition(m_pieceGapePositionTmp);
|
|
}
|
|
|
|
RefreshLayout();
|
|
RefreshPieces();
|
|
}
|
|
|
|
//---------------------------------------------------------------------------------------------------------------------
|
|
auto VPSheetSceneData::GraphicsPieces() const -> const QList<VPGraphicsPiece *> &
|
|
{
|
|
return m_graphicsPieces;
|
|
}
|
|
|
|
//---------------------------------------------------------------------------------------------------------------------
|
|
auto VPSheetSceneData::GraphicsPiecesAsItems() const -> QList<QGraphicsItem *>
|
|
{
|
|
QList<QGraphicsItem *> items;
|
|
items.reserve(m_graphicsPieces.size());
|
|
|
|
for (auto *item : m_graphicsPieces)
|
|
{
|
|
items.append(item);
|
|
}
|
|
|
|
return items;
|
|
}
|
|
|
|
//---------------------------------------------------------------------------------------------------------------------
|
|
auto VPSheetSceneData::RotationControls() const -> VPGraphicsPieceControls *
|
|
{
|
|
return m_rotationControls;
|
|
}
|
|
|
|
//---------------------------------------------------------------------------------------------------------------------
|
|
auto VPSheetSceneData::ScenePiece(const VPPiecePtr &piece) const -> VPGraphicsPiece *
|
|
{
|
|
if (auto _graphicsPiece =
|
|
std::find_if(m_graphicsPieces.begin(), m_graphicsPieces.end(),
|
|
[piece](VPGraphicsPiece *graphicPiece) { return graphicPiece->GetPiece() == piece; });
|
|
_graphicsPiece != m_graphicsPieces.end())
|
|
{
|
|
return *_graphicsPiece;
|
|
}
|
|
|
|
return nullptr;
|
|
}
|
|
|
|
//---------------------------------------------------------------------------------------------------------------------
|
|
void VPSheetSceneData::RemovePiece(VPGraphicsPiece *piece)
|
|
{
|
|
m_graphicsPieces.removeAll(piece);
|
|
}
|
|
|
|
//---------------------------------------------------------------------------------------------------------------------
|
|
void VPSheetSceneData::AddPiece(VPGraphicsPiece *piece)
|
|
{
|
|
m_graphicsPieces.append(piece);
|
|
ConnectPiece(piece);
|
|
}
|
|
|
|
//---------------------------------------------------------------------------------------------------------------------
|
|
void VPSheetSceneData::SetTextAsPaths(bool textAsPaths) const
|
|
{
|
|
for (auto *piece : m_graphicsPieces)
|
|
{
|
|
if (piece != nullptr)
|
|
{
|
|
piece->SetTextAsPaths(textAsPaths);
|
|
}
|
|
}
|
|
}
|
|
|
|
//---------------------------------------------------------------------------------------------------------------------
|
|
void VPSheetSceneData::PrepareTilesScheme()
|
|
{
|
|
if (VPLayoutPtr const layout = m_layout.toStrongRef(); not layout.isNull())
|
|
{
|
|
m_showTilesSchemeTmp = layout->LayoutSettings().GetShowTiles();
|
|
layout->LayoutSettings().SetShowTiles(true);
|
|
|
|
m_showTilesWatermarkSchemeTmp = layout->LayoutSettings().GetShowWatermark();
|
|
layout->LayoutSettings().SetShowWatermark(false);
|
|
}
|
|
|
|
RefreshLayout();
|
|
}
|
|
|
|
//---------------------------------------------------------------------------------------------------------------------
|
|
void VPSheetSceneData::ClearTilesScheme()
|
|
{
|
|
if (VPLayoutPtr const layout = m_layout.toStrongRef(); not layout.isNull())
|
|
{
|
|
layout->LayoutSettings().SetShowTiles(m_showTilesSchemeTmp);
|
|
layout->LayoutSettings().SetShowWatermark(m_showTilesWatermarkSchemeTmp);
|
|
}
|
|
|
|
RefreshLayout();
|
|
}
|
|
|
|
//---------------------------------------------------------------------------------------------------------------------
|
|
void VPSheetSceneData::RefreshSheetSize()
|
|
{
|
|
if (m_graphicsSheet != nullptr)
|
|
{
|
|
m_graphicsSheet->RefreshBoundingRect();
|
|
}
|
|
}
|
|
|
|
//---------------------------------------------------------------------------------------------------------------------
|
|
void VPSheetSceneData::ConnectPiece(VPGraphicsPiece *piece) const
|
|
{
|
|
SCASSERT(piece != nullptr)
|
|
|
|
VPLayoutPtr const layout = m_layout.toStrongRef();
|
|
if (layout.isNull())
|
|
{
|
|
return;
|
|
}
|
|
|
|
QObject::connect(layout.data(), &VPLayout::PieceTransformationChanged, piece, &VPGraphicsPiece::on_RefreshPiece);
|
|
QObject::connect(layout.data(), &VPLayout::PieceZValueChanged, piece, &VPGraphicsPiece::PieceZValueChanged);
|
|
QObject::connect(layout.data(), &VPLayout::PieceSelectionChanged, m_rotationControls,
|
|
&VPGraphicsPieceControls::on_UpdateControls, Qt::UniqueConnection);
|
|
QObject::connect(layout.data(), &VPLayout::PiecePositionValidityChanged, piece, &VPGraphicsPiece::on_RefreshPiece);
|
|
QObject::connect(piece, &VPGraphicsPiece::PieceTransformationChanged, m_rotationControls,
|
|
&VPGraphicsPieceControls::on_UpdateControls);
|
|
QObject::connect(piece, &VPGraphicsPiece::HideTransformationHandles, m_rotationControls,
|
|
&VPGraphicsPieceControls::on_HideHandles);
|
|
QObject::connect(piece, &VPGraphicsPiece::HideTransformationHandles, m_rotationOrigin,
|
|
&VPGraphicsTransformationOrigin::on_HideHandles);
|
|
QObject::connect(layout.data(), &VPLayout::BoundaryTogetherWithNotchesChanged, piece,
|
|
&VPGraphicsPiece::on_RefreshPiece);
|
|
}
|
|
|
|
// VPSheet
|
|
//---------------------------------------------------------------------------------------------------------------------
|
|
VPSheet::VPSheet(const VPLayoutPtr &layout, QObject *parent)
|
|
: QObject(parent),
|
|
m_layout(layout),
|
|
m_sceneData(QSharedPointer<VPSheetSceneData>::create(layout, Uuid())),
|
|
m_validityWatcher(new QFutureWatcher<QHash<QString, VPiecePositionValidity>>(this))
|
|
{
|
|
SCASSERT(not layout.isNull())
|
|
|
|
VPSettings *settings = VPApplication::VApp()->PuzzleSettings();
|
|
SetIgnoreMargins(settings->GetLayoutSheetIgnoreMargins());
|
|
SetSheetMargins(settings->GetLayoutSheetMargins());
|
|
SetSheetSize(QSizeF(settings->GetLayoutSheetPaperWidth(), settings->GetLayoutSheetPaperHeight()));
|
|
|
|
connect(qApp, &QCoreApplication::aboutToQuit, m_validityWatcher,
|
|
[this]()
|
|
{
|
|
m_validityWatcher->cancel();
|
|
m_validityWatcher->waitForFinished();
|
|
});
|
|
connect(m_validityWatcher, &QFutureWatcher<QHash<QString, VPiecePositionValidity>>::finished, this,
|
|
&VPSheet::UpdatePiecesValidity);
|
|
}
|
|
|
|
//---------------------------------------------------------------------------------------------------------------------
|
|
VPSheet::~VPSheet()
|
|
{
|
|
m_validityWatcher->cancel();
|
|
}
|
|
|
|
//---------------------------------------------------------------------------------------------------------------------
|
|
auto VPSheet::GetLayout() const -> VPLayoutPtr
|
|
{
|
|
return m_layout;
|
|
}
|
|
|
|
//---------------------------------------------------------------------------------------------------------------------
|
|
auto VPSheet::GetPieces() const -> QList<VPPiecePtr>
|
|
{
|
|
if (VPLayoutPtr const layout = GetLayout(); not layout.isNull())
|
|
{
|
|
return layout->PiecesForSheet(m_uuid);
|
|
}
|
|
|
|
return {};
|
|
}
|
|
|
|
//---------------------------------------------------------------------------------------------------------------------
|
|
auto VPSheet::GetSelectedPieces() const -> QList<VPPiecePtr>
|
|
{
|
|
if (VPLayoutPtr const layout = GetLayout(); not layout.isNull())
|
|
{
|
|
QList<VPPiecePtr> const list = layout->PiecesForSheet(m_uuid);
|
|
|
|
QList<VPPiecePtr> selected;
|
|
selected.reserve(list.size());
|
|
|
|
for (const auto &piece : list)
|
|
{
|
|
if (not piece.isNull() && piece->IsSelected())
|
|
{
|
|
selected.append(piece);
|
|
}
|
|
}
|
|
|
|
return selected;
|
|
}
|
|
|
|
return {};
|
|
}
|
|
|
|
//---------------------------------------------------------------------------------------------------------------------
|
|
auto VPSheet::GetAsLayoutPieces() const -> QVector<VLayoutPiece>
|
|
{
|
|
QList<VPPiecePtr> const pieces = GetPieces();
|
|
|
|
QVector<VLayoutPiece> details;
|
|
details.reserve(pieces.size());
|
|
|
|
for (const auto &piece : pieces)
|
|
{
|
|
if (not piece.isNull())
|
|
{
|
|
details.append(*piece);
|
|
}
|
|
}
|
|
|
|
return details;
|
|
}
|
|
|
|
//---------------------------------------------------------------------------------------------------------------------
|
|
auto VPSheet::GetName() const -> QString
|
|
{
|
|
return m_name;
|
|
}
|
|
|
|
//---------------------------------------------------------------------------------------------------------------------
|
|
void VPSheet::SetName(const QString &name)
|
|
{
|
|
m_name = name;
|
|
}
|
|
|
|
//---------------------------------------------------------------------------------------------------------------------
|
|
auto VPSheet::Uuid() const -> const QUuid &
|
|
{
|
|
return m_uuid;
|
|
}
|
|
|
|
//---------------------------------------------------------------------------------------------------------------------
|
|
auto VPSheet::IsVisible() const -> bool
|
|
{
|
|
return m_visible;
|
|
}
|
|
|
|
//---------------------------------------------------------------------------------------------------------------------
|
|
void VPSheet::SetVisible(bool visible)
|
|
{
|
|
m_visible = visible;
|
|
}
|
|
|
|
//---------------------------------------------------------------------------------------------------------------------
|
|
auto VPSheet::GrainlineOrientation() const -> GrainlineType
|
|
{
|
|
if (m_grainlineType == GrainlineType::NotFixed)
|
|
{
|
|
if (m_size.height() < m_size.width())
|
|
{
|
|
return GrainlineType::Horizontal;
|
|
}
|
|
|
|
return GrainlineType::Vertical;
|
|
}
|
|
|
|
return m_grainlineType;
|
|
}
|
|
|
|
//---------------------------------------------------------------------------------------------------------------------
|
|
auto VPSheet::GetGrainlineType() const -> GrainlineType
|
|
{
|
|
return m_grainlineType;
|
|
}
|
|
|
|
//---------------------------------------------------------------------------------------------------------------------
|
|
void VPSheet::SetGrainlineType(GrainlineType type)
|
|
{
|
|
m_grainlineType = type;
|
|
}
|
|
|
|
//---------------------------------------------------------------------------------------------------------------------
|
|
auto VPSheet::TransformationOrigin() const -> const VPTransformationOrigon &
|
|
{
|
|
return m_transformationOrigin;
|
|
}
|
|
|
|
//---------------------------------------------------------------------------------------------------------------------
|
|
void VPSheet::SetTransformationOrigin(const VPTransformationOrigon &newTransformationOrigin)
|
|
{
|
|
m_transformationOrigin = newTransformationOrigin;
|
|
}
|
|
|
|
//---------------------------------------------------------------------------------------------------------------------
|
|
void VPSheet::Clear()
|
|
{
|
|
QT_WARNING_PUSH
|
|
QT_WARNING_DISABLE_GCC("-Wnoexcept")
|
|
|
|
m_name.clear();
|
|
m_visible = true;
|
|
m_transformationOrigin = VPTransformationOrigon();
|
|
m_trashSheet = false;
|
|
|
|
QT_WARNING_POP
|
|
}
|
|
|
|
//---------------------------------------------------------------------------------------------------------------------
|
|
auto VPSheet::TrashSheet() const -> bool
|
|
{
|
|
return m_trashSheet;
|
|
}
|
|
|
|
//---------------------------------------------------------------------------------------------------------------------
|
|
void VPSheet::SetTrashSheet(bool newTrashSheet)
|
|
{
|
|
m_trashSheet = newTrashSheet;
|
|
}
|
|
|
|
//---------------------------------------------------------------------------------------------------------------------
|
|
auto VPSheet::GetSheetRect() const -> QRectF
|
|
{
|
|
return {QPoint(0, 0), m_size};
|
|
}
|
|
|
|
//---------------------------------------------------------------------------------------------------------------------
|
|
auto VPSheet::GetMarginsRect() const -> QRectF
|
|
{
|
|
if (not m_ignoreMargins)
|
|
{
|
|
auto rect = QRectF(QPointF(m_margins.left(), m_margins.top()),
|
|
QPointF(m_size.width() - m_margins.right(), m_size.height() - m_margins.bottom()));
|
|
return rect;
|
|
}
|
|
|
|
return {0, 0, m_size.width(), m_size.height()};
|
|
}
|
|
|
|
//---------------------------------------------------------------------------------------------------------------------
|
|
void VPSheet::RemoveUnusedLength()
|
|
{
|
|
VPLayoutPtr const layout = GetLayout();
|
|
if (layout.isNull())
|
|
{
|
|
return;
|
|
}
|
|
|
|
QList<VPPiecePtr> const pieces = GetPieces();
|
|
if (pieces.isEmpty())
|
|
{
|
|
return;
|
|
}
|
|
|
|
QRectF piecesBoundingRect;
|
|
|
|
for (const auto &piece : pieces)
|
|
{
|
|
if (not piece.isNull())
|
|
{
|
|
piece->SetSelected(false);
|
|
emit layout->PieceSelectionChanged(piece);
|
|
piecesBoundingRect = piecesBoundingRect.united(piece->MappedDetailBoundingRect());
|
|
}
|
|
}
|
|
|
|
const qreal extra = 2;
|
|
QRectF const sheetRect = GetSheetRect();
|
|
GrainlineType const type = GrainlineOrientation();
|
|
|
|
if (type == GrainlineType::Vertical)
|
|
{
|
|
qreal margin = 0;
|
|
if (not m_ignoreMargins)
|
|
{
|
|
margin = m_margins.bottom();
|
|
}
|
|
|
|
if (sheetRect.bottomRight().y() - margin > piecesBoundingRect.bottomRight().y())
|
|
{
|
|
m_size = QSizeF(m_size.width(), piecesBoundingRect.bottomRight().y() + margin + extra);
|
|
}
|
|
}
|
|
else if (type == GrainlineType::Horizontal)
|
|
{
|
|
qreal margin = 0;
|
|
if (not m_ignoreMargins)
|
|
{
|
|
margin = m_margins.right();
|
|
}
|
|
|
|
if (sheetRect.topRight().x() - margin > piecesBoundingRect.topRight().x())
|
|
{
|
|
m_size = QSizeF(piecesBoundingRect.topRight().x() + margin + extra, m_size.height());
|
|
}
|
|
}
|
|
}
|
|
|
|
//---------------------------------------------------------------------------------------------------------------------
|
|
void VPSheet::CheckPiecesPositionValidity() const
|
|
{
|
|
if (m_validityWatcher->isFinished())
|
|
{
|
|
VPLayoutPtr const layout = GetLayout();
|
|
if (layout.isNull())
|
|
{
|
|
return;
|
|
}
|
|
|
|
VPiecesValidationData data;
|
|
data.m_warnPiecesOutOfBound = layout->LayoutSettings().GetWarningPiecesOutOfBound();
|
|
data.m_warnSuperpositionOfPieces = layout->LayoutSettings().GetWarningSuperpositionOfPieces();
|
|
data.m_warnPieceGapePosition = layout->LayoutSettings().GetWarningPieceGapePosition();
|
|
data.m_cutOnFold = layout->LayoutSettings().IsCutOnFold();
|
|
data.m_sheetRect = GetMarginsRect();
|
|
data.m_pieceGap = layout->LayoutSettings().GetPiecesGap();
|
|
|
|
QList<VPPiecePtr> const pieces = GetPieces();
|
|
QVector<VSheetPiece> sheetPieces;
|
|
sheetPieces.reserve(pieces.size());
|
|
|
|
for (const auto &piece : pieces)
|
|
{
|
|
VSheetPiece data;
|
|
data.m_showFullPiece = piece->IsShowFullPiece();
|
|
data.m_id = piece->GetUniqueID();
|
|
data.m_seamMirrorLine = piece->GetMappedSeamMirrorLine();
|
|
data.m_seamAllowanceMirrorLine = piece->GetMappedSeamAllowanceMirrorLine();
|
|
|
|
QVector<QPointF> points;
|
|
CastTo(piece->GetMappedExternalContourPoints(), points);
|
|
data.m_externalContourPoints = points;
|
|
|
|
sheetPieces.append(data);
|
|
}
|
|
|
|
data.m_pieces = sheetPieces;
|
|
|
|
m_validationStale = false;
|
|
m_validityWatcher->setFuture(QtConcurrent::run([data]() { return ValidatePiecesPositions(data); }));
|
|
}
|
|
else
|
|
{
|
|
m_validationStale = true;
|
|
}
|
|
}
|
|
|
|
//---------------------------------------------------------------------------------------------------------------------
|
|
void VPSheet::UpdatePiecesValidity()
|
|
{
|
|
if (m_validityWatcher->isCanceled())
|
|
{
|
|
return;
|
|
}
|
|
|
|
QHash<QString, VPiecePositionValidity> const newValidations = m_validityWatcher->future().result();
|
|
|
|
QList<VPPiecePtr> const pieces = GetPieces();
|
|
QHash<QString, VPPiecePtr> sortedPieces;
|
|
sortedPieces.reserve(pieces.size());
|
|
|
|
for (const auto &piece : pieces)
|
|
{
|
|
sortedPieces.insert(piece->GetUniqueID(), piece);
|
|
}
|
|
|
|
VPLayoutPtr const layout = GetLayout();
|
|
if (layout.isNull())
|
|
{
|
|
return;
|
|
}
|
|
|
|
for (auto i = newValidations.cbegin(), end = newValidations.cend(); i != end; ++i)
|
|
{
|
|
if (sortedPieces.contains(i.key()))
|
|
{
|
|
VPPiecePtr const piece = sortedPieces.value(i.key());
|
|
piece->SetOutOfBound(i.value().m_outOfBound);
|
|
piece->SetHasSuperpositionWithPieces(i.value().m_superposition);
|
|
piece->SetHasInvalidPieceGapPosition(i.value().m_gap);
|
|
|
|
emit layout->PiecePositionValidityChanged(piece);
|
|
}
|
|
}
|
|
|
|
if (m_validationStale)
|
|
{
|
|
m_validationStale = false;
|
|
CheckPiecesPositionValidity();
|
|
}
|
|
}
|
|
|
|
//---------------------------------------------------------------------------------------------------------------------
|
|
auto VPSheet::SceneData() const -> QSharedPointer<VPSheetSceneData>
|
|
{
|
|
return m_sceneData;
|
|
}
|
|
|
|
//---------------------------------------------------------------------------------------------------------------------
|
|
void VPSheet::ClearSelection() const
|
|
{
|
|
QList<VPPiecePtr> const selectedPieces = GetSelectedPieces();
|
|
for (const auto &piece : selectedPieces)
|
|
{
|
|
if (piece->IsSelected())
|
|
{
|
|
piece->SetSelected(false);
|
|
}
|
|
}
|
|
|
|
if (not selectedPieces.isEmpty())
|
|
{
|
|
VPLayoutPtr const layout = GetLayout();
|
|
if (not layout.isNull())
|
|
{
|
|
emit layout->PieceSelectionChanged(VPPiecePtr());
|
|
}
|
|
}
|
|
}
|
|
|
|
//---------------------------------------------------------------------------------------------------------------------
|
|
auto VPSheet::GetSheetOrientation() const -> QPageLayout::Orientation
|
|
{
|
|
return m_size.height() >= m_size.width() ? QPageLayout::Portrait : QPageLayout::Landscape;
|
|
}
|
|
|
|
//---------------------------------------------------------------------------------------------------------------------
|
|
auto VPSheet::SheetUnits() const -> Unit
|
|
{
|
|
if (VPLayoutPtr const layout = GetLayout(); not layout.isNull())
|
|
{
|
|
return layout->LayoutSettings().GetUnit();
|
|
}
|
|
|
|
return Unit::Cm;
|
|
}
|
|
|
|
//---------------------------------------------------------------------------------------------------------------------
|
|
void VPSheet::SetSheetSize(qreal width, qreal height)
|
|
{
|
|
m_size.setWidth(width);
|
|
m_size.setHeight(height);
|
|
|
|
if (m_sceneData != nullptr)
|
|
{
|
|
m_sceneData->RefreshSheetSize();
|
|
}
|
|
}
|
|
|
|
//---------------------------------------------------------------------------------------------------------------------
|
|
void VPSheet::SetSheetSizeConverted(qreal width, qreal height)
|
|
{
|
|
Unit const unit = SheetUnits();
|
|
m_size.setWidth(UnitConvertor(width, unit, Unit::Px));
|
|
m_size.setHeight(UnitConvertor(height, unit, Unit::Px));
|
|
|
|
if (m_sceneData != nullptr)
|
|
{
|
|
m_sceneData->RefreshSheetSize();
|
|
}
|
|
}
|
|
|
|
//---------------------------------------------------------------------------------------------------------------------
|
|
void VPSheet::SetSheetSize(const QSizeF &size)
|
|
{
|
|
m_size = size;
|
|
|
|
if (m_sceneData != nullptr)
|
|
{
|
|
m_sceneData->RefreshSheetSize();
|
|
}
|
|
}
|
|
|
|
//---------------------------------------------------------------------------------------------------------------------
|
|
void VPSheet::SetSheetSizeConverted(const QSizeF &size)
|
|
{
|
|
Unit const unit = SheetUnits();
|
|
m_size = QSizeF(UnitConvertor(size.width(), unit, Unit::Px), UnitConvertor(size.height(), unit, Unit::Px));
|
|
|
|
if (m_sceneData != nullptr)
|
|
{
|
|
m_sceneData->RefreshSheetSize();
|
|
}
|
|
}
|
|
|
|
//---------------------------------------------------------------------------------------------------------------------
|
|
auto VPSheet::GetSheetSize() const -> QSizeF
|
|
{
|
|
return m_size;
|
|
}
|
|
|
|
//---------------------------------------------------------------------------------------------------------------------
|
|
auto VPSheet::GetSheetSizeConverted() const -> QSizeF
|
|
{
|
|
Unit const unit = SheetUnits();
|
|
auto convertedSize =
|
|
QSizeF(UnitConvertor(m_size.width(), Unit::Px, unit), UnitConvertor(m_size.height(), Unit::Px, unit));
|
|
|
|
return convertedSize;
|
|
}
|
|
|
|
//---------------------------------------------------------------------------------------------------------------------
|
|
void VPSheet::SetSheetMargins(qreal left, qreal top, qreal right, qreal bottom)
|
|
{
|
|
m_margins.setLeft(left);
|
|
m_margins.setTop(top);
|
|
m_margins.setRight(right);
|
|
m_margins.setBottom(bottom);
|
|
}
|
|
//---------------------------------------------------------------------------------------------------------------------
|
|
void VPSheet::SetSheetMarginsConverted(qreal left, qreal top, qreal right, qreal bottom)
|
|
{
|
|
Unit const unit = SheetUnits();
|
|
m_margins.setLeft(UnitConvertor(left, unit, Unit::Px));
|
|
m_margins.setTop(UnitConvertor(top, unit, Unit::Px));
|
|
m_margins.setRight(UnitConvertor(right, unit, Unit::Px));
|
|
m_margins.setBottom(UnitConvertor(bottom, unit, Unit::Px));
|
|
}
|
|
|
|
//---------------------------------------------------------------------------------------------------------------------
|
|
void VPSheet::SetSheetMargins(const QMarginsF &margins)
|
|
{
|
|
m_margins = margins;
|
|
}
|
|
|
|
//---------------------------------------------------------------------------------------------------------------------
|
|
void VPSheet::SetSheetMarginsConverted(const QMarginsF &margins)
|
|
{
|
|
Unit const unit = SheetUnits();
|
|
m_margins = UnitConvertor(margins, unit, Unit::Px);
|
|
}
|
|
|
|
//---------------------------------------------------------------------------------------------------------------------
|
|
auto VPSheet::GetSheetMargins() const -> QMarginsF
|
|
{
|
|
return m_margins;
|
|
}
|
|
|
|
//---------------------------------------------------------------------------------------------------------------------
|
|
auto VPSheet::GetSheetMarginsConverted() const -> QMarginsF
|
|
{
|
|
Unit const unit = SheetUnits();
|
|
return UnitConvertor(m_margins, Unit::Px, unit);
|
|
}
|
|
|
|
//---------------------------------------------------------------------------------------------------------------------
|
|
auto VPSheet::IgnoreMargins() const -> bool
|
|
{
|
|
return m_ignoreMargins;
|
|
}
|
|
|
|
//---------------------------------------------------------------------------------------------------------------------
|
|
void VPSheet::SetIgnoreMargins(bool newIgnoreMargins)
|
|
{
|
|
m_ignoreMargins = newIgnoreMargins;
|
|
}
|