2013-11-15 13:41:26 +01:00
|
|
|
/************************************************************************
|
2013-09-18 21:16:19 +02:00
|
|
|
**
|
2013-11-15 13:50:05 +01:00
|
|
|
** @file vnodedetail.cpp
|
2014-04-30 07:38:52 +02:00
|
|
|
** @author Roman Telezhynskyi <dismine(at)gmail.com>
|
2013-11-15 13:50:05 +01:00
|
|
|
** @date November 15, 2013
|
2013-09-18 21:16:19 +02:00
|
|
|
**
|
2013-11-15 13:41:26 +01:00
|
|
|
** @brief
|
|
|
|
** @copyright
|
2017-10-05 11:20:01 +02:00
|
|
|
** This source code is part of the Valentina project, a pattern making
|
2013-11-15 13:41:26 +01:00
|
|
|
** program, whose allow create and modeling patterns of clothing.
|
2015-02-27 11:27:48 +01:00
|
|
|
** Copyright (C) 2013-2015 Valentina project
|
2020-01-31 07:00:05 +01:00
|
|
|
** <https://gitlab.com/smart-pattern/valentina> All Rights Reserved.
|
2013-09-18 21:16:19 +02:00
|
|
|
**
|
2013-11-15 13:41:26 +01:00
|
|
|
** Valentina is free software: you can redistribute it and/or modify
|
2013-09-18 21:16:19 +02:00
|
|
|
** 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.
|
|
|
|
**
|
2013-10-27 13:36:29 +01:00
|
|
|
** Valentina is distributed in the hope that it will be useful,
|
2013-09-18 21:16:19 +02:00
|
|
|
** 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/>.
|
|
|
|
**
|
2013-11-15 13:41:26 +01:00
|
|
|
*************************************************************************/
|
2013-09-18 21:16:19 +02:00
|
|
|
|
2013-08-28 10:55:11 +02:00
|
|
|
#include "vnodedetail.h"
|
2014-08-20 12:33:29 +02:00
|
|
|
#include "vnodedetail_p.h"
|
2016-12-01 19:47:15 +01:00
|
|
|
#include "vpiecenode.h"
|
|
|
|
#include "vpiecepath.h"
|
|
|
|
#include "../vgeometry/vpointf.h"
|
|
|
|
#include "../vpatterndb/vcontainer.h"
|
|
|
|
|
|
|
|
#include <QLineF>
|
|
|
|
#include <QVector>
|
2013-08-28 10:55:11 +02:00
|
|
|
|
2017-01-25 10:32:52 +01:00
|
|
|
namespace
|
|
|
|
{
|
|
|
|
//---------------------------------------------------------------------------------------------------------------------
|
|
|
|
bool IsOX(const QLineF &line)
|
|
|
|
{
|
|
|
|
return VFuzzyComparePossibleNulls(line.angle(), 0)
|
|
|
|
|| VFuzzyComparePossibleNulls(line.angle(), 360)
|
|
|
|
|| VFuzzyComparePossibleNulls(line.angle(), 180);
|
|
|
|
}
|
|
|
|
|
|
|
|
//---------------------------------------------------------------------------------------------------------------------
|
|
|
|
bool IsOY(const QLineF &line)
|
|
|
|
{
|
|
|
|
return VFuzzyComparePossibleNulls(line.angle(), 90) || VFuzzyComparePossibleNulls(line.angle(), 270);
|
|
|
|
}
|
|
|
|
|
|
|
|
//---------------------------------------------------------------------------------------------------------------------
|
|
|
|
QString LocalWidth(const QLineF &line, const QLineF &movedLine)
|
|
|
|
{
|
|
|
|
if (VFuzzyComparePossibleNulls(line.angle(), movedLine.angle()))
|
|
|
|
{
|
|
|
|
return QString().setNum(movedLine.length());
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{// different direction means value is negative
|
2018-04-12 12:13:19 +02:00
|
|
|
return QChar('0');
|
2017-01-25 10:32:52 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
//---------------------------------------------------------------------------------------------------------------------
|
|
|
|
void ConvertBefore(VPieceNode &node, const QLineF &line, qreal mX, qreal mY)
|
|
|
|
{
|
|
|
|
if (not qFuzzyIsNull(mX) && IsOX(line))
|
|
|
|
{
|
|
|
|
const QLineF movedLine(line.p1().x(), line.p1().y(), line.p2().x() + mX, line.p2().y());
|
|
|
|
node.SetFormulaSABefore(LocalWidth(line, movedLine));
|
|
|
|
}
|
|
|
|
else if (not qFuzzyIsNull(mY) && IsOY(line))
|
|
|
|
{
|
|
|
|
const QLineF movedLine(line.p1().x(), line.p1().y(), line.p2().x(), line.p2().y() + mY);
|
|
|
|
node.SetFormulaSABefore(LocalWidth(line, movedLine));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
//---------------------------------------------------------------------------------------------------------------------
|
|
|
|
void ConvertAfter(VPieceNode &node, const QLineF &line, qreal mX, qreal mY)
|
|
|
|
{
|
|
|
|
if (not qFuzzyIsNull(mX) && IsOX(line))
|
|
|
|
{
|
|
|
|
const QLineF movedLine(line.p1().x(), line.p1().y(), line.p2().x() + mX, line.p2().y());
|
|
|
|
node.SetFormulaSAAfter(LocalWidth(line, movedLine));
|
|
|
|
}
|
|
|
|
else if (not qFuzzyIsNull(mY) && IsOY(line))
|
|
|
|
{
|
|
|
|
const QLineF movedLine(line.p1().x(), line.p1().y(), line.p2().x(), line.p2().y() + mY);
|
|
|
|
node.SetFormulaSAAfter(LocalWidth(line, movedLine));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}//static functions
|
|
|
|
|
2014-05-02 13:11:30 +02:00
|
|
|
//---------------------------------------------------------------------------------------------------------------------
|
2013-11-04 21:35:15 +01:00
|
|
|
VNodeDetail::VNodeDetail()
|
2014-08-20 12:33:29 +02:00
|
|
|
:d(new VNodeDetailData)
|
2014-05-02 13:11:30 +02:00
|
|
|
{}
|
2013-08-28 10:55:11 +02:00
|
|
|
|
2014-05-02 13:11:30 +02:00
|
|
|
//---------------------------------------------------------------------------------------------------------------------
|
2014-12-17 13:56:14 +01:00
|
|
|
VNodeDetail::VNodeDetail(quint32 id, Tool typeTool, NodeDetail typeNode, qreal mx, qreal my, bool reverse)
|
|
|
|
:d(new VNodeDetailData(id, typeTool, typeNode, mx, my, reverse))
|
2014-05-02 13:11:30 +02:00
|
|
|
{}
|
2013-08-28 10:55:11 +02:00
|
|
|
|
2014-05-02 13:11:30 +02:00
|
|
|
//---------------------------------------------------------------------------------------------------------------------
|
2013-11-04 21:35:15 +01:00
|
|
|
VNodeDetail::VNodeDetail(const VNodeDetail &node)
|
2014-08-20 12:33:29 +02:00
|
|
|
:d (node.d)
|
2014-05-02 13:11:30 +02:00
|
|
|
{}
|
2013-10-25 13:10:45 +02:00
|
|
|
|
2014-05-02 13:11:30 +02:00
|
|
|
//---------------------------------------------------------------------------------------------------------------------
|
2013-11-04 21:35:15 +01:00
|
|
|
VNodeDetail &VNodeDetail::operator =(const VNodeDetail &node)
|
|
|
|
{
|
2014-07-27 14:30:28 +02:00
|
|
|
if ( &node == this )
|
|
|
|
{
|
|
|
|
return *this;
|
|
|
|
}
|
2014-08-20 12:33:29 +02:00
|
|
|
d = node.d;
|
2013-10-25 13:10:45 +02:00
|
|
|
return *this;
|
2013-08-28 10:55:11 +02:00
|
|
|
}
|
2014-08-20 12:33:29 +02:00
|
|
|
|
2019-12-30 12:00:57 +01:00
|
|
|
#ifdef Q_COMPILER_RVALUE_REFS
|
|
|
|
//---------------------------------------------------------------------------------------------------------------------
|
2019-12-30 16:13:18 +01:00
|
|
|
VNodeDetail::VNodeDetail(const VNodeDetail &&node) Q_DECL_NOTHROW
|
|
|
|
:d (node.d)
|
|
|
|
{}
|
2019-12-30 12:00:57 +01:00
|
|
|
|
|
|
|
//---------------------------------------------------------------------------------------------------------------------
|
2019-12-30 16:13:18 +01:00
|
|
|
VNodeDetail &VNodeDetail::operator=(VNodeDetail &&node) Q_DECL_NOTHROW
|
2019-12-30 12:00:57 +01:00
|
|
|
{
|
|
|
|
std::swap(d, node.d);
|
2019-12-30 16:13:18 +01:00
|
|
|
return *this;
|
2019-12-30 12:00:57 +01:00
|
|
|
}
|
2019-12-30 16:13:18 +01:00
|
|
|
#endif
|
2019-12-30 12:00:57 +01:00
|
|
|
|
2014-08-20 12:33:29 +02:00
|
|
|
//---------------------------------------------------------------------------------------------------------------------
|
|
|
|
VNodeDetail::~VNodeDetail()
|
|
|
|
{}
|
|
|
|
|
|
|
|
//---------------------------------------------------------------------------------------------------------------------
|
|
|
|
quint32 VNodeDetail::getId() const
|
|
|
|
{
|
|
|
|
return d->id;
|
|
|
|
}
|
|
|
|
|
|
|
|
//---------------------------------------------------------------------------------------------------------------------
|
|
|
|
void VNodeDetail::setId(const quint32 &value)
|
|
|
|
{
|
|
|
|
d->id = value;
|
|
|
|
}
|
|
|
|
|
|
|
|
//---------------------------------------------------------------------------------------------------------------------
|
|
|
|
Tool VNodeDetail::getTypeTool() const
|
|
|
|
{
|
|
|
|
return d->typeTool;
|
|
|
|
}
|
|
|
|
|
|
|
|
//---------------------------------------------------------------------------------------------------------------------
|
2015-04-15 14:44:57 +02:00
|
|
|
// cppcheck-suppress unusedFunction
|
2014-08-20 12:33:29 +02:00
|
|
|
void VNodeDetail::setTypeTool(const Tool &value)
|
|
|
|
{
|
|
|
|
d->typeTool = value;
|
|
|
|
}
|
|
|
|
|
|
|
|
//---------------------------------------------------------------------------------------------------------------------
|
|
|
|
NodeDetail VNodeDetail::getTypeNode() const
|
|
|
|
{
|
|
|
|
return d->typeNode;
|
|
|
|
}
|
|
|
|
|
|
|
|
//---------------------------------------------------------------------------------------------------------------------
|
2015-04-15 14:44:57 +02:00
|
|
|
// cppcheck-suppress unusedFunction
|
2014-08-20 12:33:29 +02:00
|
|
|
void VNodeDetail::setTypeNode(const NodeDetail &value)
|
|
|
|
{
|
|
|
|
d->typeNode = value;
|
|
|
|
}
|
|
|
|
|
|
|
|
//---------------------------------------------------------------------------------------------------------------------
|
|
|
|
qreal VNodeDetail::getMx() const
|
|
|
|
{
|
|
|
|
return d->mx;
|
|
|
|
}
|
|
|
|
|
|
|
|
//---------------------------------------------------------------------------------------------------------------------
|
|
|
|
void VNodeDetail::setMx(const qreal &value)
|
|
|
|
{
|
|
|
|
d->mx = value;
|
|
|
|
}
|
|
|
|
|
|
|
|
//---------------------------------------------------------------------------------------------------------------------
|
|
|
|
qreal VNodeDetail::getMy() const
|
|
|
|
{
|
|
|
|
return d->my;
|
|
|
|
}
|
|
|
|
|
|
|
|
//---------------------------------------------------------------------------------------------------------------------
|
|
|
|
void VNodeDetail::setMy(const qreal &value)
|
|
|
|
{
|
|
|
|
d->my = value;
|
|
|
|
}
|
2014-12-17 13:56:14 +01:00
|
|
|
|
|
|
|
//---------------------------------------------------------------------------------------------------------------------
|
|
|
|
bool VNodeDetail::getReverse() const
|
|
|
|
{
|
|
|
|
if (getTypeTool() == Tool::NodePoint)
|
|
|
|
{
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
return d->reverse;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
//---------------------------------------------------------------------------------------------------------------------
|
|
|
|
void VNodeDetail::setReverse(bool reverse)
|
|
|
|
{
|
|
|
|
if (getTypeTool() == Tool::NodePoint)
|
|
|
|
{
|
|
|
|
d->reverse = false;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
d->reverse = reverse;
|
|
|
|
}
|
|
|
|
}
|
2016-12-01 19:47:15 +01:00
|
|
|
|
|
|
|
//---------------------------------------------------------------------------------------------------------------------
|
|
|
|
QVector<VPieceNode> VNodeDetail::Convert(const VContainer *data, const QVector<VNodeDetail> &nodes, qreal width,
|
|
|
|
bool closed)
|
|
|
|
{
|
|
|
|
if (width < 0)
|
|
|
|
{
|
|
|
|
width = 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
VPiecePath path;
|
2018-04-03 13:36:38 +02:00
|
|
|
for (auto &node : nodes)
|
2016-12-01 19:47:15 +01:00
|
|
|
{
|
|
|
|
path.Append(VPieceNode(node.getId(), node.getTypeTool(), node.getReverse()));
|
|
|
|
}
|
|
|
|
|
|
|
|
if (path.PathPoints(data).size() > 2)
|
|
|
|
{
|
|
|
|
for (int i = 0; i < nodes.size(); ++i)
|
|
|
|
{
|
|
|
|
const VNodeDetail &node = nodes.at(i);
|
|
|
|
if (node.getTypeTool() == Tool::NodePoint)
|
|
|
|
{
|
|
|
|
if (not qFuzzyIsNull(node.getMx()) || not qFuzzyIsNull(node.getMy()))
|
|
|
|
{
|
|
|
|
const QPointF previosPoint = path.NodePreviousPoint(data, i);
|
|
|
|
const QPointF nextPoint = path.NodeNextPoint(data, i);
|
|
|
|
|
|
|
|
const QPointF point = data->GeometricObject<VPointF>(node.getId())->toQPointF();
|
|
|
|
|
2017-01-25 10:32:52 +01:00
|
|
|
QLineF lineBefore(point, previosPoint);
|
|
|
|
lineBefore.setAngle(lineBefore.angle()-90);
|
|
|
|
lineBefore.setLength(width);
|
|
|
|
|
|
|
|
ConvertBefore(path[i], lineBefore, node.getMx(), node.getMy());
|
|
|
|
|
|
|
|
QLineF lineAfter(point, nextPoint);
|
|
|
|
lineAfter.setAngle(lineAfter.angle()+90);
|
|
|
|
lineAfter.setLength(width);
|
|
|
|
|
|
|
|
ConvertAfter(path[i], lineAfter, node.getMx(), node.getMy());
|
2016-12-01 19:47:15 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (not closed && path.CountNodes() > 1)
|
|
|
|
{
|
2018-04-12 12:13:19 +02:00
|
|
|
path[0].SetFormulaSABefore(QChar('0'));
|
|
|
|
path[path.CountNodes()-1].SetFormulaSAAfter(QChar('0'));
|
2016-12-01 19:47:15 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
return path.GetNodes();
|
|
|
|
}
|