2020-05-23 17:47:04 +02:00
|
|
|
/************************************************************************
|
|
|
|
**
|
|
|
|
** @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/>.
|
|
|
|
**
|
|
|
|
*************************************************************************/
|
2020-05-23 17:46:46 +02:00
|
|
|
#include "vpsheet.h"
|
|
|
|
|
2020-05-23 17:47:04 +02:00
|
|
|
#include "vplayout.h"
|
2021-08-25 15:58:50 +02:00
|
|
|
#include "vppiece.h"
|
2020-05-23 17:47:04 +02:00
|
|
|
|
|
|
|
//---------------------------------------------------------------------------------------------------------------------
|
2021-08-17 17:49:28 +02:00
|
|
|
VPSheet::VPSheet(const VPLayoutPtr &layout) :
|
2020-05-23 17:47:04 +02:00
|
|
|
m_layout(layout)
|
|
|
|
{
|
2021-07-29 16:11:18 +02:00
|
|
|
SCASSERT(layout != nullptr)
|
2020-05-23 17:47:04 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
//---------------------------------------------------------------------------------------------------------------------
|
2021-08-17 17:49:28 +02:00
|
|
|
auto VPSheet::GetLayout() const -> VPLayoutPtr
|
2020-05-23 17:47:04 +02:00
|
|
|
{
|
2021-07-29 16:11:18 +02:00
|
|
|
return m_layout;
|
2020-11-20 15:51:24 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
//---------------------------------------------------------------------------------------------------------------------
|
2021-08-17 17:49:28 +02:00
|
|
|
auto VPSheet::GetPieces() const -> QList<VPPiecePtr>
|
2020-11-20 15:51:24 +01:00
|
|
|
{
|
2021-08-17 17:49:28 +02:00
|
|
|
VPLayoutPtr layout = GetLayout();
|
|
|
|
if (not layout.isNull())
|
2020-11-20 15:51:24 +01:00
|
|
|
{
|
2021-08-17 17:49:28 +02:00
|
|
|
return layout->PiecesForSheet(m_uuid);
|
2020-11-20 15:51:24 +01:00
|
|
|
}
|
|
|
|
|
2021-07-29 16:11:18 +02:00
|
|
|
return {};
|
2020-05-23 17:47:04 +02:00
|
|
|
}
|
|
|
|
|
2021-08-25 15:58:50 +02:00
|
|
|
//---------------------------------------------------------------------------------------------------------------------
|
|
|
|
auto VPSheet::GetSelectedPieces() const -> QList<VPPiecePtr>
|
|
|
|
{
|
|
|
|
VPLayoutPtr layout = GetLayout();
|
|
|
|
if (not layout.isNull())
|
|
|
|
{
|
|
|
|
QList<VPPiecePtr> 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 {};
|
|
|
|
}
|
|
|
|
|
2020-05-23 17:47:04 +02:00
|
|
|
//---------------------------------------------------------------------------------------------------------------------
|
2021-07-29 16:11:18 +02:00
|
|
|
auto VPSheet::GetName() const -> QString
|
2020-05-23 17:47:04 +02:00
|
|
|
{
|
|
|
|
return m_name;
|
|
|
|
}
|
|
|
|
|
|
|
|
//---------------------------------------------------------------------------------------------------------------------
|
|
|
|
void VPSheet::SetName(const QString &name)
|
|
|
|
{
|
|
|
|
m_name = name;
|
|
|
|
}
|
2021-07-31 11:21:07 +02:00
|
|
|
|
|
|
|
//---------------------------------------------------------------------------------------------------------------------
|
|
|
|
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;
|
|
|
|
}
|
2021-08-09 14:09:10 +02:00
|
|
|
|
|
|
|
//---------------------------------------------------------------------------------------------------------------------
|
|
|
|
auto VPSheet::GrainlineType() const -> enum GrainlineType
|
|
|
|
{
|
2021-08-17 17:49:28 +02:00
|
|
|
VPLayoutPtr layout = GetLayout();
|
|
|
|
if (not layout.isNull())
|
2021-08-09 14:09:10 +02:00
|
|
|
{
|
2021-08-17 17:49:28 +02:00
|
|
|
QSizeF size = layout->LayoutSettings().GetSheetSize();
|
2021-08-09 14:09:10 +02:00
|
|
|
if (size.height() < size.width())
|
|
|
|
{
|
|
|
|
return GrainlineType::Horizontal;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return GrainlineType::Vertical;
|
|
|
|
}
|
|
|
|
|
|
|
|
//---------------------------------------------------------------------------------------------------------------------
|
|
|
|
auto VPSheet::TransformationOrigin() const -> const VPTransformationOrigon &
|
|
|
|
{
|
|
|
|
return m_transformationOrigin;
|
|
|
|
}
|
|
|
|
|
|
|
|
//---------------------------------------------------------------------------------------------------------------------
|
|
|
|
void VPSheet::SetTransformationOrigin(const VPTransformationOrigon &newTransformationOrigin)
|
|
|
|
{
|
|
|
|
m_transformationOrigin = newTransformationOrigin;
|
|
|
|
}
|
2021-08-17 17:49:28 +02:00
|
|
|
|
|
|
|
//---------------------------------------------------------------------------------------------------------------------
|
|
|
|
void VPSheet::Clear()
|
|
|
|
{
|
|
|
|
m_name.clear();
|
|
|
|
m_visible = true;
|
|
|
|
m_transformationOrigin = VPTransformationOrigon();
|
2021-08-19 14:13:54 +02:00
|
|
|
m_trashSheet = false;
|
|
|
|
}
|
|
|
|
|
|
|
|
//---------------------------------------------------------------------------------------------------------------------
|
|
|
|
auto VPSheet::TrashSheet() const -> bool
|
|
|
|
{
|
|
|
|
return m_trashSheet;
|
|
|
|
}
|
|
|
|
|
|
|
|
//---------------------------------------------------------------------------------------------------------------------
|
|
|
|
void VPSheet::SetTrashSheet(bool newTrashSheet)
|
|
|
|
{
|
|
|
|
m_trashSheet = newTrashSheet;
|
2021-08-17 17:49:28 +02:00
|
|
|
}
|
2021-08-27 17:27:38 +02:00
|
|
|
|
|
|
|
//---------------------------------------------------------------------------------------------------------------------
|
|
|
|
void VPSheet::ValidateSuperpositionOfPieces() const
|
|
|
|
{
|
|
|
|
QList<VPPiecePtr> pieces = GetPieces();
|
|
|
|
|
|
|
|
for (const auto &piece : pieces)
|
|
|
|
{
|
|
|
|
if (piece.isNull())
|
|
|
|
{
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
const bool oldSuperpositionOfPieces = piece->HasSuperpositionWithPieces();
|
|
|
|
QVector<QPointF> path1 = piece->GetMappedExternalContourPoints();
|
|
|
|
bool hasSuperposition = false;
|
|
|
|
|
|
|
|
for (const auto &p : pieces)
|
|
|
|
{
|
|
|
|
if (p.isNull() || piece == p)
|
|
|
|
{
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
QVector<QPointF> path2 = p->GetMappedExternalContourPoints();
|
|
|
|
|
|
|
|
bool superposition = PathsSuperposition(path1, path2);
|
|
|
|
if (superposition)
|
|
|
|
{
|
|
|
|
hasSuperposition = superposition;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
piece->SetHasSuperpositionWithPieces(hasSuperposition);
|
|
|
|
|
|
|
|
if (oldSuperpositionOfPieces != piece->HasSuperpositionWithPieces())
|
|
|
|
{
|
|
|
|
VPLayoutPtr layout = GetLayout();
|
|
|
|
if (not layout.isNull())
|
|
|
|
{
|
|
|
|
emit layout->PiecePositionValidityChanged(piece);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
//---------------------------------------------------------------------------------------------------------------------
|
|
|
|
void VPSheet::ValidatePieceOutOfBound(const VPPiecePtr &piece) const
|
|
|
|
{
|
|
|
|
if (piece.isNull())
|
|
|
|
{
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
const bool oldOutOfBound = piece->OutOfBound();
|
|
|
|
|
|
|
|
QRectF pieceRect = piece->MappedDetailBoundingRect();
|
|
|
|
QRectF sheetRect = GetMarginsRect();
|
|
|
|
|
|
|
|
piece->SetOutOfBound(not sheetRect.contains(pieceRect));
|
|
|
|
|
|
|
|
if (oldOutOfBound != piece->OutOfBound())
|
|
|
|
{
|
|
|
|
VPLayoutPtr layout = GetLayout();
|
|
|
|
if (not layout.isNull())
|
|
|
|
{
|
|
|
|
emit layout->PiecePositionValidityChanged(piece);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
//---------------------------------------------------------------------------------------------------------------------
|
|
|
|
void VPSheet::ValidatePiecesOutOfBound() const
|
|
|
|
{
|
|
|
|
QList<VPPiecePtr> pieces = GetPieces();
|
|
|
|
for (const auto &piece : pieces)
|
|
|
|
{
|
|
|
|
ValidatePieceOutOfBound(piece);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
//---------------------------------------------------------------------------------------------------------------------
|
|
|
|
auto VPSheet::GetSheetRect() const -> QRectF
|
|
|
|
{
|
|
|
|
return GetSheetRect(GetLayout());
|
|
|
|
}
|
|
|
|
|
|
|
|
//---------------------------------------------------------------------------------------------------------------------
|
|
|
|
auto VPSheet::GetMarginsRect() const -> QRectF
|
|
|
|
{
|
|
|
|
return GetMarginsRect(GetLayout());
|
|
|
|
}
|
|
|
|
|
|
|
|
//---------------------------------------------------------------------------------------------------------------------
|
|
|
|
auto VPSheet::GetSheetRect(const VPLayoutPtr &layout) -> QRectF
|
|
|
|
{
|
|
|
|
if (layout.isNull())
|
|
|
|
{
|
|
|
|
return {};
|
|
|
|
}
|
|
|
|
|
|
|
|
QPoint topLeft = QPoint(0,0);
|
|
|
|
QSizeF size = layout->LayoutSettings().GetSheetSize();
|
|
|
|
QRectF rect = QRectF(topLeft, size);
|
|
|
|
return rect;
|
|
|
|
}
|
|
|
|
|
|
|
|
//---------------------------------------------------------------------------------------------------------------------
|
|
|
|
auto VPSheet::GetMarginsRect(const VPLayoutPtr &layout) -> QRectF
|
|
|
|
{
|
|
|
|
if (layout.isNull())
|
|
|
|
{
|
|
|
|
return {};
|
|
|
|
}
|
|
|
|
|
|
|
|
QSizeF size = layout->LayoutSettings().GetSheetSize();
|
|
|
|
|
|
|
|
if (not layout->LayoutSettings().IgnoreMargins())
|
|
|
|
{
|
|
|
|
QMarginsF margins = layout->LayoutSettings().GetSheetMargins();
|
|
|
|
QRectF rect = QRectF(QPointF(margins.left(), margins.top()),
|
|
|
|
QPointF(size.width()-margins.right(), size.height()-margins.bottom()));
|
|
|
|
return rect;
|
|
|
|
}
|
|
|
|
|
|
|
|
return QRectF(0, 0, size.width(), size.height());
|
|
|
|
}
|
|
|
|
|
|
|
|
//---------------------------------------------------------------------------------------------------------------------
|
|
|
|
void VPSheet::CheckPiecePositionValidity(const VPPiecePtr &piece) const
|
|
|
|
{
|
|
|
|
VPLayoutPtr layout = GetLayout();
|
|
|
|
if (layout.isNull())
|
|
|
|
{
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
QList<VPPiecePtr> pieces = GetPieces();
|
|
|
|
if (piece.isNull() || not pieces.contains(piece))
|
|
|
|
{
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (layout->LayoutSettings().GetWarningPiecesOutOfBound())
|
|
|
|
{
|
|
|
|
ValidatePieceOutOfBound(piece);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (layout->LayoutSettings().GetWarningSuperpositionOfPieces())
|
|
|
|
{
|
|
|
|
ValidateSuperpositionOfPieces();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
//---------------------------------------------------------------------------------------------------------------------
|
|
|
|
auto VPSheet::PathsSuperposition(const QVector<QPointF> &path1, const QVector<QPointF> &path2) const -> bool
|
|
|
|
{
|
|
|
|
const QRectF path1Rect = VLayoutPiece::BoundingRect(path1);
|
|
|
|
const QPainterPath path1Path = VAbstractPiece::PainterPath(path1);
|
|
|
|
|
|
|
|
const QRectF path2Rect = VLayoutPiece::BoundingRect(path2);
|
|
|
|
const QPainterPath path2Path = VAbstractPiece::PainterPath(path2);
|
|
|
|
|
|
|
|
if (path1Rect.intersects(path2Rect) || path2Rect.contains(path1Rect) || path1Rect.contains(path2Rect))
|
|
|
|
{
|
|
|
|
if (path1Path.contains(path2Path) || path2Path.contains(path1Path) || path1Path.intersects(path2Path))
|
|
|
|
{
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return false;
|
|
|
|
}
|