/************************************************************************ ** ** @file mainwindow.cpp ** @author Roman Telezhynskyi ** @date November 15, 2013 ** ** @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) 2013-2015 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 "mainwindow.h" #include "ui_mainwindow.h" #include "../vgeometry/vspline.h" #include "../ifc/exception/vexceptionobjecterror.h" #include "../ifc/exception/vexceptionconversionerror.h" #include "../ifc/exception/vexceptionemptyparameter.h" #include "../ifc/exception/vexceptionwrongid.h" #include "../ifc/exception/vexceptionundo.h" #include "version.h" #include "core/vapplication.h" #include "../vmisc/customevents.h" #include "../vmisc/vsettings.h" #include "../vmisc/def.h" #include "../vmisc/qxtcsvmodel.h" #include "undocommands/renamepp.h" #include "core/vtooloptionspropertybrowser.h" #include "options.h" #include "../ifc/xml/vpatternconverter.h" #include "../vmisc/logging.h" #include "../vformat/vmeasurements.h" #include "../ifc/xml/vvstconverter.h" #include "../ifc/xml/vvitconverter.h" #include "../vwidgets/vwidgetpopup.h" #include "../vwidgets/vmaingraphicsscene.h" #include "tools/drawTools/drawtools.h" #include "../vtools/dialogs/tooldialogs.h" #include "tools/vtoolseamallowance.h" #include "tools/nodeDetails/vtoolpiecepath.h" #include "tools/nodeDetails/vtoolpin.h" #include "tools/nodeDetails/vtoolplacelabel.h" #include "tools/vtooluniondetails.h" #include "dialogs/dialogs.h" #include "dialogs/vwidgetgroups.h" #include "../vtools/undocommands/addgroup.h" #include "dialogs/vwidgetdetails.h" #include "../vpatterndb/vpiecepath.h" #include "../qmuparser/qmuparsererror.h" #include "../vtools/dialogs/support/dialogeditlabel.h" #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #if defined(Q_OS_WIN32) && QT_VERSION >= QT_VERSION_CHECK(5, 7, 0) #include #include #endif #if defined(Q_OS_MAC) #include #include #endif //defined(Q_OS_MAC) QT_WARNING_PUSH QT_WARNING_DISABLE_CLANG("-Wmissing-prototypes") QT_WARNING_DISABLE_INTEL(1418) Q_LOGGING_CATEGORY(vMainWindow, "v.mainwindow") QT_WARNING_POP namespace { Q_GLOBAL_STATIC_WITH_ARGS(const QString, autosavePrefix, (QLatin1String(".autosave"))) // String below need for getting translation for key Ctrl Q_GLOBAL_STATIC_WITH_ARGS(const QString, strQShortcut, (QLatin1String("QShortcut"))) // Context Q_GLOBAL_STATIC_WITH_ARGS(const QString, strCtrl, (QLatin1String("Ctrl"))) // String } //--------------------------------------------------------------------------------------------------------------------- /** * @brief MainWindow constructor. * @param parent parent widget. */ MainWindow::MainWindow(QWidget *parent) :MainWindowsNoGUI(parent), ui(new Ui::MainWindow), watcher(new QFileSystemWatcher(this)), currentTool(Tool::Arrow), lastUsedTool(Tool::Arrow), sceneDraw(nullptr), sceneDetails(nullptr), mouseCoordinate(nullptr), isInitialized(false), mChanges(false), mChangesAsked(true), patternReadOnly(false), dialogTable(nullptr), dialogTool(), dialogHistory(nullptr), dialogFMeasurements(nullptr), comboBoxDraws(nullptr), patternPieceLabel(nullptr), mode(Draw::Calculation), currentDrawIndex(0), currentToolBoxIndex(0), isDockToolOptionsVisible(true), isDockGroupsVisible(true), drawMode(true), recentFileActs(), separatorAct(nullptr), leftGoToStage(nullptr), rightGoToStage(nullptr), autoSaveTimer(nullptr), guiEnabled(true), gradationHeights(nullptr), gradationSizes(nullptr), gradationHeightsLabel(nullptr), gradationSizesLabel(nullptr), zoomScale(nullptr), doubleSpinBoxScale(nullptr), toolOptions(nullptr), groupsWidget(nullptr), detailsWidget(nullptr), lock(nullptr), toolButtonPointerList(), m_progressBar(new QProgressBar(this)), m_statusLabel(new QLabel(this)) { for (int i = 0; i < MaxRecentFiles; ++i) { recentFileActs[i] = nullptr; } CreateActions(); InitScenes(); doc = new VPattern(pattern, &mode, sceneDraw, sceneDetails); connect(doc, &VPattern::ClearMainWindow, this, &MainWindow::Clear); connect(doc, &VPattern::patternChanged, this, &MainWindow::PatternChangesWereSaved); connect(doc, &VPattern::UndoCommand, this, &MainWindow::FullParseFile); connect(doc, &VPattern::SetEnabledGUI, this, &MainWindow::SetEnabledGUI); connect(doc, &VPattern::CheckLayout, this, [this]() { if (pattern->DataPieces()->count() == 0) { if(not ui->actionDraw->isChecked()) { ActionDraw(true); } } }); connect(doc, &VPattern::SetCurrentPP, this, &MainWindow::GlobalChangePP); connect(doc, &VPattern::MadeProgress, this, &MainWindow::ShowProgress); qApp->setCurrentDocument(doc); InitDocksContain(); CreateMenus(); ToolBarDraws(); ToolBarStages(); InitToolButtons(); m_progressBar->setVisible(false); #if defined(Q_OS_WIN32) && QT_VERSION >= QT_VERSION_CHECK(5, 7, 0) m_taskbarProgress->setVisible(false); #endif m_statusLabel->setText(tr("Create new pattern piece to start working.")); statusBar()->addPermanentWidget(m_statusLabel, 1); statusBar()->addPermanentWidget(m_progressBar, 1); ToolBarTools(); connect(qApp->getUndoStack(), &QUndoStack::cleanChanged, this, &MainWindow::PatternChangesWereSaved); InitAutoSave(); ui->toolBox->setCurrentIndex(0); ReadSettings(); setCurrentFile(QString()); WindowsLocale(); connect(ui->listWidget, &QListWidget::currentRowChanged, this, &MainWindow::ShowPaper); ui->dockWidgetLayoutPages->setVisible(false); connect(watcher, &QFileSystemWatcher::fileChanged, this, &MainWindow::MeasurementsChanged); connect(qApp, &QApplication::focusChanged, this, [this](QWidget *old, QWidget *now) { if (old == nullptr && isAncestorOf(now) == true) {// focus IN static bool asking = false; if (not asking && mChanges && not mChangesAsked) { asking = true; mChangesAsked = true; const auto answer = QMessageBox::question(this, tr("Measurements"), tr("Measurements were changed. Do you want to sync measurements now?"), QMessageBox::Yes|QMessageBox::No, QMessageBox::Yes); if (answer == QMessageBox::Yes) { SyncMeasurements(); } asking = false; } } // In case we will need it // else if (isAncestorOf(old) == true && now == nullptr) // focus OUT }); #if defined(Q_OS_MAC) // On Mac deafault icon size is 32x32. ui->toolBarDraws->setIconSize(QSize(24, 24)); ui->toolBarOption->setIconSize(QSize(24, 24)); ui->toolBarStages->setIconSize(QSize(24, 24)); ui->toolBarTools->setIconSize(QSize(24, 24)); setUnifiedTitleAndToolBarOnMac(true); // Mac OS Dock Menu QMenu *menu = new QMenu(this); QAction *actionNewPattern = menu->addAction(tr("New pattern")); actionNewPattern->setMenuRole(QAction::NoRole); connect(actionNewPattern, &QAction::triggered, this, &MainWindow::New); QAction *actionOpenPattern = menu->addAction(tr("Open pattern")); actionOpenPattern->setMenuRole(QAction::NoRole); connect(actionOpenPattern, &QAction::triggered, this, &MainWindow::Open); QAction *actionOpenTape = menu->addAction(tr("Create/Edit measurements")); actionOpenTape->setMenuRole(QAction::NoRole); connect(actionOpenTape, &QAction::triggered, this, &MainWindow::CreateMeasurements); QAction *actionPreferences = menu->addAction(tr("Preferences")); actionPreferences->setMenuRole(QAction::NoRole); connect(actionPreferences, &QAction::triggered, this, &MainWindow::Preferences); menu->setAsDockMenu(); #endif //defined(Q_OS_MAC) } //--------------------------------------------------------------------------------------------------------------------- void MainWindow::AddPP(const QString &PPName) { if (doc->appendPP(PPName) == false) { qCDebug(vMainWindow, "Error creating pattern piece with the name %s.", qUtf8Printable(PPName)); return; } if (comboBoxDraws->count() == 0) { sceneDraw->InitOrigins(); sceneDetails->InitOrigins(); } comboBoxDraws->blockSignals(true); comboBoxDraws->addItem(PPName); pattern->ClearCalculationGObjects(); //Create single point emit ui->view->itemClicked(nullptr);//hide options previous tool const QString label = doc->GenerateLabel(LabelType::NewPatternPiece); const QPointF startPosition = StartPositionNewPP(); VToolBasePointInitData initData; initData.scene = sceneDraw; initData.doc = doc; initData.data = pattern; initData.parse = Document::FullParse; initData.typeCreation = Source::FromGui; initData.x = startPosition.x(); initData.y = startPosition.y(); initData.name = label; initData.nameActivPP = PPName; auto spoint = VToolBasePoint::Create(initData); emit ui->view->itemClicked(spoint); SetEnableTool(true); SetEnableWidgets(true); const qint32 index = comboBoxDraws->findText(PPName); if ( index != -1 ) { // -1 for not found comboBoxDraws->setCurrentIndex(index); } else { comboBoxDraws->setCurrentIndex(0); } comboBoxDraws->blockSignals(false); // Show best for new PP VMainGraphicsView::NewSceneRect(ui->view->scene(), ui->view, spoint); //ZoomFitBestCurrent(); QApplication::postEvent(this, new FitBestCurrentEvent()); ui->actionNewDraw->setEnabled(true); m_statusLabel->setText(QString()); } //--------------------------------------------------------------------------------------------------------------------- QPointF MainWindow::StartPositionNewPP() const { const qreal originX = 30.0; const qreal originY = 40.0; const qreal margin = 40.0; if (comboBoxDraws->count() > 1) { const QRectF rect = sceneDraw->VisibleItemsBoundingRect(); if (rect.width() <= rect.height()) { return QPointF(rect.width()+margin, originY); } else { return QPointF(originX, rect.height()+margin); } } else { return QPointF(originX, originY); } } //--------------------------------------------------------------------------------------------------------------------- void MainWindow::InitScenes() { sceneDraw = new VMainGraphicsScene(this); currentScene = sceneDraw; qApp->setCurrentScene(¤tScene); connect(this, &MainWindow::EnableItemMove, sceneDraw, &VMainGraphicsScene::EnableItemMove); connect(this, &MainWindow::ItemsSelection, sceneDraw, &VMainGraphicsScene::ItemsSelection); connect(this, &MainWindow::EnableLabelSelection, sceneDraw, &VMainGraphicsScene::ToggleLabelSelection); connect(this, &MainWindow::EnablePointSelection, sceneDraw, &VMainGraphicsScene::TogglePointSelection); connect(this, &MainWindow::EnableLineSelection, sceneDraw, &VMainGraphicsScene::ToggleLineSelection); connect(this, &MainWindow::EnableArcSelection, sceneDraw, &VMainGraphicsScene::ToggleArcSelection); connect(this, &MainWindow::EnableElArcSelection, sceneDraw, &VMainGraphicsScene::ToggleElArcSelection); connect(this, &MainWindow::EnableSplineSelection, sceneDraw, &VMainGraphicsScene::ToggleSplineSelection); connect(this, &MainWindow::EnableSplinePathSelection, sceneDraw, &VMainGraphicsScene::ToggleSplinePathSelection); connect(this, &MainWindow::EnableLabelHover, sceneDraw, &VMainGraphicsScene::ToggleLabelHover); connect(this, &MainWindow::EnablePointHover, sceneDraw, &VMainGraphicsScene::TogglePointHover); connect(this, &MainWindow::EnableLineHover, sceneDraw, &VMainGraphicsScene::ToggleLineHover); connect(this, &MainWindow::EnableArcHover, sceneDraw, &VMainGraphicsScene::ToggleArcHover); connect(this, &MainWindow::EnableElArcHover, sceneDraw, &VMainGraphicsScene::ToggleElArcHover); connect(this, &MainWindow::EnableSplineHover, sceneDraw, &VMainGraphicsScene::ToggleSplineHover); connect(this, &MainWindow::EnableSplinePathHover, sceneDraw, &VMainGraphicsScene::ToggleSplinePathHover); connect(sceneDraw, &VMainGraphicsScene::mouseMove, this, &MainWindow::MouseMove); sceneDetails = new VMainGraphicsScene(this); connect(this, &MainWindow::EnableItemMove, sceneDetails, &VMainGraphicsScene::EnableItemMove); connect(this, &MainWindow::EnableNodeLabelSelection, sceneDetails, &VMainGraphicsScene::ToggleNodeLabelSelection); connect(this, &MainWindow::EnableNodePointSelection, sceneDetails, &VMainGraphicsScene::ToggleNodePointSelection); connect(this, &MainWindow::EnableDetailSelection, sceneDetails, &VMainGraphicsScene::ToggleDetailSelection); connect(this, &MainWindow::EnableNodeLabelHover, sceneDetails, &VMainGraphicsScene::ToggleNodeLabelHover); connect(this, &MainWindow::EnableNodePointHover, sceneDetails, &VMainGraphicsScene::ToggleNodePointHover); connect(this, &MainWindow::EnableDetailHover, sceneDetails, &VMainGraphicsScene::ToggleDetailHover); connect(sceneDetails, &VMainGraphicsScene::mouseMove, this, &MainWindow::MouseMove); ui->view->setScene(currentScene); sceneDraw->setTransform(ui->view->transform()); sceneDetails->setTransform(ui->view->transform()); connect(ui->view, &VMainGraphicsView::MouseRelease, this, [this](){EndVisualization(true);}); connect(ui->view, &VMainGraphicsView::ScaleChanged, this, &MainWindow::ScaleChanged); QSizePolicy policy(QSizePolicy::Expanding, QSizePolicy::Expanding); policy.setHorizontalStretch(12); ui->view->setSizePolicy(policy); qApp->setSceneView(ui->view); } //--------------------------------------------------------------------------------------------------------------------- QSharedPointer MainWindow::OpenMeasurementFile(const QString &path) { QSharedPointer m; if (path.isEmpty()) { return m; } try { m = QSharedPointer(new VMeasurements(pattern)); m->SetSize(VContainer::rsize()); m->SetHeight(VContainer::rheight()); m->setXMLContent(path); if (m->Type() == MeasurementsType::Unknown) { VException e(tr("Measurement file has unknown format.")); throw e; } if (m->Type() == MeasurementsType::Multisize) { VVSTConverter converter(path); m->setXMLContent(converter.Convert());// Read again after conversion } else { VVITConverter converter(path); m->setXMLContent(converter.Convert());// Read again after conversion } if (not m->IsDefinedKnownNamesValid()) { VException e(tr("Measurement file contains invalid known measurement(s).")); throw e; } CheckRequiredMeasurements(m.data()); if (m->Type() == MeasurementsType::Multisize) { if (m->MUnit() == Unit::Inch) { qCCritical(vMainWindow, "%s\n\n%s", qUtf8Printable(tr("Wrong units.")), qUtf8Printable(tr("Application doesn't support multisize table with inches."))); m->clear(); if (not VApplication::IsGUIMode()) { qApp->exit(V_EX_DATAERR); } return m; } } } catch (VException &e) { qCCritical(vMainWindow, "%s\n\n%s\n\n%s", qUtf8Printable(tr("File error.")), qUtf8Printable(e.ErrorMessage()), qUtf8Printable(e.DetailedInformation())); m->clear(); if (not VApplication::IsGUIMode()) { qApp->exit(V_EX_NOINPUT); } return m; } return m; } //--------------------------------------------------------------------------------------------------------------------- bool MainWindow::LoadMeasurements(const QString &path) { QSharedPointer m = OpenMeasurementFile(path); if (m->isNull()) { return false; } if (qApp->patternUnit() == Unit::Inch && m->Type() == MeasurementsType::Multisize) { qWarning()<setPatternType(m->Type()); if (m->Type() == MeasurementsType::Individual) { qApp->SetCustomerName(m->Customer()); } ToolBarOption(); pattern->ClearVariables(VarType::Measurement); m->ReadMeasurements(); } catch (VExceptionEmptyParameter &e) { qCCritical(vMainWindow, "%s\n\n%s\n\n%s", qUtf8Printable(tr("File error.")), qUtf8Printable(e.ErrorMessage()), qUtf8Printable(e.DetailedInformation())); if (not VApplication::IsGUIMode()) { qApp->exit(V_EX_NOINPUT); } return false; } if (m->Type() == MeasurementsType::Multisize) { VContainer::SetSize(UnitConvertor(m->BaseSize(), m->MUnit(), *m->GetData()->GetPatternUnit())); VContainer::SetHeight(UnitConvertor(m->BaseHeight(), m->MUnit(), *m->GetData()->GetPatternUnit())); doc->SetPatternWasChanged(true); emit doc->UpdatePatternLabel(); } else if (m->Type() == MeasurementsType::Individual) { SetSizeHeightForIndividualM(); } return true; } //--------------------------------------------------------------------------------------------------------------------- bool MainWindow::UpdateMeasurements(const QString &path, int size, int height) { QSharedPointer m = OpenMeasurementFile(path); if (m->isNull()) { return false; } if (qApp->patternType() != m->Type()) { qCCritical(vMainWindow, "%s", qUtf8Printable(tr("Measurement files types have not match."))); if (not VApplication::IsGUIMode()) { qApp->exit(V_EX_DATAERR); } return false; } try { pattern->ClearVariables(VarType::Measurement); m->ReadMeasurements(); if (m->Type() == MeasurementsType::Individual) { qApp->SetCustomerName(m->Customer()); } } catch (VExceptionEmptyParameter &e) { qCCritical(vMainWindow, "%s\n\n%s\n\n%s", qUtf8Printable(tr("File error.")), qUtf8Printable(e.ErrorMessage()), qUtf8Printable(e.DetailedInformation())); if (not VApplication::IsGUIMode()) { qApp->exit(V_EX_NOINPUT); } return false; } if (m->Type() == MeasurementsType::Multisize) { VContainer::SetSize(size); VContainer::SetHeight(height); doc->SetPatternWasChanged(true); emit doc->UpdatePatternLabel(); } else if (m->Type() == MeasurementsType::Individual) { SetSizeHeightForIndividualM(); } return true; } //--------------------------------------------------------------------------------------------------------------------- void MainWindow::CheckRequiredMeasurements(const VMeasurements *m) { const QSet match = doc->ListMeasurements().toSet().subtract(m->ListAll().toSet()); if (not match.isEmpty()) { QList list = match.toList(); for (int i = 0; i < list.size(); ++i) { list[i] = qApp->TrVars()->MToUser(list.at(i)); } VException e(tr("Measurement file doesn't include all required measurements.")); e.AddMoreInformation(tr("Please, additionally provide: %1").arg(QStringList(list).join(", "))); throw e; } } //--------------------------------------------------------------------------------------------------------------------- /** * @brief SetToolButton set tool and show dialog. * @param checked true if tool button checked. * @param t tool type. * @param cursor path tool cursor icon. * @param toolTip first tooltipe. * @param closeDialogSlot function what handle after close dialog. */ template void MainWindow::SetToolButton(bool checked, Tool t, const QString &cursor, const QString &toolTip, Func closeDialogSlot) { if (checked) { CancelTool(); emit EnableItemMove(false); currentTool = lastUsedTool = t; auto cursorResource = cursor; if (qApp->devicePixelRatio() >= 2) { // Try to load HiDPI versions of the cursors if availible auto cursorHidpiResource = QString(cursor).replace(".png", "@2x.png"); if (QFileInfo::exists(cursorResource)) { cursorResource = cursorHidpiResource; } } QPixmap pixmap(cursorResource); QCursor cur(pixmap, 2, 2); ui->view->viewport()->setCursor(cur); ui->view->setCurrentCursorShape(); // Hack to fix problem with a cursor m_statusLabel->setText(toolTip); ui->view->setShowToolOptions(false); dialogTool = QSharedPointer(new Dialog(pattern, 0, this)); // This check helps to find missed tools in the switch Q_STATIC_ASSERT_X(static_cast(Tool::LAST_ONE_DO_NOT_USE) == 55, "Check if need to extend."); switch(t) { case Tool::Midpoint: dialogTool->Build(t); break; case Tool::PiecePath: case Tool::Pin: case Tool::InsertNode: case Tool::PlaceLabel: dialogTool->SetPiecesList(doc->GetActivePPPieces()); break; default: break; } VMainGraphicsScene *scene = qobject_cast(currentScene); SCASSERT(scene != nullptr) connect(scene, &VMainGraphicsScene::ChoosedObject, dialogTool.data(), &DialogTool::ChosenObject); connect(scene, &VMainGraphicsScene::SelectedObject, dialogTool.data(), &DialogTool::SelectedObject); connect(dialogTool.data(), &DialogTool::DialogClosed, this, closeDialogSlot); connect(dialogTool.data(), &DialogTool::ToolTip, this, &MainWindow::ShowToolTip); emit ui->view->itemClicked(nullptr); } else { if (QToolButton *tButton = qobject_cast< QToolButton * >(this->sender())) { tButton->setChecked(true); } } } //--------------------------------------------------------------------------------------------------------------------- template /** * @brief SetToolButtonWithApply set tool and show dialog. * @param checked true if tool button checked. * @param t tool type. * @param cursor path tool cursor icon. * @param toolTip first tooltipe. * @param closeDialogSlot function to handle close of dialog. * @param applyDialogSlot function to handle apply in dialog. */ void MainWindow::SetToolButtonWithApply(bool checked, Tool t, const QString &cursor, const QString &toolTip, Func closeDialogSlot, Func2 applyDialogSlot) { if (checked) { SetToolButton(checked, t, cursor, toolTip, closeDialogSlot); connect(dialogTool.data(), &DialogTool::DialogApplied, this, applyDialogSlot); } else { if (QToolButton *tButton = qobject_cast< QToolButton * >(this->sender())) { tButton->setChecked(true); } } } //--------------------------------------------------------------------------------------------------------------------- /** * @brief ClosedDialog handle close dialog * @param result result working dialog. */ template void MainWindow::ClosedDialog(int result) { SCASSERT(not dialogTool.isNull()) if (result == QDialog::Accepted) { VMainGraphicsScene *scene = qobject_cast(currentScene); SCASSERT(scene != nullptr) QGraphicsItem *tool = dynamic_cast(DrawTool::Create(dialogTool, scene, doc, pattern)); // Do not check for nullptr! See issue #719. emit ui->view->itemClicked(tool); } ArrowTool(true); } //--------------------------------------------------------------------------------------------------------------------- /** * @brief ClosedDialogWithApply handle close dialog that has apply button * @param result result working dialog. */ template void MainWindow::ClosedDialogWithApply(int result, VMainGraphicsScene *scene) { SCASSERT(not dialogTool.isNull()) if (result == QDialog::Accepted) { ApplyDialog(scene); } // If before Cancel was used Apply we have an item DrawTool *vtool = qobject_cast(dialogTool->GetAssociatedTool());// Don't check for nullptr here if (dialogTool->GetAssociatedTool() != nullptr) { SCASSERT(vtool != nullptr) vtool->DialogLinkDestroy(); connect(vtool, &DrawTool::ToolTip, this, &MainWindow::ShowToolTip); } ArrowTool(true); ui->view->itemClicked(vtool);// Don't check for nullptr here if (vtool) { vtool->setFocus(); } // If insert not to the end of file call lite parse if (doc->getCursor() > 0) { doc->LiteParseTree(Document::LiteParse); if (dialogHistory) { dialogHistory->UpdateHistory(); } } } //--------------------------------------------------------------------------------------------------------------------- /** * @brief ApplyDialog handle apply in dialog */ template void MainWindow::ApplyDialog(VMainGraphicsScene *scene) { SCASSERT(not dialogTool.isNull()) // Only create tool if not already created with apply if (dialogTool->GetAssociatedTool() == nullptr) { SCASSERT(scene != nullptr) dialogTool->SetAssociatedTool(DrawTool::Create(dialogTool, scene, doc, pattern)); } else { // Or update associated tool with data DrawTool * vtool = qobject_cast(dialogTool->GetAssociatedTool()); SCASSERT(vtool != nullptr) vtool->FullUpdateFromGuiApply(); } } //--------------------------------------------------------------------------------------------------------------------- template void MainWindow::ClosedDrawDialogWithApply(int result) { ClosedDialogWithApply(result, sceneDraw); } //--------------------------------------------------------------------------------------------------------------------- template void MainWindow::ApplyDrawDialog() { ApplyDialog(sceneDraw); } //--------------------------------------------------------------------------------------------------------------------- template void MainWindow::ClosedDetailsDialogWithApply(int result) { ClosedDialogWithApply(result, sceneDetails); } //--------------------------------------------------------------------------------------------------------------------- template void MainWindow::ApplyDetailsDialog() { ApplyDialog(sceneDetails); } //--------------------------------------------------------------------------------------------------------------------- /** * @brief ToolEndLine handler tool endLine. * @param checked true - button checked. */ void MainWindow::ToolEndLine(bool checked) { ToolSelectPointByRelease(); SetToolButtonWithApply(checked, Tool::EndLine, ":/cursor/endline_cursor.png", tr("Select point"), &MainWindow::ClosedDrawDialogWithApply, &MainWindow::ApplyDrawDialog); } //--------------------------------------------------------------------------------------------------------------------- /** * @brief ToolLine handler tool line. * @param checked true - button checked. */ void MainWindow::ToolLine(bool checked) { ToolSelectPointByRelease(); SetToolButtonWithApply(checked, Tool::Line, ":/cursor/line_cursor.png", tr("Select first point"), &MainWindow::ClosedDrawDialogWithApply, &MainWindow::ApplyDrawDialog); } //--------------------------------------------------------------------------------------------------------------------- /** * @brief ToolAlongLine handler tool alongLine. * @param checked true - button checked. */ void MainWindow::ToolAlongLine(bool checked) { ToolSelectPointByRelease(); SetToolButtonWithApply(checked, Tool::AlongLine, ":/cursor/alongline_cursor.png", tr("Select point"), &MainWindow::ClosedDrawDialogWithApply, &MainWindow::ApplyDrawDialog); } //--------------------------------------------------------------------------------------------------------------------- void MainWindow::ToolMidpoint(bool checked) { ToolSelectPointByRelease(); // Reuse DialogAlongLine and VToolAlongLine but with different cursor SetToolButtonWithApply(checked, Tool::Midpoint, ":/cursor/midpoint_cursor.png", tr("Select point"), &MainWindow::ClosedDrawDialogWithApply, &MainWindow::ApplyDrawDialog); } //--------------------------------------------------------------------------------------------------------------------- /** * @brief ToolShoulderPoint handler tool shoulderPoint. * @param checked true - button checked. */ void MainWindow::ToolShoulderPoint(bool checked) { ToolSelectPointByRelease(); SetToolButtonWithApply(checked, Tool::ShoulderPoint, ":/cursor/shoulder_cursor.png", tr("Select point"), &MainWindow::ClosedDrawDialogWithApply, &MainWindow::ApplyDrawDialog); } //--------------------------------------------------------------------------------------------------------------------- /** * @brief ToolNormal handler tool normal. * @param checked true - button checked. */ void MainWindow::ToolNormal(bool checked) { ToolSelectPointByRelease(); SetToolButtonWithApply(checked, Tool::Normal, ":/cursor/normal_cursor.png", tr("Select first point of line"), &MainWindow::ClosedDrawDialogWithApply, &MainWindow::ApplyDrawDialog); } //--------------------------------------------------------------------------------------------------------------------- /** * @brief ToolBisector handler tool bisector. * @param checked true - button checked. */ void MainWindow::ToolBisector(bool checked) { ToolSelectPointByRelease(); SetToolButtonWithApply(checked, Tool::Bisector, ":/cursor/bisector_cursor.png", tr("Select first point of angle"), &MainWindow::ClosedDrawDialogWithApply, &MainWindow::ApplyDrawDialog); } //--------------------------------------------------------------------------------------------------------------------- /** * @brief ToolLineIntersect handler tool lineIntersect. * @param checked true - button checked. */ void MainWindow::ToolLineIntersect(bool checked) { ToolSelectPointByRelease(); SetToolButtonWithApply(checked, Tool::LineIntersect, ":/cursor/intersect_cursor.png", tr("Select first point of first line"), &MainWindow::ClosedDrawDialogWithApply, &MainWindow::ApplyDrawDialog); } //--------------------------------------------------------------------------------------------------------------------- /** * @brief ToolSpline handler tool spline. * @param checked true - button checked. */ void MainWindow::ToolSpline(bool checked) { ToolSelectPointByPress(); SetToolButtonWithApply(checked, Tool::Spline, ":/cursor/spline_cursor.png", tr("Select first point curve"), &MainWindow::ClosedDrawDialogWithApply, &MainWindow::ApplyDrawDialog); } //--------------------------------------------------------------------------------------------------------------------- void MainWindow::ToolCubicBezier(bool checked) { ToolSelectPointByRelease(); SetToolButtonWithApply(checked, Tool::CubicBezier, ":/cursor/cubic_bezier_cursor.png", tr("Select first curve point"), &MainWindow::ClosedDrawDialogWithApply, &MainWindow::ApplyDrawDialog); } //--------------------------------------------------------------------------------------------------------------------- /** * @brief ToolCutSpline handler tool CutSpline. * @param checked true - button is checked */ void MainWindow::ToolCutSpline(bool checked) { ToolSelectSpline(); SetToolButtonWithApply(checked, Tool::CutSpline, ":/cursor/spline_cut_point_cursor.png", tr("Select simple curve"), &MainWindow::ClosedDrawDialogWithApply, &MainWindow::ApplyDrawDialog); } //--------------------------------------------------------------------------------------------------------------------- /** * @brief ToolArc handler tool arc. * @param checked true - button checked. */ void MainWindow::ToolArc(bool checked) { ToolSelectPointByRelease(); SetToolButtonWithApply(checked, Tool::Arc, ":/cursor/arc_cursor.png", tr("Select point of center of arc"), &MainWindow::ClosedDrawDialogWithApply, &MainWindow::ApplyDrawDialog); } //--------------------------------------------------------------------------------------------------------------------- /** * @brief ToolEllipticalArc handler tool arc. * @param checked true - button checked. */ void MainWindow::ToolEllipticalArc(bool checked) { ToolSelectPointByRelease(); SetToolButtonWithApply(checked, Tool::EllipticalArc, ":/cursor/el_arc_cursor.png", tr("Select point of center of elliptical arc"), &MainWindow::ClosedDrawDialogWithApply, &MainWindow::ApplyDrawDialog); } //--------------------------------------------------------------------------------------------------------------------- /** * @brief ToolSplinePath handler tool splinePath. * @param checked true - button checked. */ void MainWindow::ToolSplinePath(bool checked) { ToolSelectPointByPress(); SetToolButtonWithApply(checked, Tool::SplinePath, ":/cursor/splinepath_cursor.png", tr("Select point of curve path"), &MainWindow::ClosedDrawDialogWithApply, &MainWindow::ApplyDrawDialog); } //--------------------------------------------------------------------------------------------------------------------- void MainWindow::ToolCubicBezierPath(bool checked) { ToolSelectPointByRelease(); SetToolButtonWithApply(checked, Tool::CubicBezierPath, ":/cursor/cubic_bezier_path_cursor.png", tr("Select point of cubic bezier path"), &MainWindow::ClosedDrawDialogWithApply, &MainWindow::ApplyDrawDialog); } //--------------------------------------------------------------------------------------------------------------------- /** * @brief ToolCutSplinePath handler tool CutSplinePath. * @param checked true - button is checked */ void MainWindow::ToolCutSplinePath(bool checked) { ToolSelectSplinePath(); SetToolButtonWithApply(checked, Tool::CutSplinePath, ":/cursor/splinepath_cut_point_cursor.png", tr("Select curve path"), &MainWindow::ClosedDrawDialogWithApply, &MainWindow::ApplyDrawDialog); } //--------------------------------------------------------------------------------------------------------------------- /** * @brief ToolPointOfContact handler tool pointOfContact. * @param checked true - button checked. */ void MainWindow::ToolPointOfContact(bool checked) { ToolSelectPointByRelease(); SetToolButtonWithApply(checked, Tool::PointOfContact, ":/cursor/pointcontact_cursor.png", tr("Select first point of line"), &MainWindow::ClosedDrawDialogWithApply, &MainWindow::ApplyDrawDialog); } //--------------------------------------------------------------------------------------------------------------------- /** * @brief ToolDetail handler tool detail. * @param checked true - button checked. */ void MainWindow::ToolDetail(bool checked) { ToolSelectAllDrawObjects(); SetToolButtonWithApply(checked, Tool::Piece, "://cursor/new_detail_cursor.png", tr("Select main path objects clockwise."), &MainWindow::ClosedDetailsDialogWithApply, &MainWindow::ApplyDetailsDialog); } //--------------------------------------------------------------------------------------------------------------------- void MainWindow::ToolPiecePath(bool checked) { ToolSelectAllDrawObjects(); SetToolButton(checked, Tool::PiecePath, "://cursor/path_cursor.png", tr("Select path objects, Shift - reverse direction curve"), &MainWindow::ClosedDialogPiecePath); } //--------------------------------------------------------------------------------------------------------------------- void MainWindow::ToolPin(bool checked) { ToolSelectPointByRelease(); SetToolButton(checked, Tool::Pin, "://cursor/pin_cursor.png", tr("Select pin point"), &MainWindow::ClosedDialogPin); } //--------------------------------------------------------------------------------------------------------------------- void MainWindow::ToolPlaceLabel(bool checked) { ToolSelectPointByRelease(); SetToolButton(checked, Tool::PlaceLabel, "://cursor/place_label_cursor.png", tr("Select placelabel center point"), &MainWindow::ClosedDialogPlaceLabel); } //--------------------------------------------------------------------------------------------------------------------- /** * @brief ToolHeight handler tool height. * @param checked true - button checked. */ void MainWindow::ToolHeight(bool checked) { ToolSelectPointByRelease(); SetToolButtonWithApply(checked, Tool::Height, ":/cursor/height_cursor.png", tr("Select base point"), &MainWindow::ClosedDrawDialogWithApply, &MainWindow::ApplyDrawDialog); } //--------------------------------------------------------------------------------------------------------------------- /** * @brief ToolTriangle handler tool triangle. * @param checked true - button checked. */ void MainWindow::ToolTriangle(bool checked) { ToolSelectPointByRelease(); SetToolButtonWithApply(checked, Tool::Triangle, ":/cursor/triangle_cursor.png", tr("Select first point of axis"), &MainWindow::ClosedDrawDialogWithApply, &MainWindow::ApplyDrawDialog); } //--------------------------------------------------------------------------------------------------------------------- /** * @brief ToolPointOfIntersection handler tool pointOfIntersection. * @param checked true - button checked. */ void MainWindow::ToolPointOfIntersection(bool checked) { ToolSelectPointByRelease(); SetToolButtonWithApply(checked, Tool::PointOfIntersection, ":/cursor/pointofintersect_cursor.png", tr("Select point for X value (vertical)"), &MainWindow::ClosedDrawDialogWithApply, &MainWindow::ApplyDrawDialog); } //--------------------------------------------------------------------------------------------------------------------- /** * @brief ToolUnionDetails handler tool unionDetails. * @param checked true - button checked. */ void MainWindow::ToolUnionDetails(bool checked) { ToolSelectDetail(); SetToolButton(checked, Tool::UnionDetails, ":/cursor/union_cursor.png", tr("Select detail"), &MainWindow::ClosedDialogUnionDetails); } //--------------------------------------------------------------------------------------------------------------------- /** * @brief ClosedDialogUnionDetails actions after closing DialogUnionDetails. * @param result result of dialog working. */ void MainWindow::ClosedDialogUnionDetails(int result) { ClosedDialog(result);// Avoid error: Template function as signal or slot } //--------------------------------------------------------------------------------------------------------------------- void MainWindow::ToolDuplicateDetail(bool checked) { ToolSelectDetail(); SetToolButton(checked, Tool::DuplicateDetail, ":/cursor/duplicate_detail_cursor.png", tr("Select detail"), &MainWindow::ClosedDialogDuplicateDetail); } //--------------------------------------------------------------------------------------------------------------------- void MainWindow::ClosedDialogDuplicateDetail(int result) { SCASSERT(not dialogTool.isNull()) if (result == QDialog::Accepted) { VMainGraphicsScene *scene = qobject_cast(currentScene); SCASSERT(scene != nullptr) QGraphicsItem *tool = dynamic_cast(VToolSeamAllowance::Duplicate(dialogTool, scene, doc)); // Do not check for nullptr! See issue #719. emit ui->view->itemClicked(tool); } ArrowTool(true); } //--------------------------------------------------------------------------------------------------------------------- void MainWindow::ToolGroup(bool checked) { ToolSelectGroupObjects(); const QString tooltip = tr("Select one or more objects, hold %1 - for multiple selection, " "Enter - finish creation") .arg(QCoreApplication::translate(strQShortcut->toUtf8().constData(), strCtrl->toUtf8().constData())); SetToolButton(checked, Tool::Group, ":/cursor/group_plus_cursor.png", tooltip, &MainWindow::ClosedDialogGroup); } //--------------------------------------------------------------------------------------------------------------------- void MainWindow::ToolRotation(bool checked) { ToolSelectOperationObjects(); const QString tooltip = tr("Select one or more objects, hold %1 - for multiple selection, " "Enter - confirm selection") .arg(QCoreApplication::translate(strQShortcut->toUtf8().constData(), strCtrl->toUtf8().constData())); SetToolButtonWithApply(checked, Tool::Rotation, ":/cursor/rotation_cursor.png", tooltip, &MainWindow::ClosedDrawDialogWithApply, &MainWindow::ApplyDrawDialog); } //--------------------------------------------------------------------------------------------------------------------- void MainWindow::ToolFlippingByLine(bool checked) { ToolSelectOperationObjects(); const QString tooltip = tr("Select one or more objects, hold %1 - for multiple selection, " "Enter - confirm selection") .arg(QCoreApplication::translate(strQShortcut->toUtf8().constData(), strCtrl->toUtf8().constData())); SetToolButtonWithApply(checked, Tool::FlippingByLine, ":/cursor/flipping_line_cursor.png", tooltip, &MainWindow::ClosedDrawDialogWithApply, &MainWindow::ApplyDrawDialog); } //--------------------------------------------------------------------------------------------------------------------- void MainWindow::ToolFlippingByAxis(bool checked) { ToolSelectOperationObjects(); const QString tooltip = tr("Select one or more objects, hold %1 - for multiple selection, " "Enter - confirm selection") .arg(QCoreApplication::translate(strQShortcut->toUtf8().constData(), strCtrl->toUtf8().constData())); SetToolButtonWithApply(checked, Tool::FlippingByAxis, ":/cursor/flipping_axis_cursor.png", tooltip, &MainWindow::ClosedDrawDialogWithApply, &MainWindow::ApplyDrawDialog); } //--------------------------------------------------------------------------------------------------------------------- void MainWindow::ToolMove(bool checked) { ToolSelectOperationObjects(); const QString tooltip = tr("Select one or more objects, hold %1 - for multiple selection, " "Enter - confirm selection") .arg(QCoreApplication::translate(strQShortcut->toUtf8().constData(), strCtrl->toUtf8().constData())); SetToolButtonWithApply(checked, Tool::Move, ":/cursor/move_cursor.png", tooltip, &MainWindow::ClosedDrawDialogWithApply, &MainWindow::ApplyDrawDialog); } //--------------------------------------------------------------------------------------------------------------------- void MainWindow::ClosedDialogGroup(int result) { SCASSERT(dialogTool != nullptr) if (result == QDialog::Accepted) { QSharedPointer dialog = dialogTool.objectCast(); SCASSERT(dialog != nullptr) const QDomElement group = doc->CreateGroup(VContainer::getNextId(), dialog->GetName(), dialog->GetGroup()); if (not group.isNull()) { AddGroup *addGroup = new AddGroup(group, doc); connect(addGroup, &AddGroup::UpdateGroups, groupsWidget, &VWidgetGroups::UpdateGroups); qApp->getUndoStack()->push(addGroup); } } ArrowTool(true); } //--------------------------------------------------------------------------------------------------------------------- void MainWindow::ClosedDialogPiecePath(int result) { SCASSERT(dialogTool != nullptr); if (result == QDialog::Accepted) { VToolPiecePath::Create(dialogTool, sceneDetails, doc, pattern); } ArrowTool(true); doc->LiteParseTree(Document::LiteParse); } //--------------------------------------------------------------------------------------------------------------------- void MainWindow::ClosedDialogPin(int result) { SCASSERT(dialogTool != nullptr); if (result == QDialog::Accepted) { VToolPin::Create(dialogTool, doc, pattern); } ArrowTool(true); doc->LiteParseTree(Document::LiteParse); } //--------------------------------------------------------------------------------------------------------------------- void MainWindow::ClosedDialogPlaceLabel(int result) { SCASSERT(dialogTool != nullptr); if (result == QDialog::Accepted) { VToolPlaceLabel::Create(dialogTool, doc, pattern); } ArrowTool(true); doc->LiteParseTree(Document::LiteParse); } //--------------------------------------------------------------------------------------------------------------------- void MainWindow::ClosedDialogInsertNode(int result) { SCASSERT(dialogTool != nullptr); if (result == QDialog::Accepted) { QSharedPointer dTool = dialogTool.objectCast(); SCASSERT(dTool != nullptr) VToolSeamAllowance::InsertNode(dTool->GetNode(), dTool->GetPieceId(), sceneDetails, pattern, doc); } ArrowTool(true); doc->LiteParseTree(Document::LiteParse); } //--------------------------------------------------------------------------------------------------------------------- void MainWindow::ZoomFitBestCurrent() { if(drawMode) { const QRectF rect = doc->ActiveDrawBoundingRect(); if (rect.isEmpty()) { return; } ui->view->fitInView(rect, Qt::KeepAspectRatio); QTransform transform = ui->view->transform(); qreal factor = transform.m11(); factor = qMax(factor, VMainGraphicsView::MinScale()); factor = qMin(factor, VMainGraphicsView::MaxScale()); transform.setMatrix(factor, transform.m12(), transform.m13(), transform.m21(), factor, transform.m23(), transform.m31(), transform.m32(), transform.m33()); ui->view->setTransform(transform); } } //--------------------------------------------------------------------------------------------------------------------- /** * @brief ToolCutArc handler tool cutArc. * @param checked true - button checked. */ void MainWindow::ToolCutArc(bool checked) { ToolSelectArc(); SetToolButtonWithApply(checked, Tool::CutArc, ":/cursor/arc_cut_cursor.png", tr("Select arc"), &MainWindow::ClosedDrawDialogWithApply, &MainWindow::ApplyDrawDialog); } //--------------------------------------------------------------------------------------------------------------------- void MainWindow::ToolLineIntersectAxis(bool checked) { ToolSelectPointByRelease(); SetToolButtonWithApply(checked, Tool::LineIntersectAxis, ":/cursor/line_intersect_axis_cursor.png", tr("Select first point of line"), &MainWindow::ClosedDrawDialogWithApply, &MainWindow::ApplyDrawDialog); } //--------------------------------------------------------------------------------------------------------------------- void MainWindow::ToolCurveIntersectAxis(bool checked) { ToolSelectAllDrawObjects(); SetToolButtonWithApply(checked, Tool::CurveIntersectAxis, ":/cursor/curve_intersect_axis_cursor.png", tr("Select curve"), &MainWindow::ClosedDrawDialogWithApply, &MainWindow::ApplyDrawDialog); } //--------------------------------------------------------------------------------------------------------------------- void MainWindow::ToolArcIntersectAxis(bool checked) { ToolSelectAllDrawObjects(); // Reuse ToolCurveIntersectAxis but with different cursor and tool tip SetToolButtonWithApply(checked, Tool::ArcIntersectAxis, ":/cursor/arc_intersect_axis_cursor.png", tr("Select arc"), &MainWindow::ClosedDrawDialogWithApply, &MainWindow::ApplyDrawDialog); } //--------------------------------------------------------------------------------------------------------------------- void MainWindow::ToolPointOfIntersectionArcs(bool checked) { ToolSelectArc(); SetToolButtonWithApply(checked, Tool::PointOfIntersectionArcs, "://cursor/point_of_intersection_arcs.png", tr("Select first an arc"), &MainWindow::ClosedDrawDialogWithApply, &MainWindow::ApplyDrawDialog); } //--------------------------------------------------------------------------------------------------------------------- void MainWindow::ToolPointOfIntersectionCircles(bool checked) { ToolSelectPointByRelease(); SetToolButtonWithApply(checked, Tool::PointOfIntersectionCircles, "://cursor/point_of_intersection_circles.png", tr("Select first circle center"), &MainWindow::ClosedDrawDialogWithApply, &MainWindow::ApplyDrawDialog); } //--------------------------------------------------------------------------------------------------------------------- void MainWindow::ToolPointOfIntersectionCurves(bool checked) { ToolSelectCurve(); SetToolButtonWithApply(checked, Tool::PointOfIntersectionCurves, "://cursor/intersection_curves_cursor.png", tr("Select first curve"), &MainWindow::ClosedDrawDialogWithApply, &MainWindow::ApplyDrawDialog); } //--------------------------------------------------------------------------------------------------------------------- void MainWindow::ToolPointFromCircleAndTangent(bool checked) { ToolSelectPointByRelease(); SetToolButtonWithApply(checked, Tool::PointFromCircleAndTangent, "://cursor/point_from_circle_and_tangent_cursor.png", tr("Select point on tangent"), &MainWindow::ClosedDrawDialogWithApply, &MainWindow::ApplyDrawDialog); } //--------------------------------------------------------------------------------------------------------------------- void MainWindow::ToolPointFromArcAndTangent(bool checked) { ToolSelectPointArc(); SetToolButtonWithApply(checked, Tool::PointFromArcAndTangent, "://cursor/point_from_arc_and_tangent_cursor.png", tr("Select point on tangent"), &MainWindow::ClosedDrawDialogWithApply, &MainWindow::ApplyDrawDialog); } //--------------------------------------------------------------------------------------------------------------------- void MainWindow::ToolArcWithLength(bool checked) { ToolSelectPointByRelease(); SetToolButtonWithApply(checked, Tool::ArcWithLength, "://cursor/arc_with_length_cursor.png", tr("Select point of the center of the arc"), &MainWindow::ClosedDrawDialogWithApply, &MainWindow::ApplyDrawDialog); } //--------------------------------------------------------------------------------------------------------------------- void MainWindow::ToolTrueDarts(bool checked) { ToolSelectPointByRelease(); SetToolButtonWithApply(checked, Tool::TrueDarts, "://cursor/true_darts_cursor.png", tr("Select the first base line point"), &MainWindow::ClosedDrawDialogWithApply, &MainWindow::ApplyDrawDialog); } //--------------------------------------------------------------------------------------------------------------------- void MainWindow::ToolInsertNode(bool checked) { ToolSelectAllDrawObjects(); SetToolButton(checked, Tool::InsertNode, "://cursor/insert_node_cursor.png", tr("Select an item to insert"), &MainWindow::ClosedDialogInsertNode); } //--------------------------------------------------------------------------------------------------------------------- /** * @brief ShowTool highlight tool.Tip show tools tooltip. * @param toolTip tooltip text. */ void MainWindow::ShowToolTip(const QString &toolTip) { m_statusLabel->setText(toolTip); } //--------------------------------------------------------------------------------------------------------------------- /** * @brief triggers the update of the visibility groups */ void MainWindow::UpdateVisibilityGroups() { groupsWidget->UpdateGroups(); } //--------------------------------------------------------------------------------------------------------------------- /** * @brief triggers the update of the details list */ void MainWindow::UpdateDetailsList() { detailsWidget->UpdateList(); } //--------------------------------------------------------------------------------------------------------------------- /** * @brief showEvent handle after show window. * @param event show event. */ void MainWindow::showEvent( QShowEvent *event ) { QMainWindow::showEvent( event ); if ( event->spontaneous() ) { return; } #if defined(Q_OS_WIN32) && QT_VERSION >= QT_VERSION_CHECK(5, 7, 0) m_taskbarButton->setWindow(windowHandle()); #endif if (isInitialized) { return; } // do your init stuff here MinimumScrollBar(); isInitialized = true;//first show windows are held } //--------------------------------------------------------------------------------------------------------------------- void MainWindow::changeEvent(QEvent *event) { if (event->type() == QEvent::LanguageChange) { // retranslate designer form (single inheritance approach) ui->retranslateUi(this); undoAction->setText(tr("&Undo")); redoAction->setText(tr("&Redo")); statusBar()->showMessage(tr("Changes applied."), 5000); patternPieceLabel->setText(tr("Pattern Piece:")); UpdateWindowTitle(); emit sceneDetails->LanguageChanged(); } // remember to call base class implementation QMainWindow::changeEvent(event); } //--------------------------------------------------------------------------------------------------------------------- /** * @brief closeEvent handle after close window. * @param event close event. */ void MainWindow::closeEvent(QCloseEvent *event) { #if defined(Q_OS_MAC) && QT_VERSION < QT_VERSION_CHECK(5, 11, 1) // Workaround for Qt bug https://bugreports.qt.io/browse/QTBUG-43344 static int numCalled = 0; if (numCalled++ >= 1) { return; } #endif qCDebug(vMainWindow, "Closing main window"); if (MaybeSave()) { FileClosedCorrect(); event->accept(); QApplication::closeAllWindows(); } else { qCDebug(vMainWindow, "Closing canceled."); event->ignore(); } } //--------------------------------------------------------------------------------------------------------------------- void MainWindow::customEvent(QEvent *event) { if (event->type() == UNDO_EVENT) { qApp->getUndoStack()->undo(); } else if (event->type() == FIT_BEST_CURRENT_EVENT) { ZoomFitBestCurrent(); } } //--------------------------------------------------------------------------------------------------------------------- void MainWindow::CleanLayout() { qDeleteAll (scenes); scenes.clear(); shadows.clear(); papers.clear(); ui->listWidget->clear(); SetLayoutModeActions(); } //--------------------------------------------------------------------------------------------------------------------- void MainWindow::PrepareSceneList() { for (int i=1; i<=scenes.size(); ++i) { QListWidgetItem *item = new QListWidgetItem(ScenePreview(i-1), QString::number(i)); ui->listWidget->addItem(item); } if (not scenes.isEmpty()) { ui->listWidget->setCurrentRow(0); SetLayoutModeActions(); } } //--------------------------------------------------------------------------------------------------------------------- void MainWindow::ExportToCSVData(const QString &fileName, bool withHeader, int mib, const QChar &separator) { QxtCsvModel csv; csv.insertColumn(0); csv.insertColumn(1); csv.insertColumn(2); if (withHeader) { csv.setHeaderText(0, tr("Name")); csv.setHeaderText(1, tr("The calculated value")); csv.setHeaderText(2, tr("Formula")); } const QMap > increments = pattern->DataIncrements(); qint32 currentRow = -1; auto SavePreviewCalculation = [¤tRow, &csv, increments](bool save) { QMap >::const_iterator i; QMap map; //Sorting QHash by id for (i = increments.constBegin(); i != increments.constEnd(); ++i) { const QSharedPointer incr = i.value(); if (incr->IsPreviewCalculation() == save) { map.insert(incr->getIndex(), i.key()); } } QMapIterator iMap(map); while (iMap.hasNext()) { iMap.next(); QSharedPointer incr = increments.value(iMap.value()); currentRow++; csv.insertRow(currentRow); csv.setText(currentRow, 0, incr->GetName()); // name csv.setText(currentRow, 1, qApp->LocaleToString(*incr->GetValue())); // calculated value QString formula = VTranslateVars::TryFormulaToUser(incr->GetFormula(), qApp->Settings()->GetOsSeparator()); csv.setText(currentRow, 2, formula); // formula } }; SavePreviewCalculation(false); SavePreviewCalculation(true); QString error; csv.toCSV(fileName, error, withHeader, separator, QTextCodec::codecForMib(mib)); } //--------------------------------------------------------------------------------------------------------------------- void MainWindow::ScaleChanged(qreal scale) { if (not doubleSpinBoxScale.isNull()) { doubleSpinBoxScale->blockSignals(true); doubleSpinBoxScale->setMaximum(qFloor(VMainGraphicsView::MaxScale()*1000)/10.0); doubleSpinBoxScale->setMinimum(qFloor(VMainGraphicsView::MinScale()*1000)/10.0); doubleSpinBoxScale->setValue(qFloor(scale*1000)/10.0); doubleSpinBoxScale->setSingleStep(1); doubleSpinBoxScale->blockSignals(false); } } //--------------------------------------------------------------------------------------------------------------------- void MainWindow::LoadIndividual() { const QString filter = tr("Individual measurements") + QLatin1String(" (*.vit);;") + tr("Multisize measurements") + QLatin1String(" (*.vst)"); //Use standard path to individual measurements const QString path = qApp->ValentinaSettings()->GetPathIndividualMeasurements(); bool usedNotExistedDir = false; QDir directory(path); if (not directory.exists()) { usedNotExistedDir = directory.mkpath(QChar('.')); } const QString mPath = QFileDialog::getOpenFileName(this, tr("Open file"), path, filter, nullptr #ifdef Q_OS_LINUX , QFileDialog::DontUseNativeDialog #endif ); if (not mPath.isEmpty()) { if (LoadMeasurements(mPath)) { if (not doc->MPath().isEmpty()) { watcher->removePath(AbsoluteMPath(qApp->GetPPath(), doc->MPath())); } ui->actionUnloadMeasurements->setEnabled(true); doc->SetMPath(RelativeMPath(qApp->GetPPath(), mPath)); watcher->addPath(mPath); PatternChangesWereSaved(false); ui->actionEditCurrent->setEnabled(true); statusBar()->showMessage(tr("Measurements loaded"), 5000); doc->LiteParseTree(Document::LiteParse); UpdateWindowTitle(); } } if (usedNotExistedDir) { QDir directory(path); directory.rmpath(QChar('.')); } } //--------------------------------------------------------------------------------------------------------------------- void MainWindow::LoadMultisize() { const QString filter = tr("Multisize measurements") + QLatin1String(" (*.vst);;") + tr("Individual measurements") + QLatin1String("(*.vit)"); //Use standard path to multisize measurements QString path = qApp->ValentinaSettings()->GetPathMultisizeMeasurements(); path = VCommonSettings::PrepareMultisizeTables(path); const QString mPath = QFileDialog::getOpenFileName(this, tr("Open file"), path, filter, nullptr #ifdef Q_OS_LINUX , QFileDialog::DontUseNativeDialog #endif ); if (not mPath.isEmpty()) { QString hText; if (not gradationHeights.isNull()) { hText = gradationHeights->currentText(); } QString sText; if (not gradationSizes.isNull()) { sText = gradationSizes->currentText(); } if(LoadMeasurements(mPath)) { if (not doc->MPath().isEmpty()) { watcher->removePath(AbsoluteMPath(qApp->GetPPath(), doc->MPath())); } ui->actionUnloadMeasurements->setEnabled(true); doc->SetMPath(RelativeMPath(qApp->GetPPath(), mPath)); watcher->addPath(mPath); PatternChangesWereSaved(false); ui->actionEditCurrent->setEnabled(true); statusBar()->showMessage(tr("Measurements loaded"), 5000); doc->LiteParseTree(Document::LiteParse); UpdateWindowTitle(); if (qApp->patternType() == MeasurementsType::Multisize) { if (not hText.isEmpty() && not gradationHeights.isNull()) { gradationHeights->setCurrentText(hText); } if (not sText.isEmpty() && not gradationSizes.isNull()) { gradationSizes->setCurrentText(sText); } } } } } //--------------------------------------------------------------------------------------------------------------------- void MainWindow::UnloadMeasurements() { if (doc->MPath().isEmpty()) { ui->actionUnloadMeasurements->setDisabled(true); return; } if (doc->ListMeasurements().isEmpty()) { watcher->removePath(AbsoluteMPath(qApp->GetPPath(), doc->MPath())); if (qApp->patternType() == MeasurementsType::Multisize) { ToolBarOption(); } qApp->setPatternType(MeasurementsType::Unknown); doc->SetMPath(QString()); emit doc->UpdatePatternLabel(); PatternChangesWereSaved(false); ui->actionEditCurrent->setEnabled(false); ui->actionUnloadMeasurements->setDisabled(true); statusBar()->showMessage(tr("Measurements unloaded"), 5000); UpdateWindowTitle(); } else { qCWarning(vMainWindow, "%s", qUtf8Printable(tr("Couldn't unload measurements. Some of them are used in the pattern."))); } } //--------------------------------------------------------------------------------------------------------------------- void MainWindow::ShowMeasurements() { if (not doc->MPath().isEmpty()) { const QString absoluteMPath = AbsoluteMPath(qApp->GetPPath(), doc->MPath()); QStringList arguments; if (qApp->patternType() == MeasurementsType::Multisize) { arguments = QStringList() << absoluteMPath << "-u" << UnitsToStr(qApp->patternUnit()) << "-e" << QString().setNum(static_cast(UnitConvertor(VContainer::height(), doc->MUnit(), Unit::Cm))) << "-s" << QString().setNum(static_cast(UnitConvertor(VContainer::size(), doc->MUnit(), Unit::Cm))); } else { arguments = QStringList() << absoluteMPath << "-u" << UnitsToStr(qApp->patternUnit()); } if (isNoScaling) { arguments.append(QLatin1String("--") + LONG_OPTION_NO_HDPI_SCALING); } const QString tape = qApp->TapeFilePath(); const QString workingDirectory = QFileInfo(tape).absoluteDir().absolutePath(); QProcess::startDetached(tape, arguments, workingDirectory); } else { ui->actionEditCurrent->setEnabled(false); } } //--------------------------------------------------------------------------------------------------------------------- void MainWindow::MeasurementsChanged(const QString &path) { mChanges = false; QFileInfo checkFile(path); if (checkFile.exists()) { mChanges = true; mChangesAsked = false; } else { for(int i=0; i<=1000; i=i+10) { if (checkFile.exists()) { mChanges = true; mChangesAsked = false; break; } else { std::this_thread::sleep_for(std::chrono::milliseconds(10)); } } } UpdateWindowTitle(); ui->actionSyncMeasurements->setEnabled(mChanges); } //--------------------------------------------------------------------------------------------------------------------- void MainWindow::SyncMeasurements() { if (mChanges) { const QString path = AbsoluteMPath(qApp->GetPPath(), doc->MPath()); if(UpdateMeasurements(path, static_cast(VContainer::size()), static_cast(VContainer::height()))) { if (not watcher->files().contains(path)) { watcher->addPath(path); } const QString msg = tr("Measurements have been synced"); qCDebug(vMainWindow, "%s", qUtf8Printable(msg)); statusBar()->showMessage(msg, 5000); VWidgetPopup::PopupMessage(this, msg); doc->LiteParseTree(Document::LiteParse); mChanges = false; mChangesAsked = true; UpdateWindowTitle(); ui->actionSyncMeasurements->setEnabled(mChanges); } else { qCWarning(vMainWindow, "%s", qUtf8Printable(tr("Couldn't sync measurements."))); } } } //--------------------------------------------------------------------------------------------------------------------- #if defined(Q_OS_MAC) void MainWindow::OpenAt(QAction *where) { const QString path = qApp->GetPPath().left(qApp->GetPPath().indexOf(where->text())) + where->text(); if (path == qApp->GetPPath()) { return; } QProcess process; process.start(QStringLiteral("/usr/bin/open"), QStringList() << path, QIODevice::ReadOnly); process.waitForFinished(); } #endif //defined(Q_OS_MAC) //--------------------------------------------------------------------------------------------------------------------- /** * @brief ToolBarOption enable option toolbar. */ void MainWindow::ToolBarOption() { ui->toolBarOption->clear(); if (not mouseCoordinate.isNull()) { delete mouseCoordinate; } if (not gradationHeights.isNull()) { delete gradationHeights; } if (not gradationSizes.isNull()) { delete gradationSizes; } if (not gradationHeightsLabel.isNull()) { delete gradationHeightsLabel; } if (not gradationSizesLabel.isNull()) { delete gradationSizesLabel; } if (not zoomScale.isNull()) { delete zoomScale; } if (not doubleSpinBoxScale.isNull()) { delete doubleSpinBoxScale; } if (qApp->patternType() == MeasurementsType::Multisize) { const QStringList listHeights = VMeasurement::ListHeights(doc->GetGradationHeights(), qApp->patternUnit()); const QStringList listSizes = VMeasurement::ListSizes(doc->GetGradationSizes(), qApp->patternUnit()); gradationHeightsLabel = new QLabel(tr("Height:"), this); gradationHeights = SetGradationList(gradationHeightsLabel, listHeights); // set default height SetDefaultHeight(); connect(gradationHeights.data(), QOverload::of(&QComboBox::currentIndexChanged), this, &MainWindow::ChangedHeight); gradationSizesLabel = new QLabel(tr("Size:"), this); gradationSizes = SetGradationList(gradationSizesLabel, listSizes); // set default size SetDefaultSize(); connect(gradationSizes.data(), QOverload::of(&QComboBox::currentIndexChanged), this, &MainWindow::ChangedSize); ui->toolBarOption->addSeparator(); } zoomScale = new QLabel(tr("Scale:"), this); ui->toolBarOption->addWidget(zoomScale); doubleSpinBoxScale = new QDoubleSpinBox(this); doubleSpinBoxScale->setDecimals(1); doubleSpinBoxScale->setSuffix("%"); ScaleChanged(ui->view->transform().m11()); connect(doubleSpinBoxScale.data(), QOverload::of(&QDoubleSpinBox::valueChanged), this, [this](double d){ui->view->Zoom(d/100.0);}); ui->toolBarOption->addWidget(doubleSpinBoxScale); ui->toolBarOption->addSeparator(); mouseCoordinate = new QLabel(QString("0, 0 (%1)").arg(UnitsToStr(qApp->patternUnit(), true))); ui->toolBarOption->addWidget(mouseCoordinate); } //--------------------------------------------------------------------------------------------------------------------- QComboBox *MainWindow::SetGradationList(QLabel *label, const QStringList &list) { ui->toolBarOption->addWidget(label); QComboBox *comboBox = new QComboBox(this); comboBox->addItems(list); ui->toolBarOption->addWidget(comboBox); return comboBox; } //--------------------------------------------------------------------------------------------------------------------- void MainWindow::ToolBarStages() { leftGoToStage = new QLabel(this); leftGoToStage->setPixmap(QPixmap("://icon/24x24/fast_forward_left_to_right_arrow.png")); ui->toolBarStages->insertWidget(ui->actionDetails, leftGoToStage); rightGoToStage = new QLabel(this); rightGoToStage->setPixmap(QPixmap("://icon/24x24/left_to_right_arrow.png")); ui->toolBarStages->insertWidget(ui->actionLayout, rightGoToStage); } //--------------------------------------------------------------------------------------------------------------------- /** * @brief ToolBarDraws enable draw toolbar. */ void MainWindow::ToolBarDraws() { patternPieceLabel = new QLabel(tr("Pattern Piece:")); ui->toolBarDraws->addWidget(patternPieceLabel); // By using Qt UI Designer we can't add QComboBox to toolbar comboBoxDraws = new QComboBox; ui->toolBarDraws->addWidget(comboBoxDraws); comboBoxDraws->setSizeAdjustPolicy(QComboBox::AdjustToContents); comboBoxDraws->setEnabled(false); connect(comboBoxDraws, QOverload::of(&QComboBox::currentIndexChanged), this, [this](int index){ChangePP(index);}); connect(ui->actionOptionDraw, &QAction::triggered, this, [this]() { const QString activDraw = doc->GetNameActivPP(); const QString nameDraw = PatternPieceName(activDraw); if (nameDraw.isEmpty()) { return; } RenamePP *renamePP = new RenamePP(doc, nameDraw, comboBoxDraws); qApp->getUndoStack()->push(renamePP); }); } //--------------------------------------------------------------------------------------------------------------------- void MainWindow::ToolBarTools() { /*First we will try use Standard Shortcuts from Qt, but because keypad "-" and "+" not the same keys like in main keypad, shortcut Ctrl+"-" or "+" from keypad will not working with standard shortcut (QKeySequence::ZoomIn or QKeySequence::ZoomOut). For examle "+" is Qt::Key_Plus + Qt::KeypadModifier for keypad. Also for me don't work Qt:CTRL and work Qt::ControlModifier.*/ QList zoomInShortcuts; zoomInShortcuts.append(QKeySequence(QKeySequence::ZoomIn)); zoomInShortcuts.append(QKeySequence(Qt::ControlModifier + Qt::Key_Plus + Qt::KeypadModifier)); ui->actionZoomIn->setShortcuts(zoomInShortcuts); connect(ui->actionZoomIn, &QAction::triggered, ui->view, &VMainGraphicsView::ZoomIn); QList zoomOutShortcuts; zoomOutShortcuts.append(QKeySequence(QKeySequence::ZoomOut)); zoomOutShortcuts.append(QKeySequence(Qt::ControlModifier + Qt::Key_Minus + Qt::KeypadModifier)); ui->actionZoomOut->setShortcuts(zoomOutShortcuts); connect(ui->actionZoomOut, &QAction::triggered, ui->view, &VMainGraphicsView::ZoomOut); QList zoomOriginalShortcuts; zoomOriginalShortcuts.append(QKeySequence(Qt::ControlModifier + Qt::Key_0)); zoomOriginalShortcuts.append(QKeySequence(Qt::ControlModifier + Qt::Key_0 + Qt::KeypadModifier)); ui->actionZoomOriginal->setShortcuts(zoomOriginalShortcuts); connect(ui->actionZoomOriginal, &QAction::triggered, ui->view, &VMainGraphicsView::ZoomOriginal); QList zoomFitBestShortcuts; zoomFitBestShortcuts.append(QKeySequence(Qt::ControlModifier + Qt::Key_Equal)); ui->actionZoomFitBest->setShortcuts(zoomFitBestShortcuts); connect(ui->actionZoomFitBest, &QAction::triggered, ui->view, &VMainGraphicsView::ZoomFitBest); QList zoomFitBestCurrentShortcuts; zoomFitBestCurrentShortcuts.append(QKeySequence(Qt::ControlModifier + Qt::Key_M)); ui->actionZoomFitBestCurrent->setShortcuts(zoomFitBestCurrentShortcuts); connect(ui->actionZoomFitBestCurrent, &QAction::triggered, this, &MainWindow::ZoomFitBestCurrent); connect(ui->actionPreviousPatternPiece, &QAction::triggered, this, &MainWindow::PreviousPatternPiece); connect(ui->actionNextPatternPiece, &QAction::triggered, this, &MainWindow::NextPatternPiece); } //--------------------------------------------------------------------------------------------------------------------- void MainWindow::InitToolButtons() { toolButtonPointerList.append(ui->toolButtonPointerPoint); toolButtonPointerList.append(ui->toolButtonPointerLine); toolButtonPointerList.append(ui->toolButtonPointerCurve); toolButtonPointerList.append(ui->toolButtonPointerArc); toolButtonPointerList.append(ui->toolButtonPointerDetail); toolButtonPointerList.append(ui->toolButtonPointerOperations); toolButtonPointerList.append(ui->toolButtonPointerEllipticalArc); for (auto pointer : qAsConst(toolButtonPointerList)) { connect(pointer, &QToolButton::clicked, this, &MainWindow::ArrowTool); } // This check helps to find missed tools Q_STATIC_ASSERT_X(static_cast(Tool::LAST_ONE_DO_NOT_USE) == 55, "Check if all tools were connected."); connect(ui->toolButtonEndLine, &QToolButton::clicked, this, &MainWindow::ToolEndLine); connect(ui->toolButtonLine, &QToolButton::clicked, this, &MainWindow::ToolLine); connect(ui->toolButtonAlongLine, &QToolButton::clicked, this, &MainWindow::ToolAlongLine); connect(ui->toolButtonShoulderPoint, &QToolButton::clicked, this, &MainWindow::ToolShoulderPoint); connect(ui->toolButtonNormal, &QToolButton::clicked, this, &MainWindow::ToolNormal); connect(ui->toolButtonBisector, &QToolButton::clicked, this, &MainWindow::ToolBisector); connect(ui->toolButtonLineIntersect, &QToolButton::clicked, this, &MainWindow::ToolLineIntersect); connect(ui->toolButtonSpline, &QToolButton::clicked, this, &MainWindow::ToolSpline); connect(ui->toolButtonCubicBezier, &QToolButton::clicked, this, &MainWindow::ToolCubicBezier); connect(ui->toolButtonArc, &QToolButton::clicked, this, &MainWindow::ToolArc); connect(ui->toolButtonSplinePath, &QToolButton::clicked, this, &MainWindow::ToolSplinePath); connect(ui->toolButtonCubicBezierPath, &QToolButton::clicked, this, &MainWindow::ToolCubicBezierPath); connect(ui->toolButtonPointOfContact, &QToolButton::clicked, this, &MainWindow::ToolPointOfContact); connect(ui->toolButtonNewDetail, &QToolButton::clicked, this, &MainWindow::ToolDetail); connect(ui->toolButtonInternalPath, &QToolButton::clicked, this, &MainWindow::ToolPiecePath); connect(ui->toolButtonHeight, &QToolButton::clicked, this, &MainWindow::ToolHeight); connect(ui->toolButtonTriangle, &QToolButton::clicked, this, &MainWindow::ToolTriangle); connect(ui->toolButtonPointOfIntersection, &QToolButton::clicked, this, &MainWindow::ToolPointOfIntersection); connect(ui->toolButtonSplineCutPoint, &QToolButton::clicked, this, &MainWindow::ToolCutSpline); connect(ui->toolButtonSplinePathCutPoint, &QToolButton::clicked, this, &MainWindow::ToolCutSplinePath); connect(ui->toolButtonUnionDetails, &QToolButton::clicked, this, &MainWindow::ToolUnionDetails); connect(ui->toolButtonDuplicateDetail, &QToolButton::clicked, this, &MainWindow::ToolDuplicateDetail); connect(ui->toolButtonArcCutPoint, &QToolButton::clicked, this, &MainWindow::ToolCutArc); connect(ui->toolButtonLineIntersectAxis, &QToolButton::clicked, this, &MainWindow::ToolLineIntersectAxis); connect(ui->toolButtonCurveIntersectAxis, &QToolButton::clicked, this, &MainWindow::ToolCurveIntersectAxis); connect(ui->toolButtonArcIntersectAxis, &QToolButton::clicked, this, &MainWindow::ToolArcIntersectAxis); connect(ui->toolButtonLayoutSettings, &QToolButton::clicked, this, &MainWindow::ToolLayoutSettings); connect(ui->toolButtonPointOfIntersectionArcs, &QToolButton::clicked, this, &MainWindow::ToolPointOfIntersectionArcs); connect(ui->toolButtonPointOfIntersectionCircles, &QToolButton::clicked, this, &MainWindow::ToolPointOfIntersectionCircles); connect(ui->toolButtonIntersectionCurves, &QToolButton::clicked, this, &MainWindow::ToolPointOfIntersectionCurves); connect(ui->toolButtonPointFromCircleAndTangent, &QToolButton::clicked, this, &MainWindow::ToolPointFromCircleAndTangent); connect(ui->toolButtonPointFromArcAndTangent, &QToolButton::clicked, this, &MainWindow::ToolPointFromArcAndTangent); connect(ui->toolButtonArcWithLength, &QToolButton::clicked, this, &MainWindow::ToolArcWithLength); connect(ui->toolButtonTrueDarts, &QToolButton::clicked, this, &MainWindow::ToolTrueDarts); connect(ui->toolButtonGroup, &QToolButton::clicked, this, &MainWindow::ToolGroup); connect(ui->toolButtonRotation, &QToolButton::clicked, this, &MainWindow::ToolRotation); connect(ui->toolButtonFlippingByLine, &QToolButton::clicked, this, &MainWindow::ToolFlippingByLine); connect(ui->toolButtonFlippingByAxis, &QToolButton::clicked, this, &MainWindow::ToolFlippingByAxis); connect(ui->toolButtonMove, &QToolButton::clicked, this, &MainWindow::ToolMove); connect(ui->toolButtonMidpoint, &QToolButton::clicked, this, &MainWindow::ToolMidpoint); connect(ui->toolButtonLayoutExportAs, &QToolButton::clicked, this, &MainWindow::ExportLayoutAs); connect(ui->toolButtonDetailExportAs, &QToolButton::clicked, this, &MainWindow::ExportDetailsAs); connect(ui->toolButtonEllipticalArc, &QToolButton::clicked, this, &MainWindow::ToolEllipticalArc); connect(ui->toolButtonPin, &QToolButton::clicked, this, &MainWindow::ToolPin); connect(ui->toolButtonInsertNode, &QToolButton::clicked, this, &MainWindow::ToolInsertNode); connect(ui->toolButtonPlaceLabel, &QToolButton::clicked, this, &MainWindow::ToolPlaceLabel); } //--------------------------------------------------------------------------------------------------------------------- /** * @brief mouseMove save mouse position and show user. * @param scenePos position mouse. */ void MainWindow::MouseMove(const QPointF &scenePos) { if (mouseCoordinate) { //: Coords in status line: "X, Y (units)" mouseCoordinate->setText(QString("%1, %2 (%3)").arg(static_cast(qApp->fromPixel(scenePos.x()))) .arg(static_cast(qApp->fromPixel(scenePos.y()))) .arg(UnitsToStr(qApp->patternUnit(), true))); } } //--------------------------------------------------------------------------------------------------------------------- QT_WARNING_PUSH QT_WARNING_DISABLE_GCC("-Wswitch-default") /** * @brief CancelTool cancel tool. */ void MainWindow::CancelTool() { // This check helps to find missed tools in the switch Q_STATIC_ASSERT_X(static_cast(Tool::LAST_ONE_DO_NOT_USE) == 55, "Not all tools were handled."); qCDebug(vMainWindow, "Canceling tool."); dialogTool.clear(); qCDebug(vMainWindow, "Dialog closed."); currentScene->setFocus(Qt::OtherFocusReason); currentScene->clearSelection(); emit ui->view->itemClicked(nullptr); // Hide visualization to avoid a crash switch ( currentTool ) { case Tool::Arrow: for (auto pointer : qAsConst(toolButtonPointerList)) { pointer->setChecked(false); } m_statusLabel->setText(QString()); // Crash: using CRTL+Z while using line tool. // related bug report: // https://bitbucket.org/dismine/valentina/issues/454/crash-using-crtl-z-while-using-line-tool undoAction->setEnabled(false); redoAction->setEnabled(false); VAbstractTool::m_suppressContextMenu = true; return; case Tool::BasePoint: case Tool::SinglePoint: case Tool::DoublePoint: case Tool::LinePoint: case Tool::AbstractSpline: case Tool::Cut: case Tool::LAST_ONE_DO_NOT_USE: case Tool::NodePoint: case Tool::NodeArc: case Tool::NodeElArc: case Tool::NodeSpline: case Tool::NodeSplinePath: Q_UNREACHABLE(); //-V501 //Nothing to do here because we can't create this tool from main window. break; case Tool::EndLine: ui->toolButtonEndLine->setChecked(false); break; case Tool::Line: ui->toolButtonLine->setChecked(false); break; case Tool::AlongLine: ui->toolButtonAlongLine->setChecked(false); break; case Tool::Midpoint: ui->toolButtonMidpoint->setChecked(false); break; case Tool::ShoulderPoint: ui->toolButtonShoulderPoint->setChecked(false); break; case Tool::Normal: ui->toolButtonNormal->setChecked(false); break; case Tool::Bisector: ui->toolButtonBisector->setChecked(false); break; case Tool::LineIntersect: ui->toolButtonLineIntersect->setChecked(false); break; case Tool::Spline: ui->toolButtonSpline->setChecked(false); break; case Tool::CubicBezier: ui->toolButtonCubicBezier->setChecked(false); break; case Tool::Arc: ui->toolButtonArc->setChecked(false); break; case Tool::ArcWithLength: ui->toolButtonArcWithLength->setChecked(false); break; case Tool::SplinePath: ui->toolButtonSplinePath->setChecked(false); break; case Tool::CubicBezierPath: ui->toolButtonCubicBezierPath->setChecked(false); break; case Tool::PointOfContact: ui->toolButtonPointOfContact->setChecked(false); break; case Tool::Piece: ui->toolButtonNewDetail->setChecked(false); break; case Tool::PiecePath: ui->toolButtonInternalPath->setChecked(false); break; case Tool::Height: ui->toolButtonHeight->setChecked(false); break; case Tool::Triangle: ui->toolButtonTriangle->setChecked(false); break; case Tool::PointOfIntersection: ui->toolButtonPointOfIntersection->setChecked(false); break; case Tool::CutSpline: ui->toolButtonSplineCutPoint->setChecked(false); break; case Tool::CutSplinePath: ui->toolButtonSplinePathCutPoint->setChecked(false); break; case Tool::UnionDetails: ui->toolButtonUnionDetails->setChecked(false); break; case Tool::DuplicateDetail: ui->toolButtonDuplicateDetail->setChecked(false); break; case Tool::CutArc: ui->toolButtonArcCutPoint->setChecked(false); break; case Tool::LineIntersectAxis: ui->toolButtonLineIntersectAxis->setChecked(false); break; case Tool::CurveIntersectAxis: ui->toolButtonCurveIntersectAxis->setChecked(false); break; case Tool::ArcIntersectAxis: ui->toolButtonArcIntersectAxis->setChecked(false); break; case Tool::PointOfIntersectionArcs: ui->toolButtonPointOfIntersectionArcs->setChecked(false); break; case Tool::PointOfIntersectionCircles: ui->toolButtonPointOfIntersectionCircles->setChecked(false); break; case Tool::PointOfIntersectionCurves: ui->toolButtonIntersectionCurves->setChecked(false); break; case Tool::PointFromCircleAndTangent: ui->toolButtonPointFromCircleAndTangent->setChecked(false); break; case Tool::PointFromArcAndTangent: ui->toolButtonPointFromArcAndTangent->setChecked(false); break; case Tool::TrueDarts: ui->toolButtonTrueDarts->setChecked(false); break; case Tool::Group: ui->toolButtonGroup->setChecked(false); break; case Tool::Rotation: ui->toolButtonRotation->setChecked(false); break; case Tool::FlippingByLine: ui->toolButtonFlippingByLine->setChecked(false); break; case Tool::FlippingByAxis: ui->toolButtonFlippingByAxis->setChecked(false); break; case Tool::Move: ui->toolButtonMove->setChecked(false); break; case Tool::EllipticalArc: ui->toolButtonEllipticalArc->setChecked(false); break; case Tool::Pin: ui->toolButtonPin->setChecked(false); break; case Tool::InsertNode: ui->toolButtonInsertNode->setChecked(false); break; case Tool::PlaceLabel: ui->toolButtonPlaceLabel->setChecked(false); break; } // Crash: using CRTL+Z while using line tool. // related bug report: // https://bitbucket.org/dismine/valentina/issues/454/crash-using-crtl-z-while-using-line-tool undoAction->setEnabled(qApp->getUndoStack()->canUndo()); redoAction->setEnabled(qApp->getUndoStack()->canRedo()); } QT_WARNING_POP //--------------------------------------------------------------------------------------------------------------------- /** * @brief ArrowTool enable arrow tool. */ void MainWindow::ArrowTool(bool checked) { if (checked && currentTool != Tool::Arrow) { qCDebug(vMainWindow, "Arrow tool."); CancelTool(); for (auto pointer : qAsConst(toolButtonPointerList)) { pointer->setChecked(true); } currentTool = Tool::Arrow; emit EnableItemMove(true); emit ItemsSelection(SelectionType::ByMouseRelease); VAbstractTool::m_suppressContextMenu = false; // Only true for rubber band selection emit EnableLabelSelection(true); emit EnablePointSelection(false); emit EnableLineSelection(false); emit EnableArcSelection(false); emit EnableElArcSelection(false); emit EnableSplineSelection(false); emit EnableSplinePathSelection(false); emit EnableNodeLabelSelection(true); emit EnableNodePointSelection(true); emit EnableDetailSelection(true);// Disable when done visualization details // Hovering emit EnableLabelHover(true); emit EnablePointHover(true); emit EnableLineHover(true); emit EnableArcHover(true); emit EnableElArcHover(true); emit EnableSplineHover(true); emit EnableSplinePathHover(true); emit EnableNodeLabelHover(true); emit EnableNodePointHover(true); emit EnableDetailHover(true); ui->view->AllowRubberBand(true); ui->view->viewport()->unsetCursor(); ui->view->viewport()->setCursor(QCursor(Qt::ArrowCursor)); ui->view->setCurrentCursorShape(); // Hack to fix problem with a cursor m_statusLabel->setText(QString()); ui->view->setShowToolOptions(true); qCDebug(vMainWindow, "Enabled arrow tool."); } else { for (auto pointer : qAsConst(toolButtonPointerList)) { pointer->setChecked(true); } } } //--------------------------------------------------------------------------------------------------------------------- /** * @brief keyPressEvent handle key press events. * @param event key event. */ void MainWindow::keyPressEvent ( QKeyEvent * event ) { switch (event->key()) { case Qt::Key_Escape: ArrowTool(true); break; case Qt::Key_Return: case Qt::Key_Enter: EndVisualization(); break; default: break; } QMainWindow::keyPressEvent ( event ); } //--------------------------------------------------------------------------------------------------------------------- /** * @brief SaveCurrentScene save scene options before set another. */ void MainWindow::SaveCurrentScene() { if (mode == Draw::Calculation || mode == Draw::Modeling) { VMainGraphicsScene *scene = qobject_cast(currentScene); SCASSERT(scene != nullptr) /*Save transform*/ scene->setTransform(ui->view->transform()); /*Save scroll bars value for previous scene.*/ QScrollBar *horScrollBar = ui->view->horizontalScrollBar(); scene->setHorScrollBar(horScrollBar->value()); QScrollBar *verScrollBar = ui->view->verticalScrollBar(); scene->setVerScrollBar(verScrollBar->value()); } } //--------------------------------------------------------------------------------------------------------------------- /** * @brief RestoreCurrentScene restore scene options after change. */ void MainWindow::RestoreCurrentScene() { VMainGraphicsScene *scene = qobject_cast(currentScene); SCASSERT(scene != nullptr) /*Set transform for current scene*/ ui->view->setTransform(scene->transform()); /*Set value for current scene scroll bar.*/ QScrollBar *horScrollBar = ui->view->horizontalScrollBar(); horScrollBar->setValue(scene->getHorScrollBar()); QScrollBar *verScrollBar = ui->view->verticalScrollBar(); verScrollBar->setValue(scene->getVerScrollBar()); } //--------------------------------------------------------------------------------------------------------------------- /** * @brief ActionDraw show draw scene. * @param checked true - button checked. */ void MainWindow::ActionDraw(bool checked) { if (checked) { qCDebug(vMainWindow, "Show draw scene"); ArrowTool(true); leftGoToStage->setPixmap(QPixmap("://icon/24x24/fast_forward_left_to_right_arrow.png")); rightGoToStage->setPixmap(QPixmap("://icon/24x24/left_to_right_arrow.png")); ui->actionDraw->setChecked(true); ui->actionDetails->setChecked(false); ui->actionLayout->setChecked(false); SaveCurrentScene(); currentScene = sceneDraw; ui->view->setScene(currentScene); RestoreCurrentScene(); mode = Draw::Calculation; comboBoxDraws->setCurrentIndex(currentDrawIndex);//restore current pattern peace drawMode = true; SetEnableTool(true); SetEnableWidgets(true); ui->toolBox->setCurrentIndex(currentToolBoxIndex); if (qApp->patternType() == MeasurementsType::Multisize) { ui->toolBarOption->setVisible(true); } ui->dockWidgetLayoutPages->setVisible(false); ui->dockWidgetToolOptions->setVisible(isDockToolOptionsVisible); ui->dockWidgetGroups->setWidget(groupsWidget); ui->dockWidgetGroups->setWindowTitle(tr("Group")); ui->dockWidgetGroups->setVisible(isDockGroupsVisible); ui->dockWidgetGroups->setToolTip(tr("Contains all visibility groups")); } else { ui->actionDraw->setChecked(true); } } //--------------------------------------------------------------------------------------------------------------------- /** * @brief ActionDetails show details scene. * @param checked true - button checked. */ void MainWindow::ActionDetails(bool checked) { if (checked) { ArrowTool(true); if(drawMode) { currentDrawIndex = comboBoxDraws->currentIndex();//save current pattern piece drawMode = false; } comboBoxDraws->setCurrentIndex(comboBoxDraws->count()-1);// Need to get data about all details leftGoToStage->setPixmap(QPixmap("://icon/24x24/right_to_left_arrow.png")); rightGoToStage->setPixmap(QPixmap("://icon/24x24/left_to_right_arrow.png")); ui->actionDraw->setChecked(false); ui->actionDetails->setChecked(true); ui->actionLayout->setChecked(false); if(not qApp->getOpeningPattern()) { if (pattern->DataPieces()->count() == 0) { QMessageBox::information(this, tr("Detail mode"), tr("You can't use Detail mode yet. " "Please, create at least one workpiece."), QMessageBox::Ok, QMessageBox::Ok); ActionDraw(true); return; } } detailsWidget->UpdateList(); qCDebug(vMainWindow, "Show details scene"); SaveCurrentScene(); currentScene = sceneDetails; emit ui->view->itemClicked(nullptr); ui->view->setScene(currentScene); RestoreCurrentScene(); if (mode == Draw::Calculation) { currentToolBoxIndex = ui->toolBox->currentIndex(); } mode = Draw::Modeling; SetEnableTool(true); SetEnableWidgets(true); ui->toolBox->setCurrentIndex(ui->toolBox->indexOf(ui->detailPage)); if (qApp->patternType() == MeasurementsType::Multisize) { ui->toolBarOption->setVisible(true); } ui->dockWidgetLayoutPages->setVisible(false); ui->dockWidgetToolOptions->setVisible(isDockToolOptionsVisible); ui->dockWidgetGroups->setWidget(detailsWidget); ui->dockWidgetGroups->setWindowTitle(tr("Details")); ui->dockWidgetGroups->setVisible(isDockGroupsVisible); ui->dockWidgetGroups->setToolTip(tr("Show which details will go in layout")); m_statusLabel->setText(QString()); } else { ui->actionDetails->setChecked(true); } } //--------------------------------------------------------------------------------------------------------------------- /** * @brief ActionLayout begin creation layout. * @param checked true - button checked. */ void MainWindow::ActionLayout(bool checked) { if (checked) { ArrowTool(true); if(drawMode) { currentDrawIndex = comboBoxDraws->currentIndex();//save current pattern piece drawMode = false; } comboBoxDraws->setCurrentIndex(comboBoxDraws->count()-1);// Need to get data about all details leftGoToStage->setPixmap(QPixmap("://icon/24x24/right_to_left_arrow.png")); rightGoToStage->setPixmap(QPixmap("://icon/24x24/fast_forward_right_to_left_arrow.png")); ui->actionDraw->setChecked(false); ui->actionDetails->setChecked(false); ui->actionLayout->setChecked(true); QHash details; if(not qApp->getOpeningPattern()) { const QHash *allDetails = pattern->DataPieces(); if (allDetails->count() == 0) { QMessageBox::information(this, tr("Layout mode"), tr("You can't use Layout mode yet. " "Please, create at least one workpiece."), QMessageBox::Ok, QMessageBox::Ok); ActionDraw(true); return; } else { QHash::const_iterator i = allDetails->constBegin(); while (i != allDetails->constEnd()) { if (i.value().IsInLayout()) { details.insert(i.key(), i.value()); } ++i; } if (details.count() == 0) { QMessageBox::information(this, tr("Layout mode"), tr("You can't use Layout mode yet. Please, " "include at least one detail in layout."), QMessageBox::Ok, QMessageBox::Ok); mode == Draw::Calculation ? ActionDraw(true) : ActionDetails(true); return; } } } comboBoxDraws->setCurrentIndex(-1);// Hide pattern pieces qCDebug(vMainWindow, "Show layout scene"); SaveCurrentScene(); try { listDetails = PrepareDetailsForLayout(details); } catch (VException &e) { listDetails.clear(); QMessageBox::warning(this, tr("Layout mode"), tr("You can't use Layout mode yet.") + QLatin1String(" \n") + e.ErrorMessage(), QMessageBox::Ok, QMessageBox::Ok); mode == Draw::Calculation ? ActionDraw(true) : ActionDetails(true); return; } currentScene = tempSceneLayout; emit ui->view->itemClicked(nullptr); ui->view->setScene(currentScene); if (mode == Draw::Calculation) { currentToolBoxIndex = ui->toolBox->currentIndex(); } mode = Draw::Layout; SetEnableTool(true); SetEnableWidgets(true); ui->toolBox->setCurrentIndex(ui->toolBox->indexOf(ui->layoutPage)); mouseCoordinate->setText(QString()); if (qApp->patternType() == MeasurementsType::Multisize) { ui->toolBarOption->setVisible(false); } ui->dockWidgetLayoutPages->setVisible(true); ui->dockWidgetToolOptions->blockSignals(true); ui->dockWidgetToolOptions->setVisible(false); ui->dockWidgetToolOptions->blockSignals(false); ui->dockWidgetGroups->blockSignals(true); ui->dockWidgetGroups->setVisible(false); ui->dockWidgetGroups->blockSignals(false); ShowPaper(ui->listWidget->currentRow()); if (scenes.isEmpty() || isLayoutStale) { ui->toolButtonLayoutSettings->click(); } m_statusLabel->setText(QString()); } else { ui->actionLayout->setChecked(true); } } //--------------------------------------------------------------------------------------------------------------------- /** * @brief SaveAs save as pattern file. * @return true for successes saving. */ bool MainWindow::SaveAs() { QString filters(tr("Pattern files") + QLatin1String("(*.val)")); QString dir; if (qApp->GetPPath().isEmpty()) { dir = qApp->ValentinaSettings()->GetPathPattern(); } else { dir = QFileInfo(qApp->GetPPath()).absolutePath(); } bool usedNotExistedDir = false; QDir directory(dir); if (not directory.exists()) { usedNotExistedDir = directory.mkpath(QChar('.')); } QString fileName = QFileDialog::getSaveFileName(this, tr("Save as"), dir + QLatin1String("/") + tr("pattern") + QLatin1String(".val"), filters, nullptr #ifdef Q_OS_LINUX , QFileDialog::DontUseNativeDialog #endif ); auto RemoveTempDir = [usedNotExistedDir, dir]() { if (usedNotExistedDir) { QDir directory(dir); directory.rmpath(QChar('.')); } }; if (fileName.isEmpty()) { RemoveTempDir(); return false; } QFileInfo f( fileName ); if (f.suffix().isEmpty() && f.suffix() != QLatin1String("val")) { fileName += QLatin1String(".val"); } if (f.exists()) { // Temporary try to lock the file before saving // Also help to rewite current read-only pattern VLockGuard tmp(fileName); if (not tmp.IsLocked()) { qCCritical(vMainWindow, "%s", qUtf8Printable(tr("Failed to lock. This file already opened in another window."))); RemoveTempDir(); return false; } } // Need for restoring previous state in case of failure const bool wasModified = doc->IsModified(); // Need because SetReadOnly() will change internal state const bool readOnly = doc->IsReadOnly(); doc->SetReadOnly(false);// Save as... disable read only state QString error; const bool result = SavePattern(fileName, error); if (result == false) { QMessageBox messageBox(this); messageBox.setIcon(QMessageBox::Warning); messageBox.setInformativeText(tr("Could not save file")); messageBox.setDefaultButton(QMessageBox::Ok); messageBox.setDetailedText(error); messageBox.setStandardButtons(QMessageBox::Ok); messageBox.exec(); // Restoring previous state doc->SetReadOnly(readOnly); doc->SetModified(wasModified); RemoveTempDir(); return result; } patternReadOnly = false; qCDebug(vMainWindow, "Locking file"); VlpCreateLock(lock, fileName); if (lock->IsLocked()) { qCDebug(vMainWindow, "Pattern file %s was locked.", qUtf8Printable(fileName)); } else { qCDebug(vMainWindow, "Failed to lock %s", qUtf8Printable(fileName)); qCDebug(vMainWindow, "Error type: %d", lock->GetLockError()); qCCritical(vMainWindow, "%s", qUtf8Printable(tr("Failed to lock. This file already opened in another window. Expect " "collissions when run 2 copies of the program."))); } RemoveTempDir(); return result; } //--------------------------------------------------------------------------------------------------------------------- /** * @brief Save save pattern file. * @return true for successes saving. */ bool MainWindow::Save() { if (qApp->GetPPath().isEmpty() || patternReadOnly) { return SaveAs(); } else { if (m_curFileFormatVersion < VPatternConverter::PatternMaxVer && not ContinueFormatRewrite(m_curFileFormatVersionStr, VPatternConverter::PatternMaxVerStr)) { return false; } #ifdef Q_OS_WIN32 qt_ntfs_permission_lookup++; // turn checking on #endif /*Q_OS_WIN32*/ const bool isFileWritable = QFileInfo(qApp->GetPPath()).isWritable(); #ifdef Q_OS_WIN32 qt_ntfs_permission_lookup--; // turn it off again #endif /*Q_OS_WIN32*/ if (not isFileWritable) { QMessageBox messageBox(this); messageBox.setIcon(QMessageBox::Question); messageBox.setText(tr("The document has no write permissions.")); messageBox.setInformativeText("Do you want to change the premissions?"); messageBox.setStandardButtons(QMessageBox::Yes | QMessageBox::Cancel); messageBox.setDefaultButton(QMessageBox::Yes); if (messageBox.exec() == QMessageBox::Yes) { #ifdef Q_OS_WIN32 qt_ntfs_permission_lookup++; // turn checking on #endif /*Q_OS_WIN32*/ bool changed = QFile::setPermissions(qApp->GetPPath(), QFileInfo(qApp->GetPPath()).permissions() | QFileDevice::WriteUser); #ifdef Q_OS_WIN32 qt_ntfs_permission_lookup--; // turn it off again #endif /*Q_OS_WIN32*/ if (not changed) { QMessageBox messageBox(this); messageBox.setIcon(QMessageBox::Warning); messageBox.setText(tr("Cannot set permissions for %1 to writable.").arg(qApp->GetPPath())); messageBox.setInformativeText(tr("Could not save the file.")); messageBox.setDefaultButton(QMessageBox::Ok); messageBox.setStandardButtons(QMessageBox::Ok); messageBox.exec(); return false; } } else { return false; } } QString error; bool result = SavePattern(qApp->GetPPath(), error); if (result) { QFile::remove(qApp->GetPPath() + *autosavePrefix); m_curFileFormatVersion = VPatternConverter::PatternMaxVer; m_curFileFormatVersionStr = VPatternConverter::PatternMaxVerStr; } else { QMessageBox messageBox(this); messageBox.setIcon(QMessageBox::Warning); messageBox.setText(tr("Could not save the file")); messageBox.setDefaultButton(QMessageBox::Ok); messageBox.setDetailedText(error); messageBox.setStandardButtons(QMessageBox::Ok); messageBox.exec(); } return result; } } //--------------------------------------------------------------------------------------------------------------------- /** * @brief Open ask user select pattern file. */ void MainWindow::Open() { qCDebug(vMainWindow, "Openning new file."); const QString filter(tr("Pattern files") + QLatin1String(" (*.val)")); //Get list last open files const QStringList files = qApp->ValentinaSettings()->GetRecentFileList(); QString dir; if (files.isEmpty()) { dir = QDir::homePath(); } else { //Absolute path to last open file dir = QFileInfo(files.first()).absolutePath(); } qCDebug(vMainWindow, "Run QFileDialog::getOpenFileName: dir = %s.", qUtf8Printable(dir)); const QString filePath = QFileDialog::getOpenFileName(this, tr("Open file"), dir, filter, nullptr #ifdef Q_OS_LINUX , QFileDialog::DontUseNativeDialog #endif ); if (filePath.isEmpty()) { return; } LoadPattern(filePath); } //--------------------------------------------------------------------------------------------------------------------- /** * @brief Clear reset to default window. */ void MainWindow::Clear() { qCDebug(vMainWindow, "Reseting main window."); lock.reset(); qCDebug(vMainWindow, "Unlocked pattern file."); ActionDraw(true); qCDebug(vMainWindow, "Returned to Draw mode."); setCurrentFile(QString());// Keep before cleaning a pattern data to prevent a crash pattern->Clear(); qCDebug(vMainWindow, "Clearing pattern."); if (not qApp->GetPPath().isEmpty() && not doc->MPath().isEmpty()) { watcher->removePath(AbsoluteMPath(qApp->GetPPath(), doc->MPath())); } doc->clear(); qCDebug(vMainWindow, "Clearing scenes."); sceneDraw->clear(); sceneDetails->clear(); ArrowTool(true); comboBoxDraws->clear(); ui->actionDraw->setEnabled(false); ui->actionDetails->setEnabled(false); ui->actionLayout->setEnabled(false); ui->actionNewDraw->setEnabled(false); ui->actionOptionDraw->setEnabled(false); ui->actionSave->setEnabled(false); ui->actionSaveAs->setEnabled(false); ui->actionPattern_properties->setEnabled(false); ui->actionZoomIn->setEnabled(false); ui->actionZoomOut->setEnabled(false); ui->actionZoomFitBest->setEnabled(false); ui->actionZoomFitBestCurrent->setEnabled(false); ui->actionZoomOriginal->setEnabled(false); ui->actionHistory->setEnabled(false); ui->actionTable->setEnabled(false); ui->actionExportIncrementsToCSV->setEnabled(false); ui->actionExportFinalMeasurementsToCSV->setEnabled(false); ui->actionFinalMeasurements->setEnabled(false); ui->actionLast_tool->setEnabled(false); ui->actionShowCurveDetails->setEnabled(false); ui->actionLoadIndividual->setEnabled(false); ui->actionLoadMultisize->setEnabled(false); ui->actionUnloadMeasurements->setEnabled(false); ui->actionEditCurrent->setEnabled(false); ui->actionPreviousPatternPiece->setEnabled(false); ui->actionNextPatternPiece->setEnabled(false); SetEnableTool(false); qApp->setPatternUnit(Unit::Cm); qApp->setPatternType(MeasurementsType::Unknown); ui->toolBarOption->clear(); #ifndef QT_NO_CURSOR QGuiApplication::restoreOverrideCursor(); #endif CleanLayout(); listDetails.clear(); // don't move to CleanLayout() qApp->getUndoStack()->clear(); toolOptions->ClearPropertyBrowser(); toolOptions->itemClicked(nullptr); m_progressBar->setVisible(false); #if defined(Q_OS_WIN32) && QT_VERSION >= QT_VERSION_CHECK(5, 7, 0) m_taskbarProgress->setVisible(false); #endif m_statusLabel->setVisible(true); } //--------------------------------------------------------------------------------------------------------------------- void MainWindow::FileClosedCorrect() { WriteSettings(); //File was closed correct. QStringList restoreFiles = qApp->ValentinaSettings()->GetRestoreFileList(); restoreFiles.removeAll(qApp->GetPPath()); qApp->ValentinaSettings()->SetRestoreFileList(restoreFiles); // Remove autosave file QFile autofile(qApp->GetPPath() + *autosavePrefix); if (autofile.exists()) { autofile.remove(); } qCDebug(vMainWindow, "File %s closed correct.", qUtf8Printable(qApp->GetPPath())); } //--------------------------------------------------------------------------------------------------------------------- void MainWindow::FullParseFile() { qCDebug(vMainWindow, "Full parsing file"); toolOptions->ClearPropertyBrowser(); try { SetEnabledGUI(true); doc->Parse(Document::FullParse); } catch (const VExceptionUndo &e) { Q_UNUSED(e) /* If user want undo last operation before undo we need finish broken redo operation. For those we post event * myself. Later in method customEvent call undo.*/ QApplication::postEvent(this, new UndoEvent()); return; } catch (const VExceptionObjectError &e) { qCCritical(vMainWindow, "%s\n\n%s\n\n%s", qUtf8Printable(tr("Error parsing file.")), //-V807 qUtf8Printable(e.ErrorMessage()), qUtf8Printable(e.DetailedInformation())); SetEnabledGUI(false); if (not VApplication::IsGUIMode()) { qApp->exit(V_EX_NOINPUT); } return; } catch (const VExceptionConversionError &e) { qCCritical(vMainWindow, "%s\n\n%s\n\n%s", qUtf8Printable(tr("Error can't convert value.")), qUtf8Printable(e.ErrorMessage()), qUtf8Printable(e.DetailedInformation())); SetEnabledGUI(false); if (not VApplication::IsGUIMode()) { qApp->exit(V_EX_NOINPUT); } return; } catch (const VExceptionEmptyParameter &e) { qCCritical(vMainWindow, "%s\n\n%s\n\n%s", qUtf8Printable(tr("Error empty parameter.")), qUtf8Printable(e.ErrorMessage()), qUtf8Printable(e.DetailedInformation())); SetEnabledGUI(false); if (not VApplication::IsGUIMode()) { qApp->exit(V_EX_NOINPUT); } return; } catch (const VExceptionWrongId &e) { qCCritical(vMainWindow, "%s\n\n%s\n\n%s", qUtf8Printable(tr("Error wrong id.")), qUtf8Printable(e.ErrorMessage()), qUtf8Printable(e.DetailedInformation())); SetEnabledGUI(false); if (not VApplication::IsGUIMode()) { qApp->exit(V_EX_NOINPUT); } return; } catch (VException &e) { qCCritical(vMainWindow, "%s\n\n%s\n\n%s", qUtf8Printable(tr("Error parsing file.")), qUtf8Printable(e.ErrorMessage()), qUtf8Printable(e.DetailedInformation())); SetEnabledGUI(false); if (not VApplication::IsGUIMode()) { qApp->exit(V_EX_NOINPUT); } return; } catch (const std::bad_alloc &) { qCCritical(vMainWindow, "%s", qUtf8Printable(tr("Error parsing file (std::bad_alloc)."))); SetEnabledGUI(false); if (not VApplication::IsGUIMode()) { qApp->exit(V_EX_NOINPUT); } return; } QString patternPiece; if (comboBoxDraws->currentIndex() != -1) { patternPiece = comboBoxDraws->itemText(comboBoxDraws->currentIndex()); } comboBoxDraws->blockSignals(true); comboBoxDraws->clear(); QStringList patternPieceNames = doc->getPatternPieces(); patternPieceNames.sort(); comboBoxDraws->addItems(patternPieceNames); if (not drawMode) { comboBoxDraws->setCurrentIndex(comboBoxDraws->count()-1); } else { const qint32 index = comboBoxDraws->findText(patternPiece); if ( index != -1 ) { comboBoxDraws->setCurrentIndex(index); } } comboBoxDraws->blockSignals(false); ui->actionPattern_properties->setEnabled(true); GlobalChangePP(patternPiece); SetEnableTool(comboBoxDraws->count() > 0); detailsWidget->UpdateList(); VMainGraphicsView::NewSceneRect(sceneDraw, qApp->getSceneView()); VMainGraphicsView::NewSceneRect(sceneDetails, qApp->getSceneView()); } //--------------------------------------------------------------------------------------------------------------------- void MainWindow::GlobalChangePP(const QString &patternPiece) { const qint32 index = comboBoxDraws->findText(patternPiece); try { if ( index != -1 ) { // -1 for not found ChangePP(index, false); comboBoxDraws->blockSignals(true); comboBoxDraws->setCurrentIndex(index); comboBoxDraws->blockSignals(false); } else { ChangePP(0, false); } } catch (VExceptionBadId &e) { qCCritical(vMainWindow, "%s\n\n%s\n\n%s", qUtf8Printable(tr("Bad id.")), qUtf8Printable(e.ErrorMessage()), qUtf8Printable(e.DetailedInformation())); SetEnabledGUI(false); if (not VApplication::IsGUIMode()) { qApp->exit(V_EX_NOINPUT); } return; } catch (const VExceptionEmptyParameter &e) { qCCritical(vMainWindow, "%s\n\n%s\n\n%s", qUtf8Printable(tr("Error empty parameter.")), qUtf8Printable(e.ErrorMessage()), qUtf8Printable(e.DetailedInformation())); SetEnabledGUI(false); if (not VApplication::IsGUIMode()) { qApp->exit(V_EX_NOINPUT); } return; } } //--------------------------------------------------------------------------------------------------------------------- void MainWindow::PreviousPatternPiece() { int index = comboBoxDraws->currentIndex(); if (index == -1 || comboBoxDraws->count() <= 1) { return; } if (index == 0) { index = comboBoxDraws->count() - 1; } else { --index; } comboBoxDraws->setCurrentIndex(index); } //--------------------------------------------------------------------------------------------------------------------- void MainWindow::NextPatternPiece() { int index = comboBoxDraws->currentIndex(); if (index == -1 || comboBoxDraws->count() <= 1) { return; } if (index == comboBoxDraws->count()-1) { index = 0; } else { ++index; } comboBoxDraws->setCurrentIndex(index); } //--------------------------------------------------------------------------------------------------------------------- void MainWindow::SetEnabledGUI(bool enabled) { if (guiEnabled != enabled) { if (enabled == false) { ArrowTool(true); qApp->getUndoStack()->clear(); } SetEnableWidgets(enabled); guiEnabled = enabled; SetEnableTool(enabled); ui->toolBarOption->setEnabled(enabled); #ifndef QT_NO_CURSOR QGuiApplication::setOverrideCursor(Qt::ArrowCursor); #endif } } //--------------------------------------------------------------------------------------------------------------------- /** * @brief SetEnableWidgets enable action button. * @param enable enable value. */ void MainWindow::SetEnableWidgets(bool enable) { const bool drawStage = (mode == Draw::Calculation); const bool designStage = (drawStage || mode == Draw::Modeling); comboBoxDraws->setEnabled(enable && drawStage); ui->actionOptionDraw->setEnabled(enable && drawStage); ui->actionSave->setEnabled(isWindowModified() && enable && not patternReadOnly); ui->actionSaveAs->setEnabled(enable); ui->actionPattern_properties->setEnabled(enable && designStage); ui->actionZoomIn->setEnabled(enable); ui->actionZoomOut->setEnabled(enable); ui->actionArrowTool->setEnabled(enable && designStage); ui->actionHistory->setEnabled(enable && drawStage); ui->actionNewDraw->setEnabled(enable && drawStage); ui->actionDraw->setEnabled(enable); ui->actionDetails->setEnabled(enable); ui->actionLayout->setEnabled(enable); ui->actionTable->setEnabled(enable && drawStage); ui->actionExportIncrementsToCSV->setEnabled(enable); ui->actionExportFinalMeasurementsToCSV->setEnabled(enable); ui->actionFinalMeasurements->setEnabled(enable); ui->actionZoomFitBest->setEnabled(enable); ui->actionZoomFitBestCurrent->setEnabled(enable && drawStage); ui->actionZoomOriginal->setEnabled(enable); ui->actionShowCurveDetails->setEnabled(enable && drawStage); ui->actionLoadIndividual->setEnabled(enable && designStage); ui->actionLoadMultisize->setEnabled(enable && designStage); ui->actionUnloadMeasurements->setEnabled(enable && designStage); ui->actionPreviousPatternPiece->setEnabled(enable && drawStage); ui->actionNextPatternPiece->setEnabled(enable && drawStage); actionDockWidgetToolOptions->setEnabled(enable && designStage); actionDockWidgetGroups->setEnabled(enable && designStage); undoAction->setEnabled(enable && designStage && qApp->getUndoStack()->canUndo()); redoAction->setEnabled(enable && designStage && qApp->getUndoStack()->canRedo()); //Now we don't want allow user call context menu sceneDraw->SetDisableTools(!enable, doc->GetNameActivPP()); ui->view->setEnabled(enable); } //--------------------------------------------------------------------------------------------------------------------- void MainWindow::UpdateHeightsList(const QStringList &list) { QString val; if (gradationHeights->currentIndex() != -1) { val = gradationHeights->currentText(); } gradationHeights->blockSignals(true); gradationHeights->clear(); gradationHeights->addItems(list); gradationHeights->blockSignals(false); int index = gradationHeights->findText(val); if (index != -1) { gradationHeights->setCurrentIndex(index); } else { ChangedHeight(list.at(0)); } } //--------------------------------------------------------------------------------------------------------------------- void MainWindow::UpdateSizesList(const QStringList &list) { QString val; if (gradationSizes->currentIndex() != -1) { val = gradationSizes->currentText(); } gradationSizes->blockSignals(true); gradationSizes->clear(); gradationSizes->addItems(list); gradationSizes->blockSignals(false); int index = gradationSizes->findText(val); if (index != -1) { gradationSizes->setCurrentIndex(index); } else { ChangedSize(list.at(0)); } } //--------------------------------------------------------------------------------------------------------------------- /** * @brief NewPattern create new empty pattern. */ void MainWindow::New() { if (comboBoxDraws->count() == 0) { qCDebug(vMainWindow, "New PP."); QString patternPieceName = tr("Pattern piece %1").arg(comboBoxDraws->count()+1); qCDebug(vMainWindow, "Generated PP name: %s", qUtf8Printable(patternPieceName)); qCDebug(vMainWindow, "First PP"); DialogNewPattern newPattern(pattern, patternPieceName, this); if (newPattern.exec() == QDialog::Accepted) { patternPieceName = newPattern.name(); qApp->setPatternUnit(newPattern.PatternUnit()); qCDebug(vMainWindow, "PP name: %s", qUtf8Printable(patternPieceName)); } else { qCDebug(vMainWindow, "Creation a new pattern was canceled."); return; } //Set scene size to size scene view VMainGraphicsView::NewSceneRect(sceneDraw, ui->view); VMainGraphicsView::NewSceneRect(sceneDetails, ui->view); AddPP(patternPieceName); mouseCoordinate = new QLabel(QString("0, 0 (%1)").arg(UnitsToStr(qApp->patternUnit(), true))); ui->toolBarOption->addWidget(mouseCoordinate); m_curFileFormatVersion = VPatternConverter::PatternMaxVer; m_curFileFormatVersionStr = VPatternConverter::PatternMaxVerStr; ToolBarOption(); } else { OpenNewValentina(); } } //--------------------------------------------------------------------------------------------------------------------- /** * @brief haveChange enable action save if we have unsaved change. */ void MainWindow::PatternChangesWereSaved(bool saved) { if (guiEnabled) { const bool state = doc->IsModified() || !saved; setWindowModified(state); not patternReadOnly ? ui->actionSave->setEnabled(state): ui->actionSave->setEnabled(false); isLayoutStale = true; } } //--------------------------------------------------------------------------------------------------------------------- /** * @brief ChangedSize change new size value. * @param text value size. */ void MainWindow::ChangedSize(const QString & text) { const int size = static_cast(VContainer::size()); if (UpdateMeasurements(AbsoluteMPath(qApp->GetPPath(), doc->MPath()), text.toInt(), static_cast(VContainer::height()))) { doc->LiteParseTree(Document::LiteParse); emit sceneDetails->DimensionsChanged(); } else { qCWarning(vMainWindow, "%s", qUtf8Printable(tr("Couldn't update measurements."))); const qint32 index = gradationSizes->findText(QString().setNum(size)); if (index != -1) { gradationSizes->setCurrentIndex(index); } else { qCDebug(vMainWindow, "Couldn't restore size value."); } } } //--------------------------------------------------------------------------------------------------------------------- /** * @brief ChangedGrowth change new height value. * @param text value height. */ void MainWindow::ChangedHeight(const QString &text) { const int height = static_cast(VContainer::height()); if (UpdateMeasurements(AbsoluteMPath(qApp->GetPPath(), doc->MPath()), static_cast(VContainer::size()), text.toInt())) { doc->LiteParseTree(Document::LiteParse); emit sceneDetails->DimensionsChanged(); } else { qCWarning(vMainWindow, "%s", qUtf8Printable(tr("Couldn't update measurements."))); const qint32 index = gradationHeights->findText(QString().setNum(height)); if (index != -1) { gradationHeights->setCurrentIndex(index); } else { qCDebug(vMainWindow, "Couldn't restore height value."); } } } //--------------------------------------------------------------------------------------------------------------------- void MainWindow::ShowProgress() { if (m_progressBar->isVisible() && m_progressBar->value() + 1 <= m_progressBar->maximum()) { const int newValue = m_progressBar->value() + 1; m_progressBar->setValue(newValue); #if defined(Q_OS_WIN32) && QT_VERSION >= QT_VERSION_CHECK(5, 7, 0) m_taskbarProgress->setValue(newValue); #endif qApp->processEvents(); } } //--------------------------------------------------------------------------------------------------------------------- void MainWindow::SetDefaultHeight() { const QString defHeight = QString().setNum(doc->GetDefCustomHeight()); int index = gradationHeights->findText(defHeight); if (index != -1) { gradationHeights->setCurrentIndex(index); } else { index = gradationHeights->findText(QString().setNum(VContainer::height())); if (index != -1) { gradationHeights->setCurrentIndex(index); } } VContainer::SetHeight(gradationHeights->currentText().toInt()); } //--------------------------------------------------------------------------------------------------------------------- void MainWindow::SetDefaultSize() { const QString defSize = QString().setNum(doc->GetDefCustomSize()); int index = gradationSizes->findText(defSize); if (index != -1) { gradationSizes->setCurrentIndex(index); } else { index = gradationSizes->findText(QString().setNum(VContainer::size())); if (index != -1) { gradationSizes->setCurrentIndex(index); } } VContainer::SetSize(gradationSizes->currentText().toInt()); } //--------------------------------------------------------------------------------------------------------------------- /** * @brief SetEnableTool enable button. * @param enable enable value. */ void MainWindow::SetEnableTool(bool enable) { bool drawTools = false; bool modelingTools = false; bool layoutTools = false; QT_WARNING_PUSH QT_WARNING_DISABLE_GCC("-Wswitch-default") switch (mode) { case Draw::Calculation: drawTools = enable; break; case Draw::Modeling: modelingTools = enable; break; case Draw::Layout: layoutTools = enable; break; } QT_WARNING_POP // This check helps to find missed tools Q_STATIC_ASSERT_X(static_cast(Tool::LAST_ONE_DO_NOT_USE) == 55, "Not all tools were handled."); //Drawing Tools ui->toolButtonEndLine->setEnabled(drawTools); ui->toolButtonLine->setEnabled(drawTools); ui->toolButtonAlongLine->setEnabled(drawTools); ui->toolButtonShoulderPoint->setEnabled(drawTools); ui->toolButtonNormal->setEnabled(drawTools); ui->toolButtonBisector->setEnabled(drawTools); ui->toolButtonLineIntersect->setEnabled(drawTools); ui->toolButtonSpline->setEnabled(drawTools); ui->toolButtonCubicBezier->setEnabled(drawTools); ui->toolButtonArc->setEnabled(drawTools); ui->toolButtonSplinePath->setEnabled(drawTools); ui->toolButtonCubicBezierPath->setEnabled(drawTools); ui->toolButtonPointOfContact->setEnabled(drawTools); ui->toolButtonNewDetail->setEnabled(drawTools); ui->toolButtonInternalPath->setEnabled(drawTools); ui->toolButtonHeight->setEnabled(drawTools); ui->toolButtonTriangle->setEnabled(drawTools); ui->toolButtonPointOfIntersection->setEnabled(drawTools); ui->toolButtonSplineCutPoint->setEnabled(drawTools); ui->toolButtonSplinePathCutPoint->setEnabled(drawTools); ui->toolButtonArcCutPoint->setEnabled(drawTools); ui->toolButtonLineIntersectAxis->setEnabled(drawTools); ui->toolButtonCurveIntersectAxis->setEnabled(drawTools); ui->toolButtonArcIntersectAxis->setEnabled(drawTools); ui->toolButtonPointOfIntersectionArcs->setEnabled(drawTools); ui->toolButtonIntersectionCurves->setEnabled(drawTools); ui->toolButtonPointOfIntersectionCircles->setEnabled(drawTools); ui->toolButtonPointFromCircleAndTangent->setEnabled(drawTools); ui->toolButtonPointFromArcAndTangent->setEnabled(drawTools); ui->toolButtonArcWithLength->setEnabled(drawTools); ui->toolButtonTrueDarts->setEnabled(drawTools); ui->toolButtonGroup->setEnabled(drawTools); ui->toolButtonRotation->setEnabled(drawTools); ui->toolButtonFlippingByLine->setEnabled(drawTools); ui->toolButtonFlippingByAxis->setEnabled(drawTools); ui->toolButtonMove->setEnabled(drawTools); ui->toolButtonMidpoint->setEnabled(drawTools); ui->toolButtonEllipticalArc->setEnabled(drawTools); ui->toolButtonPin->setEnabled(drawTools); ui->toolButtonInsertNode->setEnabled(drawTools); ui->toolButtonPlaceLabel->setEnabled(drawTools); ui->actionLast_tool->setEnabled(drawTools); for (auto pointer : qAsConst(toolButtonPointerList)) { pointer->setEnabled(drawTools || modelingTools); pointer->setChecked(drawTools || modelingTools); } //Modeling Tools ui->toolButtonUnionDetails->setEnabled(modelingTools); ui->toolButtonDetailExportAs->setEnabled(modelingTools); ui->toolButtonDuplicateDetail->setEnabled(modelingTools); //Layout tools ui->toolButtonLayoutSettings->setEnabled(layoutTools); } //--------------------------------------------------------------------------------------------------------------------- void MainWindow::SetLayoutModeActions() { const bool enabled = not scenes.isEmpty(); ui->toolButtonLayoutExportAs->setEnabled(enabled); ui->actionExportAs->setEnabled(enabled); ui->actionPrintPreview->setEnabled(enabled); ui->actionPrintPreviewTiled->setEnabled(enabled); ui->actionPrint->setEnabled(enabled); ui->actionPrintTiled->setEnabled(enabled); } //--------------------------------------------------------------------------------------------------------------------- /** * @brief MinimumScrollBar set scroll bar to minimum. */ void MainWindow::MinimumScrollBar() { QScrollBar *horScrollBar = ui->view->horizontalScrollBar(); horScrollBar->setValue(horScrollBar->minimum()); QScrollBar *verScrollBar = ui->view->verticalScrollBar(); verScrollBar->setValue(verScrollBar->minimum()); } //--------------------------------------------------------------------------------------------------------------------- /** * @brief SavePattern save pattern file. * @param fileName pattern file name. * @return true if all is good. */ bool MainWindow::SavePattern(const QString &fileName, QString &error) { qCDebug(vMainWindow, "Saving pattern file %s.", qUtf8Printable(fileName)); QFileInfo tempInfo(fileName); const QString mPath = AbsoluteMPath(qApp->GetPPath(), doc->MPath()); if (not mPath.isEmpty() && qApp->GetPPath() != fileName) { doc->SetMPath(RelativeMPath(fileName, mPath)); } const bool result = doc->SaveDocument(fileName, error); if (result) { if (tempInfo.suffix() != QLatin1String("autosave")) { setCurrentFile(fileName); statusBar()->showMessage(tr("File saved"), 5000); qCDebug(vMainWindow, "File %s saved.", qUtf8Printable(fileName)); PatternChangesWereSaved(result); } } else { doc->SetMPath(mPath); emit doc->UpdatePatternLabel(); qCDebug(vMainWindow, "Could not save file %s. %s.", qUtf8Printable(fileName), qUtf8Printable(error)); } return result; } //--------------------------------------------------------------------------------------------------------------------- /** * @brief AutoSavePattern start safe saving. */ void MainWindow::AutoSavePattern() { qCDebug(vMainWindow, "Autosaving pattern."); if (qApp->GetPPath().isEmpty() == false && this->isWindowModified() == true) { QString autofile = qApp->GetPPath() + *autosavePrefix; QString error; SavePattern(autofile, error); } } //--------------------------------------------------------------------------------------------------------------------- /** * @brief setCurrentFile the function is called to reset the state of a few variables when a file * is loaded or saved, or when the user starts editing a new file (in which case fileName is empty). * @param fileName file name. */ void MainWindow::setCurrentFile(const QString &fileName) { qCDebug(vMainWindow, "Set current name to \"%s\"", qUtf8Printable(fileName)); qApp->SetPPath(fileName); doc->SetPatternWasChanged(true); emit doc->UpdatePatternLabel(); qApp->getUndoStack()->setClean(); if (not qApp->GetPPath().isEmpty() && VApplication::IsGUIMode()) { qCDebug(vMainWindow, "Updating recent file list."); VSettings *settings = qApp->ValentinaSettings(); QStringList files = settings->GetRecentFileList(); files.removeAll(fileName); files.prepend(fileName); while (files.size() > MaxRecentFiles) { files.removeLast(); } settings->SetRecentFileList(files); UpdateRecentFileActions(); qCDebug(vMainWindow, "Updating restore file list."); QStringList restoreFiles = settings->GetRestoreFileList(); restoreFiles.removeAll(fileName); restoreFiles.prepend(fileName); settings->SetRestoreFileList(restoreFiles); } UpdateWindowTitle(); } //--------------------------------------------------------------------------------------------------------------------- /** * @brief ReadSettings read setting for app. */ void MainWindow::ReadSettings() { qCDebug(vMainWindow, "Reading settings."); const VSettings *settings = qApp->ValentinaSettings(); restoreGeometry(settings->GetGeometry()); restoreState(settings->GetWindowState()); restoreState(settings->GetToolbarsState(), APP_VERSION); // Scene antialiasing ui->view->SetAntialiasing(settings->GetGraphicalOutput()); // Stack limit qApp->getUndoStack()->setUndoLimit(settings->GetUndoCount()); // Text under tool buton icon ToolBarStyles(); isDockToolOptionsVisible = ui->dockWidgetToolOptions->isVisible(); isDockGroupsVisible = ui->dockWidgetGroups->isVisible(); } //--------------------------------------------------------------------------------------------------------------------- /** * @brief WriteSettings save setting for app. */ void MainWindow::WriteSettings() { ActionDraw(true); VSettings *setings = qApp->ValentinaSettings(); setings->SetGeometry(saveGeometry()); setings->SetWindowState(saveState()); setings->SetToolbarsState(saveState(APP_VERSION)); } //--------------------------------------------------------------------------------------------------------------------- /** * @brief MaybeSave The function is called to save pending changes. * @return returns true in all cases, except when the user clicks Cancel. */ bool MainWindow::MaybeSave() { if (this->isWindowModified() && guiEnabled) { QScopedPointer messageBox(new QMessageBox(tr("Unsaved changes"), tr("The pattern has been modified.\n" "Do you want to save your changes?"), QMessageBox::Warning, QMessageBox::Yes, QMessageBox::No, QMessageBox::Cancel, this, Qt::Sheet)); messageBox->setDefaultButton(QMessageBox::Yes); messageBox->setEscapeButton(QMessageBox::Cancel); messageBox->setButtonText(QMessageBox::Yes, qApp->GetPPath().isEmpty() || patternReadOnly ? tr("Save…") : tr("Save")); messageBox->setButtonText(QMessageBox::No, tr("Don't Save")); messageBox->setWindowModality(Qt::ApplicationModal); const auto ret = static_cast(messageBox->exec()); switch (ret) { case QMessageBox::Yes: if (patternReadOnly) { return SaveAs(); } else { return Save(); } case QMessageBox::No: return true; case QMessageBox::Cancel: return false; default: break; } } return true; } //--------------------------------------------------------------------------------------------------------------------- void MainWindow::UpdateRecentFileActions() { qCDebug(vMainWindow, "Updating recent file actions."); const QStringList files = qApp->ValentinaSettings()->GetRecentFileList(); const int numRecentFiles = qMin(files.size(), static_cast(MaxRecentFiles)); for (int i = 0; i < numRecentFiles; ++i) { QString text = QString("&%1. %2").arg(i + 1).arg(StrippedName(files.at(i))); recentFileActs[i]->setText(text); recentFileActs[i]->setData(files.at(i)); recentFileActs[i]->setVisible(true); } for (int j = numRecentFiles; j < MaxRecentFiles; ++j) { recentFileActs[j]->setVisible(false); } separatorAct->setVisible(numRecentFiles > 0); } //--------------------------------------------------------------------------------------------------------------------- void MainWindow::CreateMenus() { for (int i = 0; i < MaxRecentFiles; ++i) { ui->menuFile->insertAction(ui->actionPreferences, recentFileActs[i]); } separatorAct = new QAction(this); separatorAct->setSeparator(true); ui->menuFile->insertAction(ui->actionPreferences, separatorAct); UpdateRecentFileActions(); //Add Undo/Redo actions. undoAction = qApp->getUndoStack()->createUndoAction(this, tr("&Undo")); connect(undoAction, &QAction::triggered, toolOptions, &VToolOptionsPropertyBrowser::RefreshOptions); undoAction->setShortcuts(QKeySequence::Undo); undoAction->setIcon(QIcon::fromTheme("edit-undo")); ui->menuPatternPiece->insertAction(ui->actionLast_tool, undoAction); ui->toolBarTools->addAction(undoAction); redoAction = qApp->getUndoStack()->createRedoAction(this, tr("&Redo")); connect(redoAction, &QAction::triggered, toolOptions, &VToolOptionsPropertyBrowser::RefreshOptions); redoAction->setShortcuts(QKeySequence::Redo); redoAction->setIcon(QIcon::fromTheme("edit-redo")); ui->menuPatternPiece->insertAction(ui->actionLast_tool, redoAction); ui->toolBarTools->addAction(redoAction); separatorAct = new QAction(this); separatorAct->setSeparator(true); ui->menuPatternPiece->insertAction(ui->actionPattern_properties, separatorAct); AddDocks(); } //--------------------------------------------------------------------------------------------------------------------- QT_WARNING_PUSH QT_WARNING_DISABLE_GCC("-Wswitch-default") void MainWindow::LastUsedTool() { // This check helps to find missed tools in the switch Q_STATIC_ASSERT_X(static_cast(Tool::LAST_ONE_DO_NOT_USE) == 55, "Not all tools were handled."); if (currentTool == lastUsedTool) { return; } switch ( lastUsedTool ) { case Tool::Arrow: for (auto pointer : qAsConst(toolButtonPointerList)) { pointer->setChecked(true); } ArrowTool(true); break; case Tool::BasePoint: case Tool::SinglePoint: case Tool::DoublePoint: case Tool::LinePoint: case Tool::AbstractSpline: case Tool::Cut: case Tool::LAST_ONE_DO_NOT_USE: case Tool::NodePoint: case Tool::NodeArc: case Tool::NodeElArc: case Tool::NodeSpline: case Tool::NodeSplinePath: Q_UNREACHABLE(); //-V501 //Nothing to do here because we can't create this tool from main window. break; case Tool::EndLine: ui->toolButtonEndLine->setChecked(true); ToolEndLine(true); break; case Tool::Line: ui->toolButtonLine->setChecked(true); ToolLine(true); break; case Tool::AlongLine: ui->toolButtonAlongLine->setChecked(true); ToolAlongLine(true); break; case Tool::Midpoint: ui->toolButtonMidpoint->setChecked(true); ToolMidpoint(true); break; case Tool::ShoulderPoint: ui->toolButtonShoulderPoint->setChecked(true); ToolShoulderPoint(true); break; case Tool::Normal: ui->toolButtonNormal->setChecked(true); ToolNormal(true); break; case Tool::Bisector: ui->toolButtonBisector->setChecked(true); ToolBisector(true); break; case Tool::LineIntersect: ui->toolButtonLineIntersect->setChecked(true); ToolLineIntersect(true); break; case Tool::Spline: ui->toolButtonSpline->setChecked(true); ToolSpline(true); break; case Tool::CubicBezier: ui->toolButtonCubicBezier->setChecked(true); ToolCubicBezier(true); break; case Tool::Arc: ui->toolButtonArc->setChecked(true); ToolArc(true); break; case Tool::SplinePath: ui->toolButtonSplinePath->setChecked(true); ToolSplinePath(true); break; case Tool::CubicBezierPath: ui->toolButtonCubicBezierPath->setChecked(true); ToolCubicBezierPath(true); break; case Tool::PointOfContact: ui->toolButtonPointOfContact->setChecked(true); ToolPointOfContact(true); break; case Tool::Piece: ui->toolButtonNewDetail->setChecked(true); ToolDetail(true); break; case Tool::PiecePath: ui->toolButtonInternalPath->setChecked(true); ToolPiecePath(true); break; case Tool::Height: ui->toolButtonHeight->setChecked(true); ToolHeight(true); break; case Tool::Triangle: ui->toolButtonTriangle->setChecked(true); ToolTriangle(true); break; case Tool::PointOfIntersection: ui->toolButtonPointOfIntersection->setChecked(true); ToolPointOfIntersection(true); break; case Tool::PointOfIntersectionArcs: ui->toolButtonPointOfIntersectionArcs->setChecked(true); ToolPointOfIntersectionArcs(true); break; case Tool::CutSpline: ui->toolButtonSplineCutPoint->setChecked(true); ToolCutSpline(true); break; case Tool::CutSplinePath: ui->toolButtonSplinePathCutPoint->setChecked(true); ToolCutSplinePath(true); break; case Tool::UnionDetails: ui->toolButtonUnionDetails->setChecked(true); ToolUnionDetails(true); break; case Tool::DuplicateDetail: ui->toolButtonDuplicateDetail->setChecked(true); ToolDuplicateDetail(true); break; case Tool::CutArc: ui->toolButtonArcCutPoint->setChecked(true); ToolCutArc(true); break; case Tool::LineIntersectAxis: ui->toolButtonLineIntersectAxis->setChecked(true); ToolLineIntersectAxis(true); break; case Tool::CurveIntersectAxis: ui->toolButtonCurveIntersectAxis->setChecked(true); ToolCurveIntersectAxis(true); break; case Tool::ArcIntersectAxis: ui->toolButtonArcIntersectAxis->setChecked(true); ToolArcIntersectAxis(true); break; case Tool::PointOfIntersectionCircles: ui->toolButtonPointOfIntersectionCircles->setChecked(true); ToolPointOfIntersectionCircles(true); break; case Tool::PointOfIntersectionCurves: ui->toolButtonIntersectionCurves->setChecked(true); ToolPointOfIntersectionCurves(true); break; case Tool::PointFromCircleAndTangent: ui->toolButtonPointFromCircleAndTangent->setChecked(true); ToolPointFromCircleAndTangent(true); break; case Tool::PointFromArcAndTangent: ui->toolButtonPointFromArcAndTangent->setChecked(true); ToolPointFromArcAndTangent(true); break; case Tool::ArcWithLength: ui->toolButtonArcWithLength->setChecked(true); ToolArcWithLength(true); break; case Tool::TrueDarts: ui->toolButtonTrueDarts->setChecked(true); ToolTrueDarts(true); break; case Tool::Group: ui->toolButtonGroup->setChecked(true); ToolGroup(true); break; case Tool::Rotation: ui->toolButtonRotation->setChecked(true); ToolRotation(true); break; case Tool::FlippingByLine: ui->toolButtonFlippingByLine->setChecked(true); ToolFlippingByLine(true); break; case Tool::FlippingByAxis: ui->toolButtonFlippingByAxis->setChecked(true); ToolFlippingByAxis(true); break; case Tool::Move: ui->toolButtonMove->setChecked(true); ToolMove(true); break; case Tool::EllipticalArc: ui->toolButtonEllipticalArc->setChecked(true); ToolEllipticalArc(true); break; case Tool::Pin: ui->toolButtonPin->setChecked(true); ToolPin(true); break; case Tool::InsertNode: ui->toolButtonInsertNode->setChecked(true); ToolInsertNode(true); break; case Tool::PlaceLabel: ui->toolButtonPlaceLabel->setChecked(true); ToolPlaceLabel(true); break; } } QT_WARNING_POP //--------------------------------------------------------------------------------------------------------------------- void MainWindow::AddDocks() { //Add dock actionDockWidgetToolOptions = ui->dockWidgetToolOptions->toggleViewAction(); ui->menuPatternPiece->insertAction(ui->actionPattern_properties, actionDockWidgetToolOptions); connect(ui->dockWidgetToolOptions, &QDockWidget::visibilityChanged, this, [this](bool visible) { isDockToolOptionsVisible = visible; }); actionDockWidgetGroups = ui->dockWidgetGroups->toggleViewAction(); ui->menuPatternPiece->insertAction(ui->actionPattern_properties, actionDockWidgetGroups); connect(ui->dockWidgetGroups, &QDockWidget::visibilityChanged, this, [this](bool visible) { isDockGroupsVisible = visible; }); separatorAct = new QAction(this); separatorAct->setSeparator(true); ui->menuPatternPiece->insertAction(ui->actionPattern_properties, separatorAct); } //--------------------------------------------------------------------------------------------------------------------- void MainWindow::InitDocksContain() { qCDebug(vMainWindow, "Initialization property browser."); toolOptions = new VToolOptionsPropertyBrowser(ui->dockWidgetToolOptions); connect(ui->view, &VMainGraphicsView::itemClicked, toolOptions, &VToolOptionsPropertyBrowser::itemClicked); connect(doc, &VPattern::FullUpdateFromFile, toolOptions, &VToolOptionsPropertyBrowser::UpdateOptions); qCDebug(vMainWindow, "Initialization groups dock."); groupsWidget = new VWidgetGroups(doc, this); ui->dockWidgetGroups->setWidget(groupsWidget); connect(doc,&VAbstractPattern::UpdateGroups , this, &MainWindow::UpdateVisibilityGroups); detailsWidget = new VWidgetDetails(pattern, doc, this); connect(doc, &VPattern::FullUpdateFromFile, detailsWidget, &VWidgetDetails::UpdateList); connect(doc, &VPattern::UpdateInLayoutList, detailsWidget, &VWidgetDetails::UpdateList); connect(doc, &VPattern::ShowDetail, detailsWidget, &VWidgetDetails::SelectDetail); connect(detailsWidget, &VWidgetDetails::Highlight, sceneDetails, &VMainGraphicsScene::HighlightItem); detailsWidget->setVisible(false); } //--------------------------------------------------------------------------------------------------------------------- bool MainWindow::OpenNewValentina(const QString &fileName) const { if (this->isWindowModified() || qApp->GetPPath().isEmpty() == false) { VApplication::NewValentina(fileName); return true; } return false; } //--------------------------------------------------------------------------------------------------------------------- void MainWindow::CreateActions() { ui->setupUi(this); connect(ui->actionArrowTool, &QAction::triggered, this, &MainWindow::ArrowTool); connect(ui->actionDraw, &QAction::triggered, this, &MainWindow::ActionDraw); connect(ui->actionDetails, &QAction::triggered, this, &MainWindow::ActionDetails); connect(ui->actionLayout, &QAction::triggered, this, &MainWindow::ActionLayout); connect(ui->actionHistory, &QAction::triggered, this, [this](bool checked) { if (checked) { dialogHistory = new DialogHistory(pattern, doc, this); dialogHistory->setWindowFlags(Qt::Window); connect(this, &MainWindow::RefreshHistory, dialogHistory.data(), &DialogHistory::UpdateHistory); connect(dialogHistory.data(), &DialogHistory::DialogClosed, this, [this]() { ui->actionHistory->setChecked(false); delete dialogHistory; }); // Fix issue #526. Dialog Detail is not on top after selection second object on Mac. dialogHistory->setWindowFlags(dialogHistory->windowFlags() | Qt::WindowStaysOnTopHint); dialogHistory->show(); } else { ui->actionHistory->setChecked(true); dialogHistory->activateWindow(); } }); connect(ui->actionNewDraw, &QAction::triggered, this, [this]() { qCDebug(vMainWindow, "New PP."); QString patternPieceName = tr("Pattern piece %1").arg(comboBoxDraws->count()+1); qCDebug(vMainWindow, "Generated PP name: %s", qUtf8Printable(patternPieceName)); qCDebug(vMainWindow, "PP count %d", comboBoxDraws->count()); patternPieceName = PatternPieceName(patternPieceName); qCDebug(vMainWindow, "PP name: %s", qUtf8Printable(patternPieceName)); if (patternPieceName.isEmpty()) { qCDebug(vMainWindow, "Name empty."); return; } AddPP(patternPieceName); }); connect(ui->actionSaveAs, &QAction::triggered, this, &MainWindow::SaveAs); connect(ui->actionSave, &QAction::triggered, this, &MainWindow::Save); connect(ui->actionOpen, &QAction::triggered, this, &MainWindow::Open); connect(ui->actionNew, &QAction::triggered, this, &MainWindow::New); connect(ui->actionExportIncrementsToCSV, &QAction::triggered, this, &MainWindow::ExportDataToCSV); connect(ui->actionExportFinalMeasurementsToCSV, &QAction::triggered, this, &MainWindow::ExportFMeasurementsToCSV); connect(ui->actionTable, &QAction::triggered, this, [this](bool checked) { if (checked) { dialogTable = new DialogIncrements(pattern, doc, this); connect(dialogTable.data(), &DialogIncrements::UpdateProperties, toolOptions, &VToolOptionsPropertyBrowser::RefreshOptions); connect(dialogTable.data(), &DialogIncrements::DialogClosed, this, [this]() { ui->actionTable->setChecked(false); delete dialogTable; }); dialogTable->show(); } else { ui->actionTable->setChecked(true); dialogTable->activateWindow(); } }); connect(ui->actionFinalMeasurements, &QAction::triggered, this, [this]() { if (dialogFMeasurements.isNull()) { dialogFMeasurements = new DialogFinalMeasurements(doc, this); dialogFMeasurements->setAttribute(Qt::WA_DeleteOnClose); connect(dialogFMeasurements.data(), &DialogFinalMeasurements::finished, this, [this](int result) { if (result == QDialog::Accepted) { doc->SetFinalMeasurements(dialogFMeasurements->FinalMeasurements()); } dialogFMeasurements->close(); }); dialogFMeasurements->show(); } else { dialogFMeasurements->activateWindow(); } }); connect(ui->actionAbout_Qt, &QAction::triggered, this, [this]() { QMessageBox::aboutQt(this, tr("About Qt")); }); connect(ui->actionAbout_Valentina, &QAction::triggered, this, [this]() { DialogAboutApp *aboutDialog = new DialogAboutApp(this); aboutDialog->setAttribute(Qt::WA_DeleteOnClose, true); aboutDialog->show(); }); connect(ui->actionExit, &QAction::triggered, this, &MainWindow::close); connect(ui->actionPreferences, &QAction::triggered, this, &MainWindow::Preferences); connect(ui->actionReportBug, &QAction::triggered, this, []() { qCDebug(vMainWindow, "Reporting bug"); QDesktopServices::openUrl(QUrl(QStringLiteral("https://bitbucket.org/dismine/valentina/issues/new"))); }); connect(ui->actionForum, &QAction::triggered, this, []() { qCDebug(vMainWindow, "Opening forum"); QDesktopServices::openUrl(QUrl(QStringLiteral("http://valentinaproject.forumotion.me/"))); }); connect(ui->actionLast_tool, &QAction::triggered, this, &MainWindow::LastUsedTool); connect(ui->actionPattern_properties, &QAction::triggered, this, [this]() { DialogPatternProperties proper(doc, pattern, this); connect(&proper, &DialogPatternProperties::UpdateGradation, this, [this]() { UpdateHeightsList(VMeasurement::ListHeights(doc->GetGradationHeights(), qApp->patternUnit())); UpdateSizesList(VMeasurement::ListSizes(doc->GetGradationSizes(), qApp->patternUnit())); }); proper.exec(); }); ui->actionPattern_properties->setEnabled(false); connect(ui->actionClosePattern, &QAction::triggered, this, [this]() { if (MaybeSave()) { FileClosedCorrect(); Clear(); } }); ui->actionShowCurveDetails->setChecked(qApp->ValentinaSettings()->IsShowCurveDetails()); connect(ui->actionShowCurveDetails, &QAction::triggered, this, [this](bool checked) { emit ui->view->itemClicked(nullptr); sceneDraw->EnableDetailsMode(checked); qApp->ValentinaSettings()->SetShowCurveDetails(checked); }); connect(ui->actionLoadIndividual, &QAction::triggered, this, &MainWindow::LoadIndividual); connect(ui->actionLoadMultisize, &QAction::triggered, this, &MainWindow::LoadMultisize); connect(ui->actionOpenTape, &QAction::triggered, this, [this]() { const QString tape = qApp->TapeFilePath(); const QString workingDirectory = QFileInfo(tape).absoluteDir().absolutePath(); QStringList arguments; if (isNoScaling) { arguments.append(QLatin1String("--") + LONG_OPTION_NO_HDPI_SCALING); } QProcess::startDetached(tape, arguments, workingDirectory); }); connect(ui->actionEditCurrent, &QAction::triggered, this, &MainWindow::ShowMeasurements); connect(ui->actionExportAs, &QAction::triggered, this, &MainWindow::ExportLayoutAs); connect(ui->actionPrintPreview, &QAction::triggered, this, &MainWindow::PrintPreviewOrigin); connect(ui->actionPrintPreviewTiled, &QAction::triggered, this, &MainWindow::PrintPreviewTiled); connect(ui->actionPrint, &QAction::triggered, this, &MainWindow::PrintOrigin); connect(ui->actionPrintTiled, &QAction::triggered, this, &MainWindow::PrintTiled); //Actions for recent files loaded by a main window application. for (int i = 0; i < MaxRecentFiles; ++i) { QAction *action = new QAction(this); action->setVisible(false); recentFileActs[i] = action; connect(recentFileActs[i], &QAction::triggered, this, [this]() { if (QAction *action = qobject_cast(sender())) { const QString filePath = action->data().toString(); if (not filePath.isEmpty()) { LoadPattern(filePath); } } }); } connect(ui->actionSyncMeasurements, &QAction::triggered, this, &MainWindow::SyncMeasurements); connect(ui->actionUnloadMeasurements, &QAction::triggered, this, &MainWindow::UnloadMeasurements); connect(ui->actionLabelTemplateEditor, &QAction::triggered, this, [this]() { DialogEditLabel editor(doc); editor.exec(); }); } //--------------------------------------------------------------------------------------------------------------------- void MainWindow::InitAutoSave() { //Autosaving file each 1 minutes delete autoSaveTimer; autoSaveTimer = nullptr; autoSaveTimer = new QTimer(this); autoSaveTimer->setTimerType(Qt::VeryCoarseTimer); connect(autoSaveTimer, &QTimer::timeout, this, &MainWindow::AutoSavePattern); autoSaveTimer->stop(); if (qApp->ValentinaSettings()->GetAutosaveState()) { const qint32 autoTime = qApp->ValentinaSettings()->GetAutosaveTime(); autoSaveTimer->start(autoTime*60000); qCDebug(vMainWindow, "Autosaving each %d minutes.", autoTime); } qApp->setAutoSaveTimer(autoSaveTimer); } //--------------------------------------------------------------------------------------------------------------------- QString MainWindow::PatternPieceName(const QString &text) { QInputDialog *dlg = new QInputDialog(this); dlg->setInputMode( QInputDialog::TextInput ); dlg->setLabelText(tr("Pattern piece:")); dlg->setTextEchoMode(QLineEdit::Normal); dlg->setWindowTitle(tr("Enter a new label for the pattern piece.")); dlg->resize(300, 100); dlg->setTextValue(text); QString nameDraw; while (1) { const bool bOk = dlg->exec(); nameDraw = dlg->textValue(); if (bOk == false || nameDraw.isEmpty()) { delete dlg; return QString(); } if (comboBoxDraws->findText(nameDraw) == -1) { break;//repeate show dialog } } delete dlg; return nameDraw; } //--------------------------------------------------------------------------------------------------------------------- MainWindow::~MainWindow() { CancelTool(); CleanLayout(); delete doc; delete ui; } //--------------------------------------------------------------------------------------------------------------------- /** * @brief LoadPattern open pattern file. * @param fileName name of file. */ bool MainWindow::LoadPattern(QString fileName, const QString& customMeasureFile) { qCDebug(vMainWindow, "Loading new file %s.", qUtf8Printable(fileName)); { // Convert to absolute path if need QFileInfo info(fileName); if (info.exists() && info.isRelative()) { fileName = QFileInfo(QDir::currentPath() + QLatin1Char('/') + fileName).canonicalFilePath(); } } //We have unsaved changes or load more then one file per time if (OpenNewValentina(fileName)) { return false; } if (fileName.isEmpty()) { qCDebug(vMainWindow, "Got empty file."); Clear(); return false; } try { // Here comes undocumented Valentina's feature. // Because app bundle in Mac OS X doesn't allow setup assosiation for Tape we must do this through Valentina VMeasurements m(pattern); m.SetSize(VContainer::rsize()); m.SetHeight(VContainer::rheight()); m.setXMLContent(fileName); if (m.Type() == MeasurementsType::Multisize || m.Type() == MeasurementsType::Individual) { const QString tape = qApp->TapeFilePath(); const QString workingDirectory = QFileInfo(tape).absoluteDir().absolutePath(); QStringList arguments = QStringList() << fileName; if (isNoScaling) { arguments.append(QLatin1String("--") + LONG_OPTION_NO_HDPI_SCALING); } QProcess::startDetached(tape, arguments, workingDirectory); qApp->exit(V_EX_OK); return false; // stop continue processing } } catch (VException &e) { qCCritical(vMainWindow, "%s\n\n%s\n\n%s", qUtf8Printable(tr("File error.")), qUtf8Printable(e.ErrorMessage()), qUtf8Printable(e.DetailedInformation())); Clear(); if (not VApplication::IsGUIMode()) { qApp->exit(V_EX_NOINPUT); } return false; } qCDebug(vMainWindow, "Loking file"); VlpCreateLock(lock, fileName); if (lock->IsLocked()) { qCDebug(vMainWindow, "Pattern file %s was locked.", qUtf8Printable(fileName)); } else { if (not IgnoreLocking(lock->GetLockError(), fileName)) { return false; } } // On this stage scene empty. Fit scene size to view size VMainGraphicsView::NewSceneRect(sceneDraw, ui->view); VMainGraphicsView::NewSceneRect(sceneDetails, ui->view); qApp->setOpeningPattern();//Begin opening file try { VPatternConverter converter(fileName); m_curFileFormatVersion = converter.GetCurrentFormatVersion(); m_curFileFormatVersionStr = converter.GetVersionStr(); doc->setXMLContent(converter.Convert()); if (!customMeasureFile.isEmpty()) { doc->SetMPath(RelativeMPath(fileName, customMeasureFile)); } qApp->setPatternUnit(doc->MUnit()); const QString path = AbsoluteMPath(fileName, doc->MPath()); if (not path.isEmpty()) { // Check if exist const QString newPath = CheckPathToMeasurements(fileName, path); if (newPath.isEmpty()) { qApp->setOpeningPattern();// End opening file Clear(); qCCritical(vMainWindow, "%s", qUtf8Printable(tr("The measurements file '%1' could not be found.") .arg(path))); if (not VApplication::IsGUIMode()) { qApp->exit(V_EX_NOINPUT); } return false; } if (not LoadMeasurements(newPath)) { qCCritical(vMainWindow, "%s", qUtf8Printable(tr("The measurements file '%1' could not be found or " "provides not enough information.") .arg(newPath))); qApp->setOpeningPattern();// End opening file Clear(); if (not VApplication::IsGUIMode()) { qApp->exit(V_EX_NOINPUT); } return false; } else { ui->actionUnloadMeasurements->setEnabled(true); watcher->addPath(path); ui->actionEditCurrent->setEnabled(true); } } if (qApp->patternType() == MeasurementsType::Unknown) {// Show toolbar only if was not uploaded any measurements. ToolBarOption(); } } catch (VException &e) { qCCritical(vMainWindow, "%s\n\n%s\n\n%s", qUtf8Printable(tr("File error.")), qUtf8Printable(e.ErrorMessage()), qUtf8Printable(e.DetailedInformation())); qApp->setOpeningPattern();// End opening file Clear(); if (not VApplication::IsGUIMode()) { qApp->exit(V_EX_NOINPUT); } return false; } m_progressBar->setVisible(true); m_statusLabel->setVisible(false); const int elements = doc->ElementsToParse(); m_progressBar->setMaximum(elements); #if defined(Q_OS_WIN32) && QT_VERSION >= QT_VERSION_CHECK(5, 7, 0) m_taskbarProgress->setVisible(true); m_taskbarProgress->setMaximum(elements); #endif FullParseFile(); /* Collect garbage only after successfully parse. This way wrongly accused items have one more time to restore * a reference. */ doc->GarbageCollector(true); m_progressBar->setVisible(false); #if defined(Q_OS_WIN32) && QT_VERSION >= QT_VERSION_CHECK(5, 7, 0) m_taskbarProgress->setVisible(false); #endif m_statusLabel->setVisible(true); if (guiEnabled) { // No errors occurred patternReadOnly = doc->IsReadOnly(); SetEnableWidgets(true); setCurrentFile(fileName); qCDebug(vMainWindow, "File loaded."); //Fit scene size to best size for first show ZoomFirstShow(); ActionDraw(true); qApp->setOpeningPattern();// End opening file return true; } else { qApp->setOpeningPattern();// End opening file return false; } } //--------------------------------------------------------------------------------------------------------------------- QStringList MainWindow::GetUnlokedRestoreFileList() const { QStringList restoreFiles; //Take all files that need to be restored QStringList files = qApp->ValentinaSettings()->GetRestoreFileList(); if (files.size() > 0) { for (auto &file : files) { // Seeking file that realy need reopen VLockGuard tmp(file); if (tmp.IsLocked()) { restoreFiles.append(file); } } // Clearing list after filtering for (auto &file : restoreFiles) { files.removeAll(file); } // Clear all files that do not exist. QStringList filtered; for (auto &file : files) { if (QFileInfo::exists(file)) { filtered.append(file); } } qApp->ValentinaSettings()->SetRestoreFileList(filtered); } return restoreFiles; } //--------------------------------------------------------------------------------------------------------------------- void MainWindow::ToolBarStyles() { ToolBarStyle(ui->toolBarDraws); ToolBarStyle(ui->toolBarOption); ToolBarStyle(ui->toolBarStages); ToolBarStyle(ui->toolBarTools); ToolBarStyle(ui->mainToolBar); } //--------------------------------------------------------------------------------------------------------------------- void MainWindow::ShowPaper(int index) { if (index < 0 || index >= scenes.size()) { ui->view->setScene(tempSceneLayout); } else { ui->view->setScene(scenes.at(index)); } ui->view->fitInView(ui->view->scene()->sceneRect(), Qt::KeepAspectRatio); } //--------------------------------------------------------------------------------------------------------------------- void MainWindow::Preferences() { // Calling constructor of the dialog take some time. Because of this user have time to call the dialog twice. static QPointer guard;// Prevent any second run if (guard.isNull()) { QGuiApplication::setOverrideCursor(QCursor(Qt::WaitCursor)); DialogPreferences *preferences = new DialogPreferences(this); // QScopedPointer needs to be sure any exception will never block guard QScopedPointer dlg(preferences); guard = preferences; connect(dlg.data(), &DialogPreferences::UpdateProperties, this, &MainWindow::WindowsLocale); // Must be first connect(dlg.data(), &DialogPreferences::UpdateProperties, toolOptions, &VToolOptionsPropertyBrowser::RefreshOptions); connect(dlg.data(), &DialogPreferences::UpdateProperties, this, &MainWindow::ToolBarStyles); connect(dlg.data(), &DialogPreferences::UpdateProperties, this, [this](){emit doc->FullUpdateFromFile();}); connect(dlg.data(), &DialogPreferences::UpdateProperties, ui->view, &VMainGraphicsView::ResetScrollingAnimation); QGuiApplication::restoreOverrideCursor(); if (guard->exec() == QDialog::Accepted) { InitAutoSave(); } } } //--------------------------------------------------------------------------------------------------------------------- #if defined(Q_OS_MAC) void MainWindow::CreateMeasurements() { const QString tape = qApp->TapeFilePath(); const QString workingDirectory = QFileInfo(tape).absoluteDir().absolutePath(); QStringList arguments; if (isNoScaling) { arguments.append(QLatin1String("--") + LONG_OPTION_NO_HDPI_SCALING); } QProcess::startDetached(tape, arguments, workingDirectory); } #endif //--------------------------------------------------------------------------------------------------------------------- void MainWindow::ExportLayoutAs() { if (isLayoutStale) { if (ContinueIfLayoutStale() == QMessageBox::No) { ui->toolButtonLayoutExportAs->setChecked(false); return; } } try { m_dialogSaveLayout = QSharedPointer(new DialogSaveLayout(scenes.size(), Draw::Layout, FileName(), this)); if (m_dialogSaveLayout->exec() == QDialog::Rejected) { m_dialogSaveLayout.clear(); ui->toolButtonLayoutExportAs->setChecked(false); return; } ExportData(QVector()); m_dialogSaveLayout.clear(); } catch (const VException &e) { m_dialogSaveLayout.clear(); ui->toolButtonLayoutExportAs->setChecked(false); qCritical("%s\n\n%s\n\n%s", qUtf8Printable(tr("Export error.")), qUtf8Printable(e.ErrorMessage()), qUtf8Printable(e.DetailedInformation())); return; } ui->toolButtonLayoutExportAs->setChecked(false); } //--------------------------------------------------------------------------------------------------------------------- void MainWindow::ExportDetailsAs() { const QHash *allDetails = pattern->DataPieces(); QHash::const_iterator i = allDetails->constBegin(); QHash detailsInLayout; while (i != allDetails->constEnd()) { if (i.value().IsInLayout()) { detailsInLayout.insert(i.key(), i.value()); } ++i; } if (detailsInLayout.count() == 0) { QMessageBox::information(this, tr("Layout mode"), tr("You don't have enough details to export. Please, " "include at least one detail in layout."), QMessageBox::Ok, QMessageBox::Ok); return; } QVector listDetails; try { listDetails = PrepareDetailsForLayout(detailsInLayout); } catch (VException &e) { QMessageBox::warning(this, tr("Export details"), tr("Can't export details.") + QLatin1String(" \n") + e.ErrorMessage(), QMessageBox::Ok, QMessageBox::Ok); return; } try { m_dialogSaveLayout = QSharedPointer(new DialogSaveLayout(1, Draw::Modeling, FileName(), this)); if (m_dialogSaveLayout->exec() == QDialog::Rejected) { m_dialogSaveLayout.clear(); ui->toolButtonDetailExportAs->setChecked(false); return; } ExportData(listDetails); m_dialogSaveLayout.clear(); } catch (const VException &e) { m_dialogSaveLayout.clear(); ui->toolButtonDetailExportAs->setChecked(false); qCritical("%s\n\n%s\n\n%s", qUtf8Printable(tr("Export error.")), qUtf8Printable(e.ErrorMessage()), qUtf8Printable(e.DetailedInformation())); return; } ui->toolButtonDetailExportAs->setChecked(false); } //--------------------------------------------------------------------------------------------------------------------- void MainWindow::ReopenFilesAfterCrash(QStringList &args) { const QStringList files = GetUnlokedRestoreFileList(); if (files.size() > 0) { qCDebug(vMainWindow, "Reopen files after crash."); QStringList restoreFiles; for (int i = 0; i < files.size(); ++i) { QFile file(files.at(i) + *autosavePrefix); if (file.exists()) { restoreFiles.append(files.at(i)); } } if (restoreFiles.size() > 0) { QMessageBox::StandardButton reply; const QString mes = tr("Valentina didn't shut down correctly. Do you want reopen files (%1) you had open?") .arg(restoreFiles.size()); reply = QMessageBox::question(this, tr("Reopen files."), mes, QMessageBox::Yes|QMessageBox::No, QMessageBox::Yes); if (reply == QMessageBox::Yes) { qCDebug(vMainWindow, "User said Yes."); for (auto &file : restoreFiles) { QString error; if (VDomDocument::SafeCopy(file + *autosavePrefix, file, error)) { QFile autoFile(file + *autosavePrefix); autoFile.remove(); LoadPattern(file); args.removeAll(file);// Do not open file twice after we restore him. } else { qCDebug(vMainWindow, "Could not copy %s%s to %s %s", qUtf8Printable(file), qUtf8Printable(*autosavePrefix), qUtf8Printable(file), qUtf8Printable(error)); } } } } } } //--------------------------------------------------------------------------------------------------------------------- QString MainWindow::CheckPathToMeasurements(const QString &patternPath, const QString &path) { if (path.isEmpty()) { return path; } auto FindLocation = [this](const QString &filter, const QString &dirPath, const QString &selectedName) { VCommonSettings::PrepareMultisizeTables(VCommonSettings::GetDefPathMultisizeMeasurements()); bool usedNotExistedDir = false; QDir directory(dirPath); if (not directory.exists()) { usedNotExistedDir = directory.mkpath(QChar('.')); } QString mPath; QFileDialog dialog(this, tr("Open file"), dirPath, filter); dialog.selectFile(selectedName); dialog.setFileMode(QFileDialog::ExistingFile); if (dialog.exec() == QDialog::Accepted) { mPath = dialog.selectedFiles().value(0); } if (usedNotExistedDir) { QDir directory(dirPath); directory.rmpath(QChar('.')); } return mPath; }; QFileInfo table(path); if (table.exists() == false) { if (!VApplication::IsGUIMode()) { return QString();// console mode doesn't support fixing path to a measurement file } else { const QString text = tr("The measurements file

%1

could not be found. Do you " "want to update the file location?").arg(path); QMessageBox::StandardButton res = QMessageBox::question(this, tr("Loading measurements file"), text, QMessageBox::Yes | QMessageBox::No, QMessageBox::Yes); if (res == QMessageBox::No) { return QString(); } else { MeasurementsType patternType; if (table.suffix() == QLatin1String("vst")) { patternType = MeasurementsType::Multisize; } else { patternType = MeasurementsType::Individual; // or Unknown } auto DirPath = [patternPath, table](const QString &defPath, QString &selectedName) { QString dirPath; const QDir patternDir = QFileInfo(patternPath).absoluteDir(); if (patternDir.exists(table.fileName())) { selectedName = table.fileName(); dirPath = patternDir.absolutePath(); } else { dirPath = defPath; } return dirPath; }; QString mPath; if (patternType == MeasurementsType::Multisize) { const QString filter = tr("Multisize measurements") + QLatin1String(" (*.vst);;") + tr("Individual measurements") + QLatin1String(" (*.vit)"); //Use standard path to multisize measurements QString selectedName; const QString dirPath = DirPath(qApp->ValentinaSettings()->GetPathMultisizeMeasurements(), selectedName); mPath = FindLocation(filter, dirPath, selectedName); } else { const QString filter = tr("Individual measurements") + QLatin1String(" (*.vit);;") + tr("Multisize measurements") + QLatin1String(" (*.vst)"); //Use standard path to individual measurements QString selectedName; const QString dirPath = DirPath(qApp->ValentinaSettings()->GetPathIndividualMeasurements(), selectedName); mPath = FindLocation(filter, dirPath, selectedName); } if (mPath.isEmpty()) { return mPath; } else { QScopedPointer m(new VMeasurements(pattern)); m->SetSize(VContainer::rsize()); m->SetHeight(VContainer::rheight()); m->setXMLContent(mPath); patternType = m->Type(); if (patternType == MeasurementsType::Unknown) { VException e(tr("Measurement file has unknown format.")); throw e; } if (patternType == MeasurementsType::Multisize) { VVSTConverter converter(mPath); m->setXMLContent(converter.Convert());// Read again after conversion } else { VVITConverter converter(mPath); m->setXMLContent(converter.Convert());// Read again after conversion } if (not m->IsDefinedKnownNamesValid()) { VException e(tr("Measurement file contains invalid known measurement(s).")); throw e; } CheckRequiredMeasurements(m.data()); qApp->setPatternType(patternType); doc->SetMPath(RelativeMPath(patternPath, mPath)); PatternChangesWereSaved(false); return mPath; } } } } return path; } //--------------------------------------------------------------------------------------------------------------------- void MainWindow::ChangePP(int index, bool zoomBestFit) { if (index != -1) { doc->ChangeActivPP(comboBoxDraws->itemText(index)); doc->setCurrentData(); emit RefreshHistory(); if (drawMode) { ArrowTool(true); if (zoomBestFit) { ZoomFitBestCurrent(); } } toolOptions->itemClicked(nullptr);//hide options for tool in previous pattern piece groupsWidget->UpdateGroups(); } } //--------------------------------------------------------------------------------------------------------------------- void MainWindow::EndVisualization(bool click) { if (not dialogTool.isNull()) { dialogTool->ShowDialog(click); } } //--------------------------------------------------------------------------------------------------------------------- /** * @brief ZoomFirstShow Fit scene size to best size for first show */ void MainWindow::ZoomFirstShow() { /* If don't call ZoomFitBest() twice, after first scaling or moving pattern piece, scene change coordinate and whole * pattern will be moved. Looks very ugly. It is best solution that i have now. */ if (pattern->DataPieces()->size() > 0) { ActionDetails(true); ui->view->ZoomFitBest(); } if (not ui->actionDraw->isChecked()) { ActionDraw(true); } ZoomFitBestCurrent(); VMainGraphicsView::NewSceneRect(sceneDraw, ui->view); VMainGraphicsView::NewSceneRect(sceneDetails, ui->view); if (pattern->DataPieces()->size() > 0) { ActionDetails(true); ui->view->ZoomFitBest(); } if (not ui->actionDraw->isChecked()) { ActionDraw(true); } } //--------------------------------------------------------------------------------------------------------------------- bool MainWindow::DoExport(const VCommandLinePtr &expParams) { QHash details; if(not qApp->getOpeningPattern()) { const QHash *allDetails = pattern->DataPieces(); if (allDetails->count() == 0) { qCCritical(vMainWindow, "%s", qUtf8Printable(tr("You can't export empty scene."))); qApp->exit(V_EX_DATAERR); return false; } else { const QString nameRegex = expParams->OptExportSuchDetails(); if (nameRegex.isEmpty()) { QHash::const_iterator i = allDetails->constBegin(); while (i != allDetails->constEnd()) { if (i.value().IsInLayout()) { details.insert(i.key(), i.value()); } ++i; } } else { const QRegularExpression nameRe(nameRegex); QHash::const_iterator i = allDetails->constBegin(); while (i != allDetails->constEnd()) { if (nameRe.match(i.value().GetName()).hasMatch()) { details.insert(i.key(), i.value()); } ++i; } } if (details.count() == 0) { qCCritical(vMainWindow, "%s", qUtf8Printable(tr("You can't export empty scene. Please, " "include at least one detail in layout."))); qApp->exit(V_EX_DATAERR); return false; } } } listDetails = PrepareDetailsForLayout(details); const bool exportOnlyDetails = expParams->IsExportOnlyDetails(); if (exportOnlyDetails) { try { m_dialogSaveLayout = QSharedPointer(new DialogSaveLayout(1, Draw::Modeling, expParams->OptBaseName(), this)); m_dialogSaveLayout->SetDestinationPath(expParams->OptDestinationPath()); m_dialogSaveLayout->SelectFormat(static_cast(expParams->OptExportType())); m_dialogSaveLayout->SetBinaryDXFFormat(expParams->IsBinaryDXF()); m_dialogSaveLayout->SetTextAsPaths(expParams->IsTextAsPaths()); if (static_cast(expParams->OptExportType()) == LayoutExportFormats::PDFTiled) { m_dialogSaveLayout->SetTiledExportMode(true); m_dialogSaveLayout->SetTiledMargins(expParams->TiledPageMargins()); m_dialogSaveLayout->SetTiledPageFormat(expParams->OptTiledPaperSize()); m_dialogSaveLayout->SetTiledPageOrientation(expParams->OptTiledPageOrientation()); } ExportData(listDetails); m_dialogSaveLayout.clear(); } catch (const VException &e) { m_dialogSaveLayout.clear(); qCCritical(vMainWindow, "%s\n\n%s", qUtf8Printable(tr("Export error.")), qUtf8Printable(e.ErrorMessage())); qApp->exit(V_EX_DATAERR); return false; } } else { auto settings = expParams->DefaultGenerator(); settings->SetTestAsPaths(expParams->IsTextAsPaths()); if (LayoutSettings(*settings.get())) { try { m_dialogSaveLayout = QSharedPointer(new DialogSaveLayout(scenes.size(), Draw::Layout, expParams->OptBaseName(), this)); m_dialogSaveLayout->SetDestinationPath(expParams->OptDestinationPath()); m_dialogSaveLayout->SelectFormat(static_cast(expParams->OptExportType())); m_dialogSaveLayout->SetBinaryDXFFormat(expParams->IsBinaryDXF()); if (static_cast(expParams->OptExportType()) == LayoutExportFormats::PDFTiled) { m_dialogSaveLayout->SetTiledExportMode(true); m_dialogSaveLayout->SetTiledMargins(expParams->TiledPageMargins()); m_dialogSaveLayout->SetTiledPageFormat(expParams->OptTiledPaperSize()); m_dialogSaveLayout->SetTiledPageOrientation(expParams->OptTiledPageOrientation()); } ExportData(listDetails); m_dialogSaveLayout.clear(); } catch (const VException &e) { m_dialogSaveLayout.clear(); qCCritical(vMainWindow, "%s\n\n%s", qUtf8Printable(tr("Export error.")), qUtf8Printable(e.ErrorMessage())); qApp->exit(V_EX_DATAERR); return false; } } else { qApp->exit(V_EX_DATAERR); return false; } } return true; } //--------------------------------------------------------------------------------------------------------------------- /** * @brief DoFMExport process export final measurements * @param expParams command line options * @return true if succesfull */ bool MainWindow::DoFMExport(const VCommandLinePtr &expParams) { QString filePath = expParams->OptExportFMTo(); if (filePath.isEmpty()) { qCCritical(vMainWindow, "%s\n\n%s", qUtf8Printable(tr("Export final measurements error.")), qUtf8Printable(tr("Destination path is empty."))); qApp->exit(V_EX_DATAERR); return false; } QFileInfo info(filePath); if (info.isRelative()) { filePath = QDir::currentPath() + QLatin1Char('/') + filePath; } const QString codecName = expParams->OptCSVCodecName(); int mib = QTextCodec::codecForLocale()->mibEnum(); if (not codecName.isEmpty()) { if (QTextCodec *codec = QTextCodec::codecForName(codecName.toLatin1())) { mib = codec->mibEnum(); } } QChar separator = expParams->OptCSVSeparator(); if (separator.isNull()) { separator = VCommonSettings::GetDefCSVSeparator(); } return ExportFMeasurementsToCSVData(filePath, expParams->IsCSVWithHeader(), mib, separator); } //--------------------------------------------------------------------------------------------------------------------- bool MainWindow::SetSize(const QString &text) { if (not VApplication::IsGUIMode()) { if (this->isWindowModified() || not qApp->GetPPath().isEmpty()) { if (qApp->patternType() == MeasurementsType::Multisize) { const int size = static_cast(UnitConvertor(text.toInt(), Unit::Cm, *pattern->GetPatternUnit())); const qint32 index = gradationSizes->findText(QString().setNum(size)); if (index != -1) { gradationSizes->setCurrentIndex(index); } else { qCCritical(vMainWindow, "%s", qUtf8Printable(tr("Not supported size value '%1' for this pattern file.").arg(text))); return false; } } else { qCCritical(vMainWindow, "%s", qUtf8Printable(tr("Couldn't set size. Need a file with multisize measurements."))); return false; } } else { qCCritical(vMainWindow, "%s", qUtf8Printable(tr("Couldn't set size. File wasn't opened."))); return false; } } else { qCWarning(vMainWindow, "%s", qUtf8Printable(tr("The method %1 does nothing in GUI mode").arg(Q_FUNC_INFO))); return false; } return true; } //--------------------------------------------------------------------------------------------------------------------- bool MainWindow::SetHeight(const QString &text) { if (not VApplication::IsGUIMode()) { if (this->isWindowModified() || not qApp->GetPPath().isEmpty()) { if (qApp->patternType() == MeasurementsType::Multisize) { const int height = static_cast(UnitConvertor(text.toInt(), Unit::Cm, *pattern->GetPatternUnit())); const qint32 index = gradationHeights->findText(QString().setNum(height)); if (index != -1) { gradationHeights->setCurrentIndex(index); } else { qCCritical(vMainWindow, "%s", qUtf8Printable(tr("Not supported height value '%1' for this pattern file.").arg(text))); return false; } } else { qCCritical(vMainWindow, "%s", qUtf8Printable(tr("Couldn't set height. Need a file with multisize measurements."))); return false; } } else { qCCritical(vMainWindow, "%s", qUtf8Printable(tr("Couldn't set height. File wasn't opened."))); return false; } } else { qCWarning(vMainWindow, "%s", qUtf8Printable(tr("The method %1 does nothing in GUI mode").arg(Q_FUNC_INFO))); return false; } return true; } //--------------------------------------------------------------------------------------------------------------------- void MainWindow::ProcessCMD() { const VCommandLinePtr cmd = qApp->CommandLine(); auto args = cmd->OptInputFileNames(); isNoScaling = cmd->IsNoScalingEnabled(); if (VApplication::IsGUIMode()) { ReopenFilesAfterCrash(args); for (auto &arg : args) { LoadPattern(arg); } } else { if (args.size() != 1) { qCritical() << tr("Please, provide one input file."); qApp->exit(V_EX_NOINPUT); return; } const bool loaded = LoadPattern(args.first(), cmd->OptMeasurePath()); if (not loaded) { return; // process only one input file } bool hSetted = true; bool sSetted = true; if (cmd->IsSetGradationSize()) { sSetted = SetSize(cmd->OptGradationSize()); } if (cmd->IsSetGradationHeight()) { hSetted = SetHeight(cmd->OptGradationHeight()); } if (not (hSetted && sSetted)) { qApp->exit(V_EX_DATAERR); return; } if (not cmd->IsTestModeEnabled()) { if (cmd->IsExportEnabled() && not DoExport(cmd)) { return; } if (cmd->IsExportFMEnabled() && not DoFMExport(cmd)) { return; } } qApp->exit(V_EX_OK);// close program after processing in console mode } } //--------------------------------------------------------------------------------------------------------------------- QString MainWindow::GetPatternFileName() { QString shownName = tr("untitled.val"); if(not qApp->GetPPath().isEmpty()) { shownName = StrippedName(qApp->GetPPath()); } shownName += QLatin1String("[*]"); return shownName; } //--------------------------------------------------------------------------------------------------------------------- QString MainWindow::GetMeasurementFileName() { if(doc->MPath().isEmpty()) { return QString(); } else { QString shownName(" ["); shownName += StrippedName(AbsoluteMPath(qApp->GetPPath(), doc->MPath())); if(mChanges) { shownName += QLatin1String("*"); } shownName += QLatin1String("]"); return shownName; } } //--------------------------------------------------------------------------------------------------------------------- void MainWindow::UpdateWindowTitle() { bool isFileWritable = true; if (not qApp->GetPPath().isEmpty()) { #ifdef Q_OS_WIN32 qt_ntfs_permission_lookup++; // turn checking on #endif /*Q_OS_WIN32*/ isFileWritable = QFileInfo(qApp->GetPPath()).isWritable(); #ifdef Q_OS_WIN32 qt_ntfs_permission_lookup--; // turn it off again #endif /*Q_OS_WIN32*/ } if (not patternReadOnly && isFileWritable) { setWindowTitle(GetPatternFileName()+GetMeasurementFileName()); } else { setWindowTitle(GetPatternFileName()+GetMeasurementFileName() + QLatin1String(" (") + tr("read only") + QLatin1String(")")); } setWindowFilePath(qApp->GetPPath()); #if defined(Q_OS_MAC) static QIcon fileIcon = QIcon(QCoreApplication::applicationDirPath() + QLatin1String("/../Resources/Valentina.icns")); QIcon icon; if (not qApp->GetPPath().isEmpty()) { if (not isWindowModified()) { icon = fileIcon; } else { static QIcon darkIcon; if (darkIcon.isNull()) { darkIcon = QIcon(darkenPixmap(fileIcon.pixmap(16, 16))); } icon = darkIcon; } } setWindowIcon(icon); #endif //defined(Q_OS_MAC) } //--------------------------------------------------------------------------------------------------------------------- bool MainWindow::IgnoreLocking(int error, const QString &path) { QMessageBox::StandardButton answer = QMessageBox::Abort; if (VApplication::IsGUIMode()) { switch(error) { case QLockFile::LockFailedError: answer = QMessageBox::warning(this, tr("Locking file"), tr("This file already opened in another window. Ignore if you want " "to continue (not recommended, can cause a data corruption)."), QMessageBox::Abort|QMessageBox::Ignore, QMessageBox::Abort); break; case QLockFile::PermissionError: answer = QMessageBox::question(this, tr("Locking file"), tr("The lock file could not be created, for lack of permissions. " "Ignore if you want to continue (not recommended, can cause " "a data corruption)."), QMessageBox::Abort|QMessageBox::Ignore, QMessageBox::Abort); break; case QLockFile::UnknownError: answer = QMessageBox::question(this, tr("Locking file"), tr("Unknown error happened, for instance a full partition prevented " "writing out the lock file. Ignore if you want to continue (not " "recommended, can cause a data corruption)."), QMessageBox::Abort|QMessageBox::Ignore, QMessageBox::Abort); break; default: answer = QMessageBox::Abort; break; } } if (answer == QMessageBox::Abort) { qCDebug(vMainWindow, "Failed to lock %s", qUtf8Printable(path)); qCDebug(vMainWindow, "Error type: %d", error); Clear(); if (not VApplication::IsGUIMode()) { switch(error) { case QLockFile::LockFailedError: qCCritical(vMainWindow, "%s", qUtf8Printable(tr("This file already opened in another window."))); break; case QLockFile::PermissionError: qCCritical(vMainWindow, "%s", qUtf8Printable(tr("The lock file could not be created, for lack of permissions."))); break; case QLockFile::UnknownError: qCCritical(vMainWindow, "%s", qUtf8Printable(tr("Unknown error happened, for instance a full partition prevented " "writing out the lock file."))); break; default: break; } qApp->exit(V_EX_NOINPUT); } return false; } return true; } //--------------------------------------------------------------------------------------------------------------------- void MainWindow::ToolSelectPoint() { // Only true for rubber band selection emit EnableLabelSelection(false); emit EnablePointSelection(false); emit EnableLineSelection(false); emit EnableArcSelection(false); emit EnableElArcSelection(false); emit EnableSplineSelection(false); emit EnableSplinePathSelection(false); // Hovering emit EnableLabelHover(true); emit EnablePointHover(true); emit EnableLineHover(false); emit EnableArcHover(false); emit EnableElArcHover(false); emit EnableSplineHover(false); emit EnableSplinePathHover(false); ui->view->AllowRubberBand(false); } //--------------------------------------------------------------------------------------------------------------------- void MainWindow::ToolSelectPointByRelease() { ToolSelectPoint(); emit ItemsSelection(SelectionType::ByMouseRelease); } //--------------------------------------------------------------------------------------------------------------------- void MainWindow::ToolSelectPointByPress() { ToolSelectPoint(); emit ItemsSelection(SelectionType::ByMousePress); } //--------------------------------------------------------------------------------------------------------------------- void MainWindow::ToolSelectSpline() { // Only true for rubber band selection emit EnableLabelSelection(false); emit EnablePointSelection(false); emit EnableLineSelection(false); emit EnableArcSelection(false); emit EnableElArcSelection(false); emit EnableSplineSelection(false); emit EnableSplinePathSelection(false); // Hovering emit EnableLabelHover(false); emit EnablePointHover(false); emit EnableLineHover(false); emit EnableArcHover(false); emit EnableElArcHover(false); emit EnableSplineHover(true); emit EnableSplinePathHover(false); emit ItemsSelection(SelectionType::ByMouseRelease); ui->view->AllowRubberBand(false); } //--------------------------------------------------------------------------------------------------------------------- void MainWindow::ToolSelectSplinePath() { // Only true for rubber band selection emit EnableLabelSelection(false); emit EnablePointSelection(false); emit EnableLineSelection(false); emit EnableArcSelection(false); emit EnableElArcSelection(false); emit EnableSplineSelection(false); emit EnableSplinePathSelection(false); // Hovering emit EnableLabelHover(false); emit EnablePointHover(false); emit EnableLineHover(false); emit EnableArcHover(false); emit EnableElArcHover(false); emit EnableSplineHover(false); emit EnableSplinePathHover(true); emit ItemsSelection(SelectionType::ByMouseRelease); ui->view->AllowRubberBand(false); } //--------------------------------------------------------------------------------------------------------------------- void MainWindow::ToolSelectArc() { // Only true for rubber band selection emit EnableLabelSelection(false); emit EnablePointSelection(false); emit EnableLineSelection(false); emit EnableArcSelection(false); emit EnableElArcSelection(false); emit EnableSplineSelection(false); emit EnableSplinePathSelection(false); // Hovering emit EnableLabelHover(false); emit EnablePointHover(false); emit EnableLineHover(false); emit EnableArcHover(true); emit EnableElArcHover(false); emit EnableSplineHover(false); emit EnableSplinePathHover(false); emit ItemsSelection(SelectionType::ByMouseRelease); ui->view->AllowRubberBand(false); } //--------------------------------------------------------------------------------------------------------------------- void MainWindow::ToolSelectPointArc() { // Only true for rubber band selection emit EnableLabelSelection(false); emit EnablePointSelection(false); emit EnableLineSelection(false); emit EnableArcSelection(false); emit EnableElArcSelection(false); emit EnableSplineSelection(false); emit EnableSplinePathSelection(false); // Hovering emit EnableLabelHover(true); emit EnablePointHover(true); emit EnableLineHover(false); emit EnableArcHover(true); emit EnableElArcHover(false); emit EnableSplineHover(false); emit EnableSplinePathHover(false); emit ItemsSelection(SelectionType::ByMouseRelease); ui->view->AllowRubberBand(false); } //--------------------------------------------------------------------------------------------------------------------- void MainWindow::ToolSelectCurve() { // Only true for rubber band selection emit EnableLabelSelection(false); emit EnablePointSelection(false); emit EnableLineSelection(false); emit EnableArcSelection(false); emit EnableElArcSelection(false); emit EnableSplineSelection(false); emit EnableSplinePathSelection(false); // Hovering emit EnableLabelHover(false); emit EnablePointHover(false); emit EnableLineHover(false); emit EnableArcHover(true); emit EnableElArcHover(true); emit EnableSplineHover(true); emit EnableSplinePathHover(true); emit ItemsSelection(SelectionType::ByMouseRelease); ui->view->AllowRubberBand(false); } //--------------------------------------------------------------------------------------------------------------------- void MainWindow::ToolSelectAllDrawObjects() { // Only true for rubber band selection emit EnableLabelSelection(false); emit EnablePointSelection(false); emit EnableLineSelection(false); emit EnableArcSelection(false); emit EnableElArcSelection(false); emit EnableSplineSelection(false); emit EnableSplinePathSelection(false); // Hovering emit EnableLabelHover(true); emit EnablePointHover(true); emit EnableLineHover(false); emit EnableArcHover(true); emit EnableElArcHover(true); emit EnableSplineHover(true); emit EnableSplinePathHover(true); emit ItemsSelection(SelectionType::ByMouseRelease); ui->view->AllowRubberBand(false); } //--------------------------------------------------------------------------------------------------------------------- void MainWindow::ToolSelectOperationObjects() { // Only true for rubber band selection emit EnableLabelSelection(true); emit EnablePointSelection(true); emit EnableLineSelection(false); emit EnableArcSelection(true); emit EnableElArcSelection(true); emit EnableSplineSelection(true); emit EnableSplinePathSelection(true); // Hovering emit EnableLabelHover(true); emit EnablePointHover(true); emit EnableLineHover(false); emit EnableArcHover(true); emit EnableElArcHover(true); emit EnableSplineHover(true); emit EnableSplinePathHover(true); emit ItemsSelection(SelectionType::ByMouseRelease); ui->view->AllowRubberBand(true); } //--------------------------------------------------------------------------------------------------------------------- void MainWindow::ToolSelectGroupObjects() { ToolSelectOperationObjects(); // Only true for rubber band selection emit EnableLineSelection(true); // Hovering emit EnableLineHover(true); } //--------------------------------------------------------------------------------------------------------------------- void MainWindow::ToolSelectDetail() { // Only true for rubber band selection emit EnableNodeLabelSelection(false); emit EnableNodePointSelection(false); emit EnableDetailSelection(true);// Disable when done visualization details // Hovering emit EnableNodeLabelHover(true); emit EnableNodePointHover(true); emit EnableDetailHover(true); emit ItemsSelection(SelectionType::ByMouseRelease); ui->view->AllowRubberBand(false); }