/************************************************************************ ** ** @file dialogrotation.cpp ** @author Roman Telezhynskyi ** @date 10 4, 2016 ** ** @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) 2016 Valentina project ** 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 . ** *************************************************************************/ #include "dialogrotation.h" #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include "../../visualization/visualization.h" #include "../../visualization/line/operation/vistoolrotation.h" #include "../ifc/xml/vabstractpattern.h" #include "../ifc/xml/vdomdocument.h" #include "../qmuparser/qmudef.h" #include "../support/dialogeditwrongformula.h" #include "../vgeometry/vpointf.h" #include "../vmisc/vabstractapplication.h" #include "../vmisc/vcommonsettings.h" #include "../vpatterndb/vcontainer.h" #include "../vpatterndb/vtranslatevars.h" #include "../vwidgets/vabstractmainwindow.h" #include "../vwidgets/vmaingraphicsscene.h" #include "ui_dialogrotation.h" //--------------------------------------------------------------------------------------------------------------------- DialogRotation::DialogRotation(const VContainer *data, const quint32 &toolId, QWidget *parent) : DialogTool(data, toolId, parent), ui(new Ui::DialogRotation), flagAngle(false), timerAngle(nullptr), formulaAngle(), formulaBaseHeightAngle(0), objects(), stage1(true), m_suffix(), m_firstRelease(false) { ui->setupUi(this); this->formulaBaseHeightAngle = ui->plainTextEditFormula->height(); ui->plainTextEditFormula->installEventFilter(this); ui->lineEditSuffix->setText(qApp->getCurrentDocument()->GenerateSuffix()); timerAngle = new QTimer(this); connect(timerAngle, &QTimer::timeout, this, &DialogRotation::EvalAngle); InitOkCancelApply(ui); FillComboBoxPoints(ui->comboBoxOriginPoint); flagName = true; CheckState(); connect(ui->lineEditSuffix, &QLineEdit::textChanged, this, &DialogRotation::SuffixChanged); connect(ui->toolButtonExprAngle, &QPushButton::clicked, this, &DialogRotation::FXAngle); connect(ui->plainTextEditFormula, &QPlainTextEdit::textChanged, this, &DialogRotation::AngleChanged); connect(ui->pushButtonGrowLength, &QPushButton::clicked, this, &DialogRotation::DeployAngleTextEdit); connect(ui->comboBoxOriginPoint, static_cast(&QComboBox::currentIndexChanged), this, &DialogRotation::PointChanged); vis = new VisToolRotation(data); } //--------------------------------------------------------------------------------------------------------------------- DialogRotation::~DialogRotation() { delete ui; } //--------------------------------------------------------------------------------------------------------------------- quint32 DialogRotation::GetOrigPointId() const { return getCurrentObjectId(ui->comboBoxOriginPoint); } //--------------------------------------------------------------------------------------------------------------------- void DialogRotation::SetOrigPointId(const quint32 &value) { ChangeCurrentData(ui->comboBoxOriginPoint, value); VisToolRotation *operation = qobject_cast(vis); SCASSERT(operation != nullptr) operation->SetOriginPointId(value); } //--------------------------------------------------------------------------------------------------------------------- QString DialogRotation::GetAngle() const { return qApp->TrVars()->TryFormulaFromUser(formulaAngle, qApp->Settings()->GetOsSeparator()); } //--------------------------------------------------------------------------------------------------------------------- void DialogRotation::SetAngle(const QString &value) { formulaAngle = qApp->TrVars()->FormulaToUser(value, qApp->Settings()->GetOsSeparator()); // increase height if needed. if (formulaAngle.length() > 80) { this->DeployAngleTextEdit(); } ui->plainTextEditFormula->setPlainText(formulaAngle); VisToolRotation *operation = qobject_cast(vis); SCASSERT(operation != nullptr) operation->SetAngle(formulaAngle); MoveCursorToEnd(ui->plainTextEditFormula); } //--------------------------------------------------------------------------------------------------------------------- QString DialogRotation::GetSuffix() const { return m_suffix; } //--------------------------------------------------------------------------------------------------------------------- void DialogRotation::SetSuffix(const QString &value) { m_suffix = value; ui->lineEditSuffix->setText(value); } //--------------------------------------------------------------------------------------------------------------------- QVector DialogRotation::GetObjects() const { return objects.toVector(); } //--------------------------------------------------------------------------------------------------------------------- void DialogRotation::ShowDialog(bool click) { if (stage1 && not click) { if (objects.isEmpty()) { return; } stage1 = false; VMainGraphicsScene *scene = qobject_cast(qApp->getCurrentScene()); SCASSERT(scene != nullptr) scene->clearSelection(); VisToolRotation *operation = qobject_cast(vis); SCASSERT(operation != nullptr) operation->SetObjects(objects.toVector()); operation->VisualMode(); scene->ToggleArcSelection(false); scene->ToggleElArcSelection(false); scene->ToggleSplineSelection(false); scene->ToggleSplinePathSelection(false); scene->ToggleArcHover(false); scene->ToggleElArcHover(false); scene->ToggleSplineHover(false); scene->ToggleSplinePathHover(false); emit ToolTip(tr("Select origin point")); } else if (not stage1 && prepare && click) { // The check need to ignore first release of mouse button. // User can select point by clicking on a label. if (not m_firstRelease) { m_firstRelease = true; return; } /*We will ignore click if pointer is in point circle*/ VMainGraphicsScene *scene = qobject_cast(qApp->getCurrentScene()); SCASSERT(scene != nullptr) const QSharedPointer point = data->GeometricObject(GetOrigPointId()); const QLineF line = QLineF(static_cast(*point), scene->getScenePos()); //Radius of point circle, but little bigger. Need handle with hover sizes. if (line.length() <= defPointRadiusPixel*1.5) { return; } VisToolRotation *operation = qobject_cast(vis); SCASSERT(operation != nullptr) SetAngle(operation->Angle());//Show in dialog angle that a user choose setModal(true); emit ToolTip(""); timerAngle->start(); show(); } } //--------------------------------------------------------------------------------------------------------------------- void DialogRotation::ChosenObject(quint32 id, const SceneObject &type) { if (not stage1 && not prepare)// After first choose we ignore all objects { if (type == SceneObject::Point) { if (objects.contains(id)) { emit ToolTip(tr("Select origin point that is not part of the list of objects")); return; } if (SetObject(id, ui->comboBoxOriginPoint, "")) { VisToolRotation *operation = qobject_cast(vis); SCASSERT(operation != nullptr) VAbstractMainWindow *window = qobject_cast(qApp->getMainWindow()); SCASSERT(window != nullptr) connect(operation, &Visualization::ToolTip, window, &VAbstractMainWindow::ShowToolTip); operation->SetOriginPointId(id); operation->RefreshGeometry(); prepare = true; } } } } //--------------------------------------------------------------------------------------------------------------------- void DialogRotation::SelectedObject(bool selected, quint32 object, quint32 tool) { Q_UNUSED(tool) if (stage1) { if (selected) { if (not objects.contains(object)) { objects.append(object); } } else { objects.removeOne(object); } } } //--------------------------------------------------------------------------------------------------------------------- void DialogRotation::DeployAngleTextEdit() { DeployFormula(ui->plainTextEditFormula, ui->pushButtonGrowLength, formulaBaseHeightAngle); } //--------------------------------------------------------------------------------------------------------------------- void DialogRotation::AngleChanged() { labelEditFormula = ui->labelEditAngle; labelResultCalculation = ui->labelResultAngle; ValFormulaChanged(flagAngle, ui->plainTextEditFormula, timerAngle, degreeSymbol); } //--------------------------------------------------------------------------------------------------------------------- void DialogRotation::FXAngle() { DialogEditWrongFormula *dialog = new DialogEditWrongFormula(data, toolId, this); dialog->setWindowTitle(tr("Edit angle")); dialog->SetFormula(GetAngle()); dialog->setPostfix(degreeSymbol); if (dialog->exec() == QDialog::Accepted) { SetAngle(dialog->GetFormula()); } delete dialog; } //--------------------------------------------------------------------------------------------------------------------- void DialogRotation::SuffixChanged() { QLineEdit* edit = qobject_cast(sender()); if (edit) { const QString suffix = edit->text(); if (suffix.isEmpty()) { flagName = false; ChangeColor(ui->labelSuffix, Qt::red); CheckState(); return; } else { if (m_suffix != suffix) { QRegularExpression rx(NameRegExp()); const QStringList uniqueNames = VContainer::AllUniqueNames(); for (int i=0; i < uniqueNames.size(); ++i) { const QString name = uniqueNames.at(i) + suffix; if (not rx.match(name).hasMatch() || not data->IsUnique(name)) { flagName = false; ChangeColor(ui->labelSuffix, Qt::red); CheckState(); return; } } } } flagName = true; ChangeColor(ui->labelSuffix, okColor); } CheckState(); } //--------------------------------------------------------------------------------------------------------------------- void DialogRotation::CheckState() { SCASSERT(bOk != nullptr) bOk->setEnabled(flagAngle && flagName && flagError); SCASSERT(bApply != nullptr) bApply->setEnabled(bOk->isEnabled()); } //--------------------------------------------------------------------------------------------------------------------- void DialogRotation::ShowVisualization() { AddVisualization(); } //--------------------------------------------------------------------------------------------------------------------- void DialogRotation::SaveData() { m_suffix = ui->lineEditSuffix->text(); formulaAngle = ui->plainTextEditFormula->toPlainText(); formulaAngle.replace("\n", " "); VisToolRotation *operation = qobject_cast(vis); SCASSERT(operation != nullptr) operation->SetObjects(objects.toVector()); operation->SetOriginPointId(GetOrigPointId()); operation->SetAngle(formulaAngle); operation->RefreshGeometry(); } //--------------------------------------------------------------------------------------------------------------------- void DialogRotation::closeEvent(QCloseEvent *event) { ui->plainTextEditFormula->blockSignals(true); DialogTool::closeEvent(event); } //--------------------------------------------------------------------------------------------------------------------- void DialogRotation::PointChanged() { QColor color = okColor; if (objects.contains(getCurrentObjectId(ui->comboBoxOriginPoint))) { flagError = false; color = errorColor; } else { flagError = true; color = okColor; } ChangeColor(ui->labelOriginPoint, color); CheckState(); } //--------------------------------------------------------------------------------------------------------------------- void DialogRotation::EvalAngle() { labelEditFormula = ui->labelEditAngle; Eval(ui->plainTextEditFormula->toPlainText(), flagAngle, ui->labelResultAngle, degreeSymbol, false); }