/************************************************************************ ** ** @file tmainwindow.cpp ** @author Roman Telezhynskyi ** @date 10 7, 2015 ** ** @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) 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 "tmainwindow.h" #include "../ifc/ifcdef.h" #include "../ifc/xml/vpatternconverter.h" #include "../ifc/xml/vvitconverter.h" #include "../ifc/xml/vvstconverter.h" #include "../qmuparser/qmudef.h" #include "../vganalytics/vganalytics.h" #include "../vmisc/compatibility.h" #include "../vmisc/dialogs/dialogaskcollectstatistic.h" #include "../vmisc/dialogs/dialogexporttocsv.h" #include "../vmisc/dialogs/dialogselectlanguage.h" #include "../vmisc/qxtcsvmodel.h" #include "../vmisc/theme/vtheme.h" #include "../vmisc/vsysexits.h" #include "../vpatterndb/calculator.h" #include "../vpatterndb/measurements.h" #include "../vpatterndb/pmsystems.h" #include "../vpatterndb/variables/vmeasurement.h" #include "../vpatterndb/vcontainer.h" #include "../vtools/dialogs/support/dialogeditwrongformula.h" #include "def.h" #include "dialogs/dialogabouttape.h" #include "dialogs/dialogdimensioncustomnames.h" #include "dialogs/dialogdimensionlabels.h" #include "dialogs/dialogmdatabase.h" #include "dialogs/dialogmeasurementscsvcolumns.h" #include "dialogs/dialognewmeasurements.h" #include "dialogs/dialogrestrictdimension.h" #include "dialogs/dialogsetupmultisize.h" #include "dialogs/dialogtapepreferences.h" #include "mapplication.h" // Should be last because of definning qApp #include "ui_tmainwindow.h" #include "vlitepattern.h" #include "vtapesettings.h" #if QT_VERSION < QT_VERSION_CHECK(5, 7, 0) #include "../vmisc/backport/qoverload.h" #endif // QT_VERSION < QT_VERSION_CHECK(5, 7, 0) #if QT_VERSION >= QT_VERSION_CHECK(6, 0, 0) #include "../vmisc/vtextcodec.h" #else #include #endif #if QT_VERSION < QT_VERSION_CHECK(5, 12, 0) #include "../vmisc/backport/qscopeguard.h" #else #include #endif #include #include #include #include #include #include #include #include #if (defined(Q_CC_GNU) && Q_CC_GNU < 409) && !defined(Q_CC_CLANG) // DO NOT WORK WITH GCC 4.8 #else #if __cplusplus >= 201402L using namespace std::chrono_literals; #else #include "../vmisc/bpstd/chrono.hpp" using namespace bpstd::literals::chrono_literals; #endif // __cplusplus >= 201402L #endif //(defined(Q_CC_GNU) && Q_CC_GNU < 409) && !defined(Q_CC_CLANG) #if defined(Q_OS_MAC) #include #include #endif // defined(Q_OS_MAC) constexpr int DIALOG_MAX_FORMULA_HEIGHT = 64; QT_WARNING_PUSH QT_WARNING_DISABLE_CLANG("-Wmissing-prototypes") QT_WARNING_DISABLE_INTEL(1418) Q_LOGGING_CATEGORY(tMainWindow, "t.mainwindow") // NOLINT QT_WARNING_POP namespace { enum class MUnits : qint8 { Table, Degrees }; struct IndividualMeasurement { IndividualMeasurement() = default; QString name{}; // NOLINT(misc-non-private-member-variables-in-classes) QString value{'0'}; // NOLINT(misc-non-private-member-variables-in-classes) QString fullName{}; // NOLINT(misc-non-private-member-variables-in-classes) QString description{}; // NOLINT(misc-non-private-member-variables-in-classes) }; //--------------------------------------------------------------------------------------------------------------------- auto SaveDirPath(const QString &curFile, MeasurementsType mType) -> QString { QString dir; if (curFile.isEmpty()) { if (mType == MeasurementsType::Individual) { dir = MApplication::VApp()->TapeSettings()->GetPathIndividualMeasurements(); } else { dir = MApplication::VApp()->TapeSettings()->GetPathMultisizeMeasurements(); dir = VCommonSettings::PrepareMultisizeTables(dir); } } else { dir = QFileInfo(curFile).absolutePath(); } return dir; } //--------------------------------------------------------------------------------------------------------------------- void InitDimensionXItems(const QVector &bases, const DimesionLabels &labels, QComboBox *control, const QString &unit) { SCASSERT(control != nullptr) for (auto base : bases) { if (VFuzzyContains(labels, base) && not VFuzzyValue(labels, base).isEmpty()) { control->addItem(VFuzzyValue(labels, base), base); } else { control->addItem(QStringLiteral("%1 %2").arg(base).arg(unit), base); } } } //--------------------------------------------------------------------------------------------------------------------- void InitDimensionYWZItems(const QVector &bases, const DimesionLabels &labels, QComboBox *control, const QString &unit, bool bodyMeasurement, bool fc) { for (auto base : bases) { if (VFuzzyContains(labels, base) && not VFuzzyValue(labels, base).isEmpty()) { control->addItem(VFuzzyValue(labels, base), base); } else { if (bodyMeasurement) { control->addItem(QStringLiteral("%1 %2").arg(fc ? base * 2 : base).arg(unit), base); } else { control->addItem(QString::number(base), base); } } } } //--------------------------------------------------------------------------------------------------------------------- auto ConverToDouble(QString text, const QString &error) -> qreal { text.replace(QStringLiteral(" "), QString()); text = VTranslateVars::TryFormulaFromUser(text, VAbstractApplication::VApp()->Settings()->GetOsSeparator()); bool ok = false; const qreal value = QLocale::c().toDouble(text, &ok); if (not ok) { throw VException(error); } return value; } //--------------------------------------------------------------------------------------------------------------------- void SetIndividualMeasurementFullName(int i, const QString &name, const QxtCsvModel &csv, const QVector &map, IndividualMeasurement &measurement) { const int columns = csv.columnCount(); const bool custom = name.startsWith(CustomMSign); if (columns > 2 && custom) { const int fullNameColumn = map.at(static_cast(IndividualMeasurementsColumns::FullName)); if (fullNameColumn >= 0) { measurement.fullName = csv.text(i, fullNameColumn).simplified(); } } } //--------------------------------------------------------------------------------------------------------------------- void SetIndividualMeasurementDescription(int i, const QString &name, const QxtCsvModel &csv, const QVector &map, IndividualMeasurement &measurement) { const int columns = csv.columnCount(); const bool custom = name.startsWith(CustomMSign); if (columns > 3 && custom) { const int descriptionColumn = map.at(static_cast(IndividualMeasurementsColumns::Description)); if (descriptionColumn >= 0) { measurement.description = csv.text(i, descriptionColumn).simplified(); } } } } // namespace // We need this enum in case we will add or delete a column. And also make code more readable. enum { ColumnName = 0, ColumnFullName = 1, ColumnCalcValue = 2, ColumnFormula = 3, ColumnBaseValue = 4, ColumnShiftA = 5, ColumnShiftB = 6, ColumnShiftC = 7, ColumnCorrection = 8 }; //--------------------------------------------------------------------------------------------------------------------- TMainWindow::TMainWindow(QWidget *parent) : VAbstractMainWindow(parent), ui(new Ui::TMainWindow), m_formulaBaseHeight(0), m_gradation(new QTimer(this)), m_searchHistory(new QMenu(this)) { ui->setupUi(this); InitIcons(); VAbstractApplication::VApp()->Settings()->GetOsSeparator() ? setLocale(QLocale()) : setLocale(QLocale::c()); ui->lineEditName->setClearButtonEnabled(true); ui->lineEditFullName->setClearButtonEnabled(true); ui->lineEditCustomerName->setClearButtonEnabled(true); ui->lineEditEmail->setClearButtonEnabled(true); ui->lineEditFind->installEventFilter(this); ui->plainTextEditFormula->installEventFilter(this); m_search = QSharedPointer(new VTableSearch(ui->tableWidget)); ui->tabWidget->setVisible(false); ui->mainToolBar->setContextMenuPolicy(Qt::PreventContextMenu); ui->toolBarGradation->setContextMenuPolicy(Qt::PreventContextMenu); m_recentFileActs.fill(nullptr); connect(m_gradation, &QTimer::timeout, this, &TMainWindow::GradationChanged); SetupMenu(); ReadSettings(); #if defined(Q_OS_MAC) // On Mac deafault icon size is 32x32. ui->toolBarGradation->setIconSize(QSize(24, 24)); ui->pushButtonShowInExplorer->setText(tr("Show in Finder")); // Mac OS Dock Menu QMenu *menu = new QMenu(this); connect(menu, &QMenu::aboutToShow, this, &TMainWindow::AboutToShowDockMenu); AboutToShowDockMenu(); menu->setAsDockMenu(); #endif // defined(Q_OS_MAC) if (MApplication::VApp()->IsAppInGUIMode()) { QTimer::singleShot(V_SECONDS(1), this, &TMainWindow::AskDefaultSettings); } } //--------------------------------------------------------------------------------------------------------------------- TMainWindow::~TMainWindow() { ui->lineEditFind->blockSignals(true); // prevents crash delete m_data; delete m_m; qDeleteAll(m_hackedWidgets); delete ui; } //--------------------------------------------------------------------------------------------------------------------- auto TMainWindow::CurrentFile() const -> QString { return m_curFile; } //--------------------------------------------------------------------------------------------------------------------- void TMainWindow::RetranslateTable() { if (m_m != nullptr) { const int row = ui->tableWidget->currentRow(); RefreshTable(); ui->tableWidget->selectRow(row); m_search->RefreshList(ui->lineEditFind->text()); } } //--------------------------------------------------------------------------------------------------------------------- void TMainWindow::SetDimensionABase(int base) { const QList dimensions = m_m->Dimensions().values(); if (dimensions.isEmpty()) { qCWarning(tMainWindow, "%s\n", qPrintable(tr("The table doesn't provide dimensions"))); return; } const qint32 i = m_gradationDimensionA->findData(base); if (i != -1) { m_gradationDimensionA->setCurrentIndex(i); } if (not VFuzzyComparePossibleNulls(base, m_currentDimensionA)) { qCWarning(tMainWindow, "%s\n", qPrintable(tr("Invalid base value for dimension A"))); return; } } //--------------------------------------------------------------------------------------------------------------------- void TMainWindow::SetDimensionBBase(int base) { const QList dimensions = m_m->Dimensions().values(); if (dimensions.size() <= 1) { qCWarning(tMainWindow, "%s\n", qPrintable(tr("The table doesn't support dimension B"))); return; } const qint32 i = m_gradationDimensionB->findData(base); if (i != -1) { m_gradationDimensionB->setCurrentIndex(i); } if (not VFuzzyComparePossibleNulls(base, m_currentDimensionB)) { qCWarning(tMainWindow, "%s\n", qPrintable(tr("Invalid base value for dimension B"))); return; } } //--------------------------------------------------------------------------------------------------------------------- void TMainWindow::SetDimensionCBase(int base) { const QList dimensions = m_m->Dimensions().values(); if (dimensions.size() <= 2) { qCWarning(tMainWindow, "%s\n", qPrintable(tr("The table doesn't support dimension C"))); return; } const qint32 i = m_gradationDimensionC->findData(base); if (i != -1) { m_gradationDimensionC->setCurrentIndex(i); } if (not VFuzzyComparePossibleNulls(base, m_currentDimensionC)) { qCWarning(tMainWindow, "%s\n", qPrintable(tr("Invalid base value for dimension C"))); return; } } //--------------------------------------------------------------------------------------------------------------------- void TMainWindow::SetPUnit(Unit unit) { m_pUnit = unit; SetCurrentPatternUnit(); UpdatePatternUnit(); } //--------------------------------------------------------------------------------------------------------------------- auto TMainWindow::LoadFile(const QString &path) -> bool { if (m_m != nullptr) { return MApplication::VApp()->NewMainWindow()->LoadFile(path); } if (not QFileInfo::exists(path)) { qCCritical(tMainWindow, "%s", qUtf8Printable(tr("File '%1' doesn't exist!").arg(path))); if (MApplication::VApp()->IsTestMode()) { QCoreApplication::exit(V_EX_NOINPUT); } return false; } // Check if file already opened const QList list = MApplication::VApp()->MainWindows(); auto w = std::find_if(list.begin(), list.end(), [path](TMainWindow *window) { return window->CurrentFile() == path; }); if (w != list.end()) { (*w)->activateWindow(); close(); return false; } VlpCreateLock(m_lock, path); if (not m_lock->IsLocked()) { if (not IgnoreLocking(m_lock->GetLockError(), path, MApplication::VApp()->IsAppInGUIMode())) { return false; } } try { m_data = new VContainer(VAbstractApplication::VApp()->TrVars(), &m_mUnit, VContainer::UniqueNamespace()); m_m = new VMeasurements(m_data); m_m->setXMLContent(path); m_mType = m_m->Type(); if (m_mType == MeasurementsType::Unknown) { throw VException(tr("File has unknown format.")); } if (m_mType == MeasurementsType::Multisize) { VVSTConverter converter(path); m_curFileFormatVersion = converter.GetCurrentFormatVersion(); m_curFileFormatVersionStr = converter.GetFormatVersionStr(); m_m->setXMLContent(converter.Convert()); // Read again after conversion } else { VVITConverter converter(path); m_curFileFormatVersion = converter.GetCurrentFormatVersion(); m_curFileFormatVersionStr = converter.GetFormatVersionStr(); m_m->setXMLContent(converter.Convert()); // Read again after conversion } if (not m_m->IsDefinedKnownNamesValid()) { throw VException(tr("File contains invalid known measurement(s).")); } m_mUnit = m_m->Units(); m_pUnit = m_mUnit; m_currentDimensionA = m_m->DimensionABase(); m_currentDimensionB = m_m->DimensionBBase(); m_currentDimensionC = m_m->DimensionCBase(); ui->labelToolTip->setVisible(false); ui->tabWidget->setVisible(true); m_mIsReadOnly = m_m->IsReadOnly(); UpdatePadlock(m_mIsReadOnly); SetCurrentFile(path); InitWindow(); const bool freshCall = true; RefreshData(freshCall); if (ui->tableWidget->rowCount() > 0) { ui->tableWidget->selectRow(0); } MeasurementGUI(); ui->actionImportFromCSV->setEnabled(true); } catch (VException &e) { qCCritical(tMainWindow, "%s\n\n%s\n\n%s", qUtf8Printable(tr("File error.")), qUtf8Printable(e.ErrorMessage()), qUtf8Printable(e.DetailedInformation())); ui->labelToolTip->setVisible(true); ui->tabWidget->setVisible(false); delete m_m; m_m = nullptr; delete m_data; m_data = nullptr; m_lock.reset(); if (MApplication::VApp()->IsTestMode()) { QCoreApplication::exit(V_EX_NOINPUT); } return false; } return true; } //--------------------------------------------------------------------------------------------------------------------- void TMainWindow::FileNew() { if (m_m == nullptr) { DialogNewMeasurements measurements(this); if (measurements.exec() == QDialog::Rejected) { return; } m_mUnit = measurements.MUnit(); m_pUnit = m_mUnit; m_mType = measurements.Type(); if (m_mType == MeasurementsType::Multisize) { DialogSetupMultisize setup(m_mUnit, this); if (setup.exec() == QDialog::Rejected) { m_mUnit = Unit::Cm; m_pUnit = m_mUnit; m_mType = MeasurementsType::Individual; return; } m_data = new VContainer(VAbstractApplication::VApp()->TrVars(), &m_mUnit, VContainer::UniqueNamespace()); m_m = new VMeasurements(m_mUnit, setup.Dimensions(), m_data); m_m->SetFullCircumference(setup.FullCircumference()); m_curFileFormatVersion = VVSTConverter::MeasurementMaxVer; m_curFileFormatVersionStr = VVSTConverter::MeasurementMaxVerStr; SetCurrentDimensionValues(); } else { m_data = new VContainer(VAbstractApplication::VApp()->TrVars(), &m_mUnit, VContainer::UniqueNamespace()); m_m = new VMeasurements(m_mUnit, m_data); m_curFileFormatVersion = VVITConverter::MeasurementMaxVer; m_curFileFormatVersionStr = VVITConverter::MeasurementMaxVerStr; } m_mIsReadOnly = m_m->IsReadOnly(); UpdatePadlock(m_mIsReadOnly); SetCurrentFile(QString()); MeasurementsWereSaved(false); InitWindow(); MeasurementGUI(); ui->actionImportFromCSV->setEnabled(true); } else { MApplication::VApp()->NewMainWindow()->FileNew(); } } //--------------------------------------------------------------------------------------------------------------------- void TMainWindow::OpenIndividual() { const QString filter = tr("Individual measurements") + QStringLiteral(" (*.vit);;") + tr("Multisize measurements") + QStringLiteral(" (*.vst);;") + tr("All files") + QStringLiteral(" (*.*)"); // Use standard path to individual measurements const QString pathTo = MApplication::VApp()->TapeSettings()->GetPathIndividualMeasurements(); bool usedNotExistedDir = false; QDir directory(pathTo); if (not directory.exists()) { usedNotExistedDir = directory.mkpath(QChar('.')); } Open(pathTo, filter); if (usedNotExistedDir) { QDir(pathTo).rmpath(QChar('.')); } } //--------------------------------------------------------------------------------------------------------------------- void TMainWindow::OpenMultisize() { const QString filter = tr("Multisize measurements") + QStringLiteral(" (*.vst);;") + tr("Individual measurements") + QStringLiteral(" (*.vit);;") + tr("All files") + QStringLiteral(" (*.*)"); // Use standard path to multisize measurements QString pathTo = MApplication::VApp()->TapeSettings()->GetPathMultisizeMeasurements(); pathTo = VCommonSettings::PrepareMultisizeTables(pathTo); Open(pathTo, filter); } //--------------------------------------------------------------------------------------------------------------------- void TMainWindow::OpenTemplate() { const QString filter = tr("Measurements") + QStringLiteral(" (*.vst *.vit);;") + tr("All files") + QStringLiteral(" (*.*)"); // Use standard path to template files QString pathTo = MApplication::VApp()->TapeSettings()->GetPathTemplate(); pathTo = VTapeSettings::PrepareStandardTemplates(pathTo); Open(pathTo, filter); if (m_m != nullptr) { // The file was opened. SetCurrentFile(QString()); // Force user to to save new file m_lock.reset(); // remove lock from template } } //--------------------------------------------------------------------------------------------------------------------- void TMainWindow::CreateFromExisting() { const QString filter = tr("Measurements") + QStringLiteral(" (*.vst *.vit);;") + tr("All files") + QStringLiteral(" (*.*)"); const QString mPath = QFileDialog::getOpenFileName(this, tr("Select file"), QDir::homePath(), filter, nullptr, VAbstractApplication::VApp()->NativeFileDialog()); if (not mPath.isEmpty()) { if (m_m == nullptr) { LoadFromExistingFile(mPath); } else { MApplication::VApp()->NewMainWindow()->CreateFromExisting(); } } } //--------------------------------------------------------------------------------------------------------------------- void TMainWindow::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)); auto *preferences = new DialogTapePreferences(this); // QScopedPointer needs to be sure any exception will never block guard QScopedPointer dlg(preferences); guard = preferences; // Must be first connect(dlg.data(), &DialogTapePreferences::UpdateProperties, this, &TMainWindow::WindowsLocale); connect(dlg.data(), &DialogTapePreferences::UpdateProperties, this, &TMainWindow::ToolBarStyles); QGuiApplication::restoreOverrideCursor(); dlg->exec(); } } //--------------------------------------------------------------------------------------------------------------------- void TMainWindow::ToolBarStyles() { ToolBarStyle(ui->toolBarGradation); ToolBarStyle(ui->mainToolBar); } //--------------------------------------------------------------------------------------------------------------------- void TMainWindow::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 if (MaybeSave()) { WriteSettings(); event->accept(); deleteLater(); } else { event->ignore(); } } //--------------------------------------------------------------------------------------------------------------------- void TMainWindow::changeEvent(QEvent *event) { if (event->type() == QEvent::LanguageChange) { WindowsLocale(); // retranslate designer form (single inheritance approach) ui->retranslateUi(this); ui->lineEditFind->setPlaceholderText(m_search->SearchPlaceholder()); UpdateSearchControlsTooltips(); UpdateWindowTitle(); InitMeasurementUnits(); if (m_mType == MeasurementsType::Multisize) { ui->labelMType->setText(tr("Multisize measurements")); InitDimensionsBaseValue(); InitDimensionControls(); InitDimesionShifts(); RetranslateTableHeaders(); } else { ui->labelMType->setText(tr("Individual measurements")); const qint32 index = ui->comboBoxGender->currentIndex(); ui->comboBoxGender->blockSignals(true); ui->comboBoxGender->clear(); InitGender(ui->comboBoxGender); ui->comboBoxGender->setCurrentIndex(index); ui->comboBoxGender->blockSignals(false); InitMeasurementDimension(); } { const qint32 index = ui->comboBoxPMSystem->currentIndex(); ui->comboBoxPMSystem->blockSignals(true); ui->comboBoxPMSystem->clear(); InitPMSystems(ui->comboBoxPMSystem); ui->comboBoxPMSystem->setCurrentIndex(index); ui->comboBoxPMSystem->blockSignals(false); } { if (m_labelPatternUnit) { m_labelPatternUnit->setText(tr("Pattern unit:")); } if (m_comboBoxUnits) { const qint32 index = m_comboBoxUnits->currentIndex(); m_comboBoxUnits->blockSignals(true); m_comboBoxUnits->clear(); InitComboBoxUnits(); m_comboBoxUnits->setCurrentIndex(index); m_comboBoxUnits->blockSignals(false); } } } if (event->type() == QEvent::PaletteChange) { InitIcons(); } // remember to call base class implementation QMainWindow::changeEvent(event); } //--------------------------------------------------------------------------------------------------------------------- auto TMainWindow::eventFilter(QObject *object, QEvent *event) -> bool { if (auto *plainTextEdit = qobject_cast(object)) { if (event->type() == QEvent::KeyPress) { auto *keyEvent = static_cast(event); // NOLINT(cppcoreguidelines-pro-type-static-cast-downcast) if ((keyEvent->key() == Qt::Key_Period) && ((keyEvent->modifiers() & Qt::KeypadModifier) != 0U)) { if (VAbstractApplication::VApp()->Settings()->GetOsSeparator()) { plainTextEdit->insertPlainText(LocaleDecimalPoint(QLocale())); } else { plainTextEdit->insertPlainText(LocaleDecimalPoint(QLocale::c())); } return true; } } } else if (auto *textEdit = qobject_cast(object)) { if (event->type() == QEvent::KeyPress) { auto *keyEvent = static_cast(event); // NOLINT(cppcoreguidelines-pro-type-static-cast-downcast) if ((keyEvent->key() == Qt::Key_Period) && ((keyEvent->modifiers() & Qt::KeypadModifier) != 0U)) { if (VAbstractApplication::VApp()->Settings()->GetOsSeparator()) { textEdit->insert(LocaleDecimalPoint(QLocale())); } else { textEdit->insert(LocaleDecimalPoint(QLocale::c())); } return true; } } } // pass the event on to the parent class return QMainWindow::eventFilter(object, event); } //--------------------------------------------------------------------------------------------------------------------- void TMainWindow::ExportToCSVData(const QString &fileName, bool withHeader, int mib, const QChar &separator) { QxtCsvModel csv; const int columns = ui->tableWidget->columnCount(); { int colCount = 0; for (int column = 0; column < columns; ++column) { if (not ui->tableWidget->isColumnHidden(column)) { csv.insertColumn(colCount++); } } } if (withHeader) { int colCount = 0; for (int column = 0; column < columns; ++column) { if (not ui->tableWidget->isColumnHidden(column)) { QString text; if (QTableWidgetItem *header = ui->tableWidget->horizontalHeaderItem(column)) { text = header->text(); } csv.setHeaderText(colCount, text); ++colCount; } } } const int rows = ui->tableWidget->rowCount(); for (int row = 0; row < rows; ++row) { csv.insertRow(row); int colCount = 0; for (int column = 0; column < columns; ++column) { if (not ui->tableWidget->isColumnHidden(column)) { QString text; if (QTableWidgetItem *item = ui->tableWidget->item(row, column)) { text = item->text(); } csv.setText(row, colCount, text); ++colCount; } } } QString error; csv.toCSV(fileName, error, withHeader, separator, VTextCodec::codecForMib(mib)); } //--------------------------------------------------------------------------------------------------------------------- auto TMainWindow::RecentFileList() const -> QStringList { return MApplication::VApp()->TapeSettings()->GetRecentFileList(); } //--------------------------------------------------------------------------------------------------------------------- auto TMainWindow::FileSave() -> bool { if (m_curFile.isEmpty() || m_mIsReadOnly) { return FileSaveAs(); } if (m_mType == MeasurementsType::Multisize && m_curFileFormatVersion < VVSTConverter::MeasurementMaxVer && not ContinueFormatRewrite(m_curFileFormatVersionStr, VVSTConverter::MeasurementMaxVerStr)) { return false; } if (m_mType == MeasurementsType::Individual && m_curFileFormatVersion < VVITConverter::MeasurementMaxVer && not ContinueFormatRewrite(m_curFileFormatVersionStr, VVITConverter::MeasurementMaxVerStr)) { return false; } if (not CheckFilePermissions(m_curFile, this)) { return false; } QString error; if (not SaveMeasurements(m_curFile, error)) { QMessageBox messageBox; 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 false; } if (m_mType == MeasurementsType::Multisize) { m_curFileFormatVersion = VVSTConverter::MeasurementMaxVer; m_curFileFormatVersionStr = VVSTConverter::MeasurementMaxVerStr; } else { m_curFileFormatVersion = VVITConverter::MeasurementMaxVer; m_curFileFormatVersionStr = VVITConverter::MeasurementMaxVerStr; } return true; } //--------------------------------------------------------------------------------------------------------------------- auto TMainWindow::FileSaveAs() -> bool { QString filters; QString fName = tr("measurements"); QString suffix; if (m_mType == MeasurementsType::Individual) { filters = tr("Individual measurements") + QStringLiteral(" (*.vit)"); suffix = QStringLiteral("vit"); } else { filters = tr("Multisize measurements") + QStringLiteral(" (*.vst)"); suffix = QStringLiteral("vst"); } fName += QChar('.') + suffix; const QString dir = SaveDirPath(m_curFile, m_mType); bool usedNotExistedDir = false; QDir directory(dir); if (not directory.exists()) { usedNotExistedDir = directory.mkpath(QChar('.')); } if (not m_curFile.isEmpty()) { fName = StrippedName(m_curFile); } QString fileName = QFileDialog::getSaveFileName(this, tr("Save as"), dir + QChar('/') + fName, filters, nullptr, VAbstractApplication::VApp()->NativeFileDialog()); auto RemoveTempDir = qScopeGuard( [usedNotExistedDir, dir]() { if (usedNotExistedDir) { QDir(dir).rmpath(QChar('.')); } }); if (fileName.isEmpty()) { return false; } QFileInfo f(fileName); if (f.suffix().isEmpty() && f.suffix() != suffix) { fileName += QChar('.') + suffix; } if (QFileInfo::exists(fileName) && m_curFile != fileName) { // Temporary try to lock the file before saving VLockGuard tmp(fileName); if (not tmp.IsLocked()) { qCCritical(tMainWindow, "%s", qUtf8Printable(tr("Failed to lock. This file already opened in another window."))); return false; } } // Need for restoring previous state in case of failure const bool readOnly = m_m->IsReadOnly(); m_m->SetReadOnly(false); m_mIsReadOnly = false; QString error; bool result = SaveMeasurements(fileName, error); if (not result) { QMessageBox messageBox; messageBox.setIcon(QMessageBox::Warning); messageBox.setInformativeText(tr("Could not save file")); messageBox.setDefaultButton(QMessageBox::Ok); messageBox.setDetailedText(error); messageBox.setStandardButtons(QMessageBox::Ok); messageBox.exec(); // Restore previous state m_m->SetReadOnly(readOnly); m_mIsReadOnly = readOnly; return false; } if (m_mType == MeasurementsType::Multisize) { m_curFileFormatVersion = VVSTConverter::MeasurementMaxVer; m_curFileFormatVersionStr = VVSTConverter::MeasurementMaxVerStr; } else { m_curFileFormatVersion = VVITConverter::MeasurementMaxVer; m_curFileFormatVersionStr = VVITConverter::MeasurementMaxVerStr; } UpdatePadlock(false); UpdateWindowTitle(); if (m_curFile == fileName && not m_lock.isNull()) { m_lock->Unlock(); } VlpCreateLock(m_lock, fileName); if (not m_lock->IsLocked()) { qCCritical(tMainWindow, "%s", qUtf8Printable(tr("Failed to lock. This file already opened in another window. " "Expect collissions when run 2 copies of the program."))); return false; } return true; } //--------------------------------------------------------------------------------------------------------------------- void TMainWindow::AboutToShowWindowMenu() { ui->menuWindow->clear(); CreateWindowMenu(ui->menuWindow); } //--------------------------------------------------------------------------------------------------------------------- void TMainWindow::ShowWindow() const { if (auto *action = qobject_cast(sender())) { const QVariant v = action->data(); if (v.canConvert()) { const int offset = qvariant_cast(v); const QList windows = MApplication::VApp()->MainWindows(); windows.at(offset)->raise(); windows.at(offset)->activateWindow(); } } } //--------------------------------------------------------------------------------------------------------------------- void TMainWindow::ImportDataFromCSV() { if (m_m == nullptr || m_m->Type() == MeasurementsType::Unknown) { return; } const QString filters = tr("Comma-Separated Values") + QStringLiteral(" (*.csv)"); const QString suffix = QStringLiteral("csv"); QString fileName = QFileDialog::getOpenFileName(this, tr("Import from CSV"), QDir::homePath(), filters, nullptr, VAbstractApplication::VApp()->NativeFileDialog()); if (fileName.isEmpty()) { return; } QFileInfo f(fileName); if (f.suffix().isEmpty() && f.suffix() != suffix) { fileName += QChar('.') + suffix; } DialogExportToCSV dialog(this); dialog.SetWithHeader(VAbstractApplication::VApp()->Settings()->GetCSVWithHeader()); dialog.SetSelectedMib(VAbstractApplication::VApp()->Settings()->GetCSVCodec()); dialog.SetSeparator(VAbstractApplication::VApp()->Settings()->GetCSVSeparator()); dialog.ShowFilePreview(fileName); if (dialog.exec() == QDialog::Accepted) { VAbstractApplication::VApp()->Settings()->SetCSVSeparator(dialog.GetSeparator()); VAbstractApplication::VApp()->Settings()->SetCSVCodec(dialog.GetSelectedMib()); VAbstractApplication::VApp()->Settings()->SetCSVWithHeader(dialog.IsWithHeader()); QSharedPointer columns; if (m_m->Type() == MeasurementsType::Multisize) { const QList dimensions = m_m->Dimensions().values(); columns = QSharedPointer::create(fileName, m_m->Type(), dimensions, this); } else { columns = QSharedPointer::create(fileName, m_m->Type(), this); } columns->SetWithHeader(dialog.IsWithHeader()); columns->SetSeparator(dialog.GetSeparator()); columns->SetCodec(VTextCodec::codecForMib(dialog.GetSelectedMib())); if (columns->exec() == QDialog::Accepted) { QxtCsvModel csv(fileName, nullptr, dialog.IsWithHeader(), dialog.GetSeparator(), VTextCodec::codecForMib(dialog.GetSelectedMib())); const QVector map = columns->ColumnsMap(); if (m_m->Type() == MeasurementsType::Individual) { ImportIndividualMeasurements(csv, map, dialog.IsWithHeader()); } else { ImportMultisizeMeasurements(csv, map, dialog.IsWithHeader()); } } } } //--------------------------------------------------------------------------------------------------------------------- #if defined(Q_OS_MAC) void TMainWindow::AboutToShowDockMenu() { if (QMenu *menu = qobject_cast(sender())) { menu->clear(); CreateWindowMenu(menu); menu->addSeparator(); menu->addAction(ui->actionOpenIndividual); menu->addAction(ui->actionOpenMultisize); menu->addAction(ui->actionOpenTemplate); menu->addSeparator(); QAction *actionPreferences = menu->addAction(tr("Preferences")); actionPreferences->setMenuRole(QAction::NoRole); connect(actionPreferences, &QAction::triggered, this, &TMainWindow::Preferences); } } //--------------------------------------------------------------------------------------------------------------------- void TMainWindow::OpenAt(QAction *where) { const QString path = m_curFile.left(m_curFile.indexOf(where->text())) + where->text(); if (path == m_curFile) { return; } QProcess process; process.start(QStringLiteral("/usr/bin/open"), QStringList() << path, QIODevice::ReadOnly); process.waitForFinished(); } #endif // defined(Q_OS_MAC) //--------------------------------------------------------------------------------------------------------------------- void TMainWindow::SaveCustomerName() { if (m_m->Customer() != ui->lineEditCustomerName->text()) { m_m->SetCustomer(ui->lineEditCustomerName->text()); MeasurementsWereSaved(false); } } //--------------------------------------------------------------------------------------------------------------------- void TMainWindow::SaveEmail() { if (m_m->Email() != ui->lineEditEmail->text()) { m_m->SetEmail(ui->lineEditEmail->text()); MeasurementsWereSaved(false); } } //--------------------------------------------------------------------------------------------------------------------- void TMainWindow::SaveGender(int index) { const GenderType type = static_cast(ui->comboBoxGender->itemData(index).toInt()); if (m_m->Gender() != type) { m_m->SetGender(type); MeasurementsWereSaved(false); } } //--------------------------------------------------------------------------------------------------------------------- void TMainWindow::SaveBirthDate(const QDate &date) { if (m_m->BirthDate() != date) { m_m->SetBirthDate(date); MeasurementsWereSaved(false); } } //--------------------------------------------------------------------------------------------------------------------- void TMainWindow::SaveNotes() { if (m_m->Notes() != ui->plainTextEditNotes->toPlainText()) { m_m->SetNotes(ui->plainTextEditNotes->toPlainText()); MeasurementsWereSaved(false); } } //--------------------------------------------------------------------------------------------------------------------- void TMainWindow::SavePMSystem(int index) { QString system = ui->comboBoxPMSystem->itemData(index).toString(); system.remove(0, 1); // clear p if (m_m->PMSystem() != system) { m_m->SetPMSystem(system); MeasurementsWereSaved(false); } } //--------------------------------------------------------------------------------------------------------------------- void TMainWindow::Remove() { ShowMDiagram(QString()); const int row = ui->tableWidget->currentRow(); if (row == -1) { return; } const QTableWidgetItem *nameField = ui->tableWidget->item(ui->tableWidget->currentRow(), 0); m_m->Remove(nameField->data(Qt::UserRole).toString()); MeasurementsWereSaved(false); m_search->RemoveRow(row); RefreshData(); m_search->RefreshList(ui->lineEditFind->text()); if (ui->tableWidget->rowCount() > 0) { ui->tableWidget->selectRow(row >= ui->tableWidget->rowCount() ? ui->tableWidget->rowCount() - 1 : row); } else { MFields(false); ui->actionExportToCSV->setEnabled(false); ui->lineEditName->blockSignals(true); ui->lineEditName->setText(QString()); ui->lineEditName->blockSignals(false); ui->plainTextEditDescription->blockSignals(true); ui->plainTextEditDescription->setPlainText(QString()); ui->plainTextEditDescription->blockSignals(false); ui->lineEditFullName->blockSignals(true); ui->lineEditFullName->setText(QString()); ui->lineEditFullName->blockSignals(false); ui->comboBoxMUnits->blockSignals(true); ui->comboBoxMUnits->setCurrentIndex(-1); ui->comboBoxMUnits->blockSignals(false); if (m_mType == MeasurementsType::Multisize) { ui->labelCalculatedValue->blockSignals(true); ui->doubleSpinBoxBaseValue->blockSignals(true); ui->doubleSpinBoxCorrection->blockSignals(true); ui->doubleSpinBoxShiftA->blockSignals(true); ui->doubleSpinBoxShiftB->blockSignals(true); ui->doubleSpinBoxShiftC->blockSignals(true); ui->labelCalculatedValue->setText(QString()); ui->doubleSpinBoxBaseValue->setValue(0); ui->doubleSpinBoxCorrection->setValue(0); ui->doubleSpinBoxShiftA->setValue(0); ui->doubleSpinBoxShiftB->setValue(0); ui->doubleSpinBoxShiftC->setValue(0); ui->labelCalculatedValue->blockSignals(false); ui->doubleSpinBoxBaseValue->blockSignals(false); ui->doubleSpinBoxCorrection->blockSignals(false); ui->doubleSpinBoxShiftA->blockSignals(false); ui->doubleSpinBoxShiftB->blockSignals(false); ui->doubleSpinBoxShiftC->blockSignals(false); } else { ui->labelCalculatedValue->blockSignals(true); ui->labelCalculatedValue->setText(QString()); ui->labelCalculatedValue->blockSignals(false); ui->plainTextEditFormula->blockSignals(true); ui->plainTextEditFormula->setPlainText(QString()); ui->plainTextEditFormula->blockSignals(false); } } ui->tableWidget->repaint(); // Force repain to fix paint artifacts on Mac OS X } //--------------------------------------------------------------------------------------------------------------------- void TMainWindow::MoveTop() { const int row = ui->tableWidget->currentRow(); if (row == -1) { return; } const QTableWidgetItem *nameField = ui->tableWidget->item(row, ColumnName); m_m->MoveTop(nameField->data(Qt::UserRole).toString()); MeasurementsWereSaved(false); RefreshData(); m_search->RefreshList(ui->lineEditFind->text()); ui->tableWidget->selectRow(0); ui->tableWidget->repaint(); // Force repain to fix paint artifacts on Mac OS X } //--------------------------------------------------------------------------------------------------------------------- void TMainWindow::MoveUp() { const int row = ui->tableWidget->currentRow(); if (row == -1) { return; } const QTableWidgetItem *nameField = ui->tableWidget->item(row, ColumnName); m_m->MoveUp(nameField->data(Qt::UserRole).toString()); MeasurementsWereSaved(false); RefreshData(); m_search->RefreshList(ui->lineEditFind->text()); ui->tableWidget->selectRow(row - 1); ui->tableWidget->repaint(); // Force repain to fix paint artifacts on Mac OS X } //--------------------------------------------------------------------------------------------------------------------- void TMainWindow::MoveDown() { const int row = ui->tableWidget->currentRow(); if (row == -1) { return; } const QTableWidgetItem *nameField = ui->tableWidget->item(row, ColumnName); m_m->MoveDown(nameField->data(Qt::UserRole).toString()); MeasurementsWereSaved(false); RefreshData(); m_search->RefreshList(ui->lineEditFind->text()); ui->tableWidget->selectRow(row + 1); ui->tableWidget->repaint(); // Force repain to fix paint artifacts on Mac OS X } //--------------------------------------------------------------------------------------------------------------------- void TMainWindow::MoveBottom() { const int row = ui->tableWidget->currentRow(); if (row == -1) { return; } const QTableWidgetItem *nameField = ui->tableWidget->item(row, ColumnName); m_m->MoveBottom(nameField->data(Qt::UserRole).toString()); MeasurementsWereSaved(false); RefreshData(); m_search->RefreshList(ui->lineEditFind->text()); ui->tableWidget->selectRow(ui->tableWidget->rowCount() - 1); ui->tableWidget->repaint(); // Force repain to fix paint artifacts on Mac OS X } //--------------------------------------------------------------------------------------------------------------------- void TMainWindow::Fx() { const int row = ui->tableWidget->currentRow(); if (row == -1) { return; } const QTableWidgetItem *nameField = ui->tableWidget->item(row, ColumnName); QSharedPointer meash; try { // Translate to internal look. meash = m_data->GetVariable(nameField->data(Qt::UserRole).toString()); } catch (const VExceptionBadId &e) { qCCritical(tMainWindow, "%s\n\n%s\n\n%s", qUtf8Printable(tr("Can't find measurement '%1'.").arg(nameField->text())), qUtf8Printable(e.ErrorMessage()), qUtf8Printable(e.DetailedInformation())); return; } auto *dialog = new DialogEditWrongFormula(meash->GetData(), NULL_ID, this); dialog->setWindowTitle(tr("Edit measurement")); dialog->SetMeasurementsMode(); dialog->SetFormula(VTranslateVars::TryFormulaFromUser(ui->plainTextEditFormula->toPlainText(), true)); const QString postfix = UnitsToStr(m_mUnit, true); // Show unit in dialog lable (cm, mm or inch) dialog->setPostfix(postfix); if (dialog->exec() == QDialog::Accepted) { // Fix the bug #492. https://bitbucket.org/dismine/valentina/issues/492/valentina-crashes-when-add-an-increment // Because of the bug need to take QTableWidgetItem twice time. Previous update "killed" the pointer. nameField = ui->tableWidget->item(row, ColumnName); m_m->SetMValue(nameField->data(Qt::UserRole).toString(), dialog->GetFormula()); MeasurementsWereSaved(false); RefreshData(); m_search->RefreshList(ui->lineEditFind->text()); ui->tableWidget->selectRow(row); } delete dialog; } //--------------------------------------------------------------------------------------------------------------------- void TMainWindow::AddCustom() { const QString name = GetCustomName(); qint32 currentRow = -1; if (ui->tableWidget->currentRow() == -1) { currentRow = ui->tableWidget->rowCount(); m_m->AddEmpty(name); } else { currentRow = ui->tableWidget->currentRow() + 1; const QTableWidgetItem *nameField = ui->tableWidget->item(ui->tableWidget->currentRow(), ColumnName); m_m->AddEmptyAfter(nameField->data(Qt::UserRole).toString(), name); } m_search->AddRow(currentRow); RefreshData(); m_search->RefreshList(ui->lineEditFind->text()); ui->tableWidget->selectRow(currentRow); ui->actionExportToCSV->setEnabled(true); MeasurementsWereSaved(false); ui->tableWidget->repaint(); // Force repain to fix paint artifacts on Mac OS X } //--------------------------------------------------------------------------------------------------------------------- void TMainWindow::AddKnown() { QScopedPointer dialog(new DialogMDataBase(m_m->ListKnown(), this)); if (dialog->exec() == QDialog::Accepted) { vsizetype currentRow; const QStringList list = dialog->GetNewNames(); if (ui->tableWidget->currentRow() == -1) { currentRow = ui->tableWidget->rowCount() + list.size() - 1; for (const auto &name : list) { if (m_mType == MeasurementsType::Individual) { m_m->AddEmpty(name, VAbstractApplication::VApp()->TrVars()->MFormula(name)); } else { m_m->AddEmpty(name); } m_search->AddRow(currentRow); } } else { currentRow = ui->tableWidget->currentRow() + list.size(); const QTableWidgetItem *nameField = ui->tableWidget->item(ui->tableWidget->currentRow(), ColumnName); QString after = nameField->data(Qt::UserRole).toString(); for (const auto &name : list) { if (m_mType == MeasurementsType::Individual) { m_m->AddEmptyAfter(after, name, VAbstractApplication::VApp()->TrVars()->MFormula(name)); } else { m_m->AddEmptyAfter(after, name); } m_search->AddRow(currentRow); after = name; } } RefreshData(); m_search->RefreshList(ui->lineEditFind->text()); ui->tableWidget->selectRow(static_cast(currentRow)); ui->actionExportToCSV->setEnabled(true); MeasurementsWereSaved(false); } ui->tableWidget->repaint(); // Force repain to fix paint artifacts on Mac OS X } //--------------------------------------------------------------------------------------------------------------------- void TMainWindow::AddSeparator() { const QString name = GetCustomName(); qint32 currentRow = -1; if (ui->tableWidget->currentRow() == -1) { currentRow = ui->tableWidget->rowCount(); m_m->AddSeparator(name); } else { currentRow = ui->tableWidget->currentRow() + 1; const QTableWidgetItem *nameField = ui->tableWidget->item(ui->tableWidget->currentRow(), ColumnName); m_m->AddSeparatorAfter(nameField->data(Qt::UserRole).toString(), name); } m_search->AddRow(currentRow); RefreshData(); m_search->RefreshList(ui->lineEditFind->text()); ui->tableWidget->selectRow(currentRow); ui->actionExportToCSV->setEnabled(true); MeasurementsWereSaved(false); ui->tableWidget->repaint(); // Force repain to fix paint artifacts on Mac OS X } //--------------------------------------------------------------------------------------------------------------------- void TMainWindow::ImportFromPattern() { if (m_m == nullptr) { return; } const QString filter(tr("Pattern files (*.val)")); // Use standard path to individual measurements QString pathTo = MApplication::VApp()->TapeSettings()->GetPathPattern(); const QString mPath = QFileDialog::getOpenFileName(this, tr("Import from a pattern"), pathTo, filter, nullptr, VAbstractApplication::VApp()->NativeFileDialog()); if (mPath.isEmpty()) { return; } VLockGuard tmp(mPath); if (not tmp.IsLocked()) { qCCritical(tMainWindow, "%s", qUtf8Printable(tr("This file already opened in another window."))); return; } QStringList measurements; try { VPatternConverter converter(mPath); QScopedPointer doc(new VLitePattern()); doc->setXMLContent(converter.Convert()); measurements = doc->ListMeasurements(); } catch (VException &e) { qCCritical(tMainWindow, "%s\n\n%s\n\n%s", qUtf8Printable(tr("File error.")), qUtf8Printable(e.ErrorMessage()), qUtf8Printable(e.DetailedInformation())); return; } measurements = FilterMeasurements(measurements, m_m->ListAll()); vsizetype currentRow; if (ui->tableWidget->currentRow() == -1) { currentRow = ui->tableWidget->rowCount() + measurements.size() - 1; for (auto &mName : measurements) { m_m->AddEmpty(mName); } } else { currentRow = ui->tableWidget->currentRow() + measurements.size(); const QTableWidgetItem *nameField = ui->tableWidget->item(ui->tableWidget->currentRow(), ColumnName); QString after = nameField->data(Qt::UserRole).toString(); for (auto &mName : measurements) { m_m->AddEmptyAfter(after, mName); after = mName; } } RefreshData(); m_search->RefreshList(ui->lineEditFind->text()); ui->tableWidget->selectRow(static_cast(currentRow)); MeasurementsWereSaved(false); } //--------------------------------------------------------------------------------------------------------------------- void TMainWindow::DimensionABaseChanged() { m_currentDimensionA = m_gradationDimensionA->currentData().toDouble(); const QList dimensions = m_m->Dimensions().values(); if (dimensions.size() > 1) { MeasurementDimension_p dimension = dimensions.at(1); InitDimensionGradation(1, dimension, m_gradationDimensionB); if (dimensions.size() > 2) { dimension = dimensions.at(2); InitDimensionGradation(2, dimension, m_gradationDimensionC); } } m_gradation->start(); } //--------------------------------------------------------------------------------------------------------------------- void TMainWindow::DimensionBBaseChanged() { m_currentDimensionB = m_gradationDimensionB->currentData().toDouble(); const QList dimensions = m_m->Dimensions().values(); if (dimensions.size() > 2) { const MeasurementDimension_p &dimension = dimensions.at(2); InitDimensionGradation(2, dimension, m_gradationDimensionC); } m_gradation->start(); } //--------------------------------------------------------------------------------------------------------------------- void TMainWindow::DimensionCBaseChanged() { m_currentDimensionC = m_gradationDimensionC->currentData().toDouble(); m_gradation->start(); } //--------------------------------------------------------------------------------------------------------------------- void TMainWindow::GradationChanged() { m_gradation->stop(); const int row = ui->tableWidget->currentRow(); RefreshData(); m_search->RefreshList(ui->lineEditFind->text()); ui->tableWidget->selectRow(row); } //--------------------------------------------------------------------------------------------------------------------- void TMainWindow::ShowMData() { ShowNewMData(true); } //--------------------------------------------------------------------------------------------------------------------- void TMainWindow::ShowNewMData(bool fresh) { if (ui->tableWidget->rowCount() <= 0) { MFields(false); return; } MFields(true); if (ui->tableWidget->currentRow() == -1) { ui->tableWidget->blockSignals(true); ui->tableWidget->selectRow(0); ui->tableWidget->blockSignals(false); } const QTableWidgetItem *nameField = ui->tableWidget->item(ui->tableWidget->currentRow(), ColumnName); // name SCASSERT(nameField != nullptr) QSharedPointer meash; try { // Translate to internal look. meash = m_data->GetVariable(nameField->data(Qt::UserRole).toString()); } catch (const VExceptionBadId &e) { Q_UNUSED(e) MFields(false); return; } ShowMDiagram(meash->GetName()); ui->labelFullName->setVisible(meash->GetType() == VarType::Measurement); ui->lineEditFullName->setVisible(meash->GetType() == VarType::Measurement); // Don't block all signal for QLineEdit. Need for correct handle with clear button. disconnect(ui->lineEditName, &QLineEdit::textEdited, this, &TMainWindow::SaveMName); ui->plainTextEditDescription->blockSignals(true); if (meash->IsCustom()) { ui->plainTextEditDescription->setPlainText(meash->GetDescription()); ui->lineEditFullName->setText(meash->GetGuiText()); ui->lineEditName->setText(ClearCustomName(meash->GetName())); } else { // Show known ui->plainTextEditDescription->setPlainText( VAbstractApplication::VApp()->TrVars()->Description(meash->GetName())); ui->lineEditFullName->setText(VAbstractApplication::VApp()->TrVars()->GuiText(meash->GetName())); ui->lineEditName->setText(nameField->text()); } connect(ui->lineEditName, &QLineEdit::textEdited, this, &TMainWindow::SaveMName); ui->plainTextEditDescription->blockSignals(false); ui->labelMUnits->setVisible(meash->GetType() == VarType::Measurement); ui->comboBoxMUnits->setVisible(meash->GetType() == VarType::Measurement); ui->comboBoxMUnits->blockSignals(true); ui->comboBoxMUnits->setCurrentIndex( ui->comboBoxMUnits->findData(static_cast(meash->IsSpecialUnits() ? MUnits::Degrees : MUnits::Table))); ui->comboBoxMUnits->blockSignals(false); if (m_mType == MeasurementsType::Multisize) { ui->labelCalculated->setVisible(meash->GetType() == VarType::Measurement); ui->labelCalculatedValue->setVisible(meash->GetType() == VarType::Measurement); ui->labelBaseValue->setVisible(meash->GetType() == VarType::Measurement); ui->doubleSpinBoxBaseValue->setVisible(meash->GetType() == VarType::Measurement); ui->labelCorrection->setVisible(meash->GetType() == VarType::Measurement); ui->doubleSpinBoxCorrection->setVisible(meash->GetType() == VarType::Measurement); ui->labelShiftA->setVisible(meash->GetType() == VarType::Measurement); ui->doubleSpinBoxShiftA->setVisible(meash->GetType() == VarType::Measurement); const QList dimensions = m_m->Dimensions().values(); if (dimensions.size() > 1) { ui->labelShiftB->setVisible(meash->GetType() == VarType::Measurement); ui->doubleSpinBoxShiftB->setVisible(meash->GetType() == VarType::Measurement); } if (dimensions.size() > 2) { ui->labelShiftC->setVisible(meash->GetType() == VarType::Measurement); ui->doubleSpinBoxShiftC->setVisible(meash->GetType() == VarType::Measurement); } ui->labelCalculatedValue->blockSignals(true); ui->doubleSpinBoxBaseValue->blockSignals(true); ui->doubleSpinBoxCorrection->blockSignals(true); ui->doubleSpinBoxShiftA->blockSignals(true); ui->doubleSpinBoxShiftB->blockSignals(true); ui->doubleSpinBoxShiftC->blockSignals(true); QString calculatedValue; if (meash->IsSpecialUnits()) { const qreal value = *m_data->DataVariables()->value(meash->GetName())->GetValue(); calculatedValue = VAbstractApplication::VApp()->LocaleToString(value) + QChar(QChar::Space) + degreeSymbol; } else { const QString postfix = UnitsToStr(m_pUnit); // Show unit in dialog lable (cm, mm or inch) const qreal value = UnitConvertor(*m_data->DataVariables()->value(meash->GetName())->GetValue(), m_mUnit, m_pUnit); calculatedValue = VAbstractApplication::VApp()->LocaleToString(value) + QChar(QChar::Space) + postfix; } ui->labelCalculatedValue->setText(calculatedValue); if (fresh) { ui->doubleSpinBoxBaseValue->setValue(meash->GetBase()); ui->doubleSpinBoxCorrection->setValue( meash->GetCorrection(m_currentDimensionA, m_currentDimensionB, m_currentDimensionC)); ui->doubleSpinBoxShiftA->setValue(meash->GetShiftA()); ui->doubleSpinBoxShiftB->setValue(meash->GetShiftB()); ui->doubleSpinBoxShiftC->setValue(meash->GetShiftC()); } ui->labelCalculatedValue->blockSignals(false); ui->doubleSpinBoxBaseValue->blockSignals(false); ui->doubleSpinBoxCorrection->blockSignals(false); ui->doubleSpinBoxShiftA->blockSignals(false); ui->doubleSpinBoxShiftB->blockSignals(false); ui->doubleSpinBoxShiftC->blockSignals(false); } else { ui->labelCalculated->setVisible(meash->GetType() == VarType::Measurement); ui->labelCalculatedValue->setVisible(meash->GetType() == VarType::Measurement); ui->labelFormula->setVisible(meash->GetType() == VarType::Measurement); ui->plainTextEditFormula->setVisible(meash->GetType() == VarType::Measurement); ui->pushButtonGrow->setVisible(meash->GetType() == VarType::Measurement); ui->toolButtonExpr->setVisible(meash->GetType() == VarType::Measurement); ui->labelDimension->setVisible(meash->GetType() == VarType::Measurement); ui->comboBoxDimension->setVisible(meash->GetType() == VarType::Measurement); EvalFormula(meash->GetFormula(), false, meash->GetData(), ui->labelCalculatedValue, meash->IsSpecialUnits()); ui->plainTextEditFormula->blockSignals(true); QString formula = VTranslateVars::TryFormulaToUser(meash->GetFormula(), VAbstractApplication::VApp()->Settings()->GetOsSeparator()); ui->plainTextEditFormula->setPlainText(formula); ui->plainTextEditFormula->blockSignals(false); ui->comboBoxDimension->blockSignals(true); ui->comboBoxDimension->setCurrentIndex( ui->comboBoxDimension->findData(static_cast(meash->GetDimension()))); ui->comboBoxDimension->blockSignals(false); } MeasurementGUI(); } //--------------------------------------------------------------------------------------------------------------------- void TMainWindow::ShowMDiagram(const QString &name) { const VTranslateVars *trv = VAbstractApplication::VApp()->TrVars(); const QString number = trv->MNumber(name); if (number.isEmpty()) { ui->labelDiagram->setText(tr("

?

" "

Unknown measurement

")); } else { ui->labelDiagram->setText(QString("

%1

" "

%2. %3

") .arg(DialogMDataBase::ImgTag(number), number, trv->GuiText(name))); } } //--------------------------------------------------------------------------------------------------------------------- void TMainWindow::DeployFormula() { SCASSERT(ui->plainTextEditFormula != nullptr) SCASSERT(ui->pushButtonGrow != nullptr) const QTextCursor cursor = ui->plainTextEditFormula->textCursor(); if (ui->plainTextEditFormula->height() < DIALOG_MAX_FORMULA_HEIGHT) { ui->plainTextEditFormula->setFixedHeight(DIALOG_MAX_FORMULA_HEIGHT); // Set icon from theme (internal for Windows system) ui->pushButtonGrow->setIcon(QIcon::fromTheme(QStringLiteral("go-next"))); } else { ui->plainTextEditFormula->setFixedHeight(m_formulaBaseHeight); // Set icon from theme (internal for Windows system) ui->pushButtonGrow->setIcon(QIcon::fromTheme(QStringLiteral("go-down"))); } // I found that after change size of formula field, it was filed for angle formula, field for formula became black. // This code prevent this. setUpdatesEnabled(false); repaint(); setUpdatesEnabled(true); ui->plainTextEditFormula->setFocus(); ui->plainTextEditFormula->setTextCursor(cursor); } //--------------------------------------------------------------------------------------------------------------------- void TMainWindow::SaveMName(const QString &text) { const int row = ui->tableWidget->currentRow(); if (row == -1) { return; } const QTableWidgetItem *nameField = ui->tableWidget->item(ui->tableWidget->currentRow(), ColumnName); QSharedPointer meash; try { // Translate to internal look. meash = m_data->GetVariable(nameField->data(Qt::UserRole).toString()); } catch (const VExceptionBadId &e) { qCWarning(tMainWindow, "%s\n\n%s\n\n%s", qUtf8Printable(tr("Can't find measurement '%1'.").arg(nameField->text())), qUtf8Printable(e.ErrorMessage()), qUtf8Printable(e.DetailedInformation())); return; } QString newName = text; if (meash->IsCustom()) { newName.isEmpty() ? newName = GetCustomName() : newName = CustomMSign + newName; if (not m_data->IsUnique(newName)) { qint32 num = 2; QString name = newName; do { name = name + QChar('_') + QString::number(num); num++; } while (not m_data->IsUnique(name)); newName = name; } m_m->SetMName(nameField->text(), newName); MeasurementsWereSaved(false); RefreshData(); m_search->RefreshList(ui->lineEditFind->text()); ui->tableWidget->blockSignals(true); ui->tableWidget->selectRow(row); ui->tableWidget->blockSignals(false); } else { qCWarning(tMainWindow, "%s", qUtf8Printable(tr("The name of known measurement forbidden to change."))); } } //--------------------------------------------------------------------------------------------------------------------- void TMainWindow::SaveMValue() { const int row = ui->tableWidget->currentRow(); if (row == -1) { return; } const QTableWidgetItem *nameField = ui->tableWidget->item(row, ColumnName); QString text = ui->plainTextEditFormula->toPlainText(); QTableWidgetItem *formulaField = ui->tableWidget->item(row, ColumnFormula); if (formulaField->text() == text) { QTableWidgetItem *result = ui->tableWidget->item(row, ColumnCalcValue); const QString postfix = UnitsToStr(m_mUnit); // Show unit in dialog lable (cm, mm or inch) ui->labelCalculatedValue->setText(result->text() + QChar(QChar::Space) + postfix); return; } if (text.isEmpty()) { const QString postfix = UnitsToStr(m_mUnit); // Show unit in dialog lable (cm, mm or inch) ui->labelCalculatedValue->setText(tr("Error") + " (" + postfix + "). " + tr("Empty field.")); return; } QSharedPointer meash; try { // Translate to internal look. meash = m_data->GetVariable(nameField->data(Qt::UserRole).toString()); } catch (const VExceptionBadId &e) { qCWarning(tMainWindow, "%s\n\n%s\n\n%s", qUtf8Printable(tr("Can't find measurement '%1'.").arg(nameField->text())), qUtf8Printable(e.ErrorMessage()), qUtf8Printable(e.DetailedInformation())); return; } if (not EvalFormula(text, true, meash->GetData(), ui->labelCalculatedValue, meash->IsSpecialUnits())) { return; } try { const QString formula = VAbstractApplication::VApp()->TrVars()->FormulaFromUser( text, VAbstractApplication::VApp()->Settings()->GetOsSeparator()); m_m->SetMValue(nameField->data(Qt::UserRole).toString(), formula); } catch (qmu::QmuParserError &e) // Just in case something bad will happen { Q_UNUSED(e) return; } MeasurementsWereSaved(false); const QTextCursor cursor = ui->plainTextEditFormula->textCursor(); RefreshData(); m_search->RefreshList(ui->lineEditFind->text()); ui->tableWidget->blockSignals(true); ui->tableWidget->selectRow(row); ui->tableWidget->blockSignals(false); ui->plainTextEditFormula->setTextCursor(cursor); } //--------------------------------------------------------------------------------------------------------------------- void TMainWindow::SaveMBaseValue(double value) { const int row = ui->tableWidget->currentRow(); if (row == -1) { return; } const QTableWidgetItem *nameField = ui->tableWidget->item(ui->tableWidget->currentRow(), ColumnName); m_m->SetMBaseValue(nameField->data(Qt::UserRole).toString(), value); MeasurementsWereSaved(false); RefreshData(); m_search->RefreshList(ui->lineEditFind->text()); ui->tableWidget->blockSignals(true); ui->tableWidget->selectRow(row); ui->tableWidget->blockSignals(false); ShowNewMData(false); } //--------------------------------------------------------------------------------------------------------------------- void TMainWindow::SaveMShiftA(double value) { const int row = ui->tableWidget->currentRow(); if (row == -1) { return; } const QTableWidgetItem *nameField = ui->tableWidget->item(ui->tableWidget->currentRow(), ColumnName); m_m->SetMShiftA(nameField->data(Qt::UserRole).toString(), value); MeasurementsWereSaved(false); RefreshData(); m_search->RefreshList(ui->lineEditFind->text()); ui->tableWidget->blockSignals(true); ui->tableWidget->selectRow(row); ui->tableWidget->blockSignals(false); ShowNewMData(false); } //--------------------------------------------------------------------------------------------------------------------- void TMainWindow::SaveMShiftB(double value) { const int row = ui->tableWidget->currentRow(); if (row == -1) { return; } const QTableWidgetItem *nameField = ui->tableWidget->item(ui->tableWidget->currentRow(), ColumnName); m_m->SetMShiftB(nameField->data(Qt::UserRole).toString(), value); MeasurementsWereSaved(false); RefreshData(); m_search->RefreshList(ui->lineEditFind->text()); ui->tableWidget->blockSignals(true); ui->tableWidget->selectRow(row); ui->tableWidget->blockSignals(false); ShowNewMData(false); } //--------------------------------------------------------------------------------------------------------------------- void TMainWindow::SaveMShiftC(double value) { const int row = ui->tableWidget->currentRow(); if (row == -1) { return; } const QTableWidgetItem *nameField = ui->tableWidget->item(ui->tableWidget->currentRow(), ColumnName); m_m->SetMShiftC(nameField->data(Qt::UserRole).toString(), value); MeasurementsWereSaved(false); RefreshData(); m_search->RefreshList(ui->lineEditFind->text()); ui->tableWidget->blockSignals(true); ui->tableWidget->selectRow(row); ui->tableWidget->blockSignals(false); ShowNewMData(false); } //--------------------------------------------------------------------------------------------------------------------- void TMainWindow::SaveMCorrectionValue(double value) { const int row = ui->tableWidget->currentRow(); if (row == -1) { return; } const QTableWidgetItem *nameField = ui->tableWidget->item(ui->tableWidget->currentRow(), ColumnName); m_m->SetMCorrectionValue(nameField->data(Qt::UserRole).toString(), m_currentDimensionA, m_currentDimensionB, m_currentDimensionC, value); MeasurementsWereSaved(false); RefreshData(); m_search->RefreshList(ui->lineEditFind->text()); ui->tableWidget->blockSignals(true); ui->tableWidget->selectRow(row); ui->tableWidget->blockSignals(false); ShowNewMData(false); } //--------------------------------------------------------------------------------------------------------------------- void TMainWindow::SaveMDescription() { const int row = ui->tableWidget->currentRow(); if (row == -1) { return; } const QTableWidgetItem *nameField = ui->tableWidget->item(ui->tableWidget->currentRow(), ColumnName); m_m->SetMDescription(nameField->data(Qt::UserRole).toString(), ui->plainTextEditDescription->toPlainText()); MeasurementsWereSaved(false); const QTextCursor cursor = ui->plainTextEditDescription->textCursor(); RefreshData(); ui->tableWidget->blockSignals(true); ui->tableWidget->selectRow(row); ui->tableWidget->blockSignals(false); ui->plainTextEditDescription->setTextCursor(cursor); } //--------------------------------------------------------------------------------------------------------------------- void TMainWindow::SaveMFullName() { const int row = ui->tableWidget->currentRow(); if (row == -1) { return; } const QTableWidgetItem *nameField = ui->tableWidget->item(ui->tableWidget->currentRow(), ColumnName); QSharedPointer meash; try { // Translate to internal look. meash = m_data->GetVariable(nameField->data(Qt::UserRole).toString()); } catch (const VExceptionBadId &e) { qCWarning(tMainWindow, "%s\n\n%s\n\n%s", qUtf8Printable(tr("Can't find measurement '%1'.").arg(nameField->text())), qUtf8Printable(e.ErrorMessage()), qUtf8Printable(e.DetailedInformation())); return; } if (meash->IsCustom()) { m_m->SetMFullName(nameField->data(Qt::UserRole).toString(), ui->lineEditFullName->text()); MeasurementsWereSaved(false); RefreshData(); ui->tableWidget->blockSignals(true); ui->tableWidget->selectRow(row); ui->tableWidget->blockSignals(false); } else { qCWarning(tMainWindow, "%s", qUtf8Printable(tr("The full name of known measurement forbidden to change."))); } } //--------------------------------------------------------------------------------------------------------------------- void TMainWindow::SaveMUnits() { const int row = ui->tableWidget->currentRow(); if (row == -1) { return; } const QTableWidgetItem *nameField = ui->tableWidget->item(ui->tableWidget->currentRow(), ColumnName); const MUnits units = static_cast(ui->comboBoxMUnits->currentData().toInt()); m_m->SetMSpecialUnits(nameField->data(Qt::UserRole).toString(), units == MUnits::Degrees); MeasurementsWereSaved(false); RefreshData(); m_search->RefreshList(ui->lineEditFind->text()); ui->tableWidget->blockSignals(true); ui->tableWidget->selectRow(row); ui->tableWidget->blockSignals(false); ShowNewMData(false); } //--------------------------------------------------------------------------------------------------------------------- void TMainWindow::SaveMDimension() { const int row = ui->tableWidget->currentRow(); if (row == -1) { return; } const QTableWidgetItem *nameField = ui->tableWidget->item(ui->tableWidget->currentRow(), ColumnName); const IMD dimension = static_cast(ui->comboBoxDimension->currentData().toInt()); m_m->SetMDimension(nameField->data(Qt::UserRole).toString(), dimension); MeasurementsWereSaved(false); RefreshData(); m_search->RefreshList(ui->lineEditFind->text()); ui->tableWidget->blockSignals(true); ui->tableWidget->selectRow(row); ui->tableWidget->blockSignals(false); ShowNewMData(false); } //--------------------------------------------------------------------------------------------------------------------- void TMainWindow::FullCircumferenceChanged(bool checked) { m_m->SetFullCircumference(checked); MeasurementsWereSaved(false); InitDimensionsBaseValue(); InitDimensionControls(); } //--------------------------------------------------------------------------------------------------------------------- void TMainWindow::ExportToIndividual() { QString dir; if (m_curFile.isEmpty()) { dir = MApplication::VApp()->TapeSettings()->GetPathIndividualMeasurements(); } else { dir = QFileInfo(m_curFile).absolutePath(); } bool usedNotExistedDir = false; QDir directory(dir); if (not directory.exists()) { usedNotExistedDir = directory.mkpath(QChar('.')); } QString filters = tr("Individual measurements") + QStringLiteral(" (*.vit)"); QString fName = tr("measurements.vit"); QString fileName = QFileDialog::getSaveFileName(this, tr("Export to individual"), dir + QChar('/') + fName, filters, nullptr, VAbstractApplication::VApp()->NativeFileDialog()); auto RemoveTempDir = qScopeGuard( [usedNotExistedDir, dir]() { if (usedNotExistedDir) { QDir(dir).rmpath(QChar('.')); } }); if (fileName.isEmpty()) { return; } QString suffix = QStringLiteral("vit"); QFileInfo f(fileName); if (f.suffix().isEmpty() && f.suffix() != suffix) { fileName += QChar('.') + suffix; } QScopedPointer tmpData( new VContainer(VAbstractApplication::VApp()->TrVars(), &m_mUnit, VContainer::UniqueNamespace())); VMeasurements individualMeasurements(m_mUnit, tmpData.data()); const QMap> orderedTable = OrderedMeasurments(); QMap>::const_iterator iMap; for (iMap = orderedTable.constBegin(); iMap != orderedTable.constEnd(); ++iMap) { const QSharedPointer &meash = iMap.value(); individualMeasurements.AddEmpty(meash->GetName()); individualMeasurements.SetMValue(meash->GetName(), QString::number(*meash->GetValue())); individualMeasurements.SetMSpecialUnits(meash->GetName(), meash->IsSpecialUnits()); if (meash->IsCustom()) { individualMeasurements.SetMDescription(meash->GetName(), meash->GetDescription()); individualMeasurements.SetMFullName(meash->GetName(), meash->GetGuiText()); } } QString error; const bool result = individualMeasurements.SaveDocument(fileName, error); if (not result) { QMessageBox messageBox; messageBox.setIcon(QMessageBox::Warning); messageBox.setInformativeText(tr("Could not save the file")); messageBox.setDefaultButton(QMessageBox::Ok); messageBox.setDetailedText(error); messageBox.setStandardButtons(QMessageBox::Ok); messageBox.exec(); return; } } //--------------------------------------------------------------------------------------------------------------------- void TMainWindow::RestrictFirstDimesion() { const QList dimensions = m_m->Dimensions().values(); const QMap restrictions = m_m->GetRestrictions(); bool fullCircumference = m_m->IsFullCircumference(); DialogRestrictDimension dialog(dimensions, restrictions, RestrictDimension::First, fullCircumference, this); if (dialog.exec() == QDialog::Rejected) { return; } m_m->SetRestrictions(dialog.Restrictions()); MeasurementsWereSaved(false); if (not dimensions.isEmpty()) { InitDimensionGradation(0, dimensions.at(0), m_gradationDimensionA); } } //--------------------------------------------------------------------------------------------------------------------- void TMainWindow::RestrictSecondDimesion() { const QList dimensions = m_m->Dimensions().values(); const QMap restrictions = m_m->GetRestrictions(); bool fullCircumference = m_m->IsFullCircumference(); DialogRestrictDimension dialog(dimensions, restrictions, RestrictDimension::Second, fullCircumference, this); if (dialog.exec() == QDialog::Rejected) { return; } m_m->SetRestrictions(dialog.Restrictions()); MeasurementsWereSaved(false); DimensionABaseChanged(); // trigger refresh } //--------------------------------------------------------------------------------------------------------------------- void TMainWindow::RestrictThirdDimesion() { const QList dimensions = m_m->Dimensions().values(); const QMap restrictions = m_m->GetRestrictions(); bool fullCircumference = m_m->IsFullCircumference(); DialogRestrictDimension dialog(dimensions, restrictions, RestrictDimension::Third, fullCircumference, this); if (dialog.exec() == QDialog::Rejected) { return; } m_m->SetRestrictions(dialog.Restrictions()); MeasurementsWereSaved(false); DimensionABaseChanged(); // trigger refresh } //--------------------------------------------------------------------------------------------------------------------- void TMainWindow::EditDimensionLabels() { const QMap dimensions = m_m->Dimensions(); DialogDimensionLabels dialog(dimensions, m_m->IsFullCircumference(), this); if (dialog.exec() == QDialog::Rejected) { return; } m_m->SetDimensionLabels(dialog.Labels()); MeasurementsWereSaved(false); InitDimensionsBaseValue(); InitDimensionControls(); } //--------------------------------------------------------------------------------------------------------------------- void TMainWindow::DimensionCustomNames() { const QMap dimensions = m_m->Dimensions(); DialogDimensionCustomNames dialog(dimensions, this); if (dialog.exec() == QDialog::Rejected) { return; } m_m->SetDimensionCustomNames(dialog.CustomNames()); MeasurementsWereSaved(false); InitDimensionControls(); } //--------------------------------------------------------------------------------------------------------------------- void TMainWindow::AskDefaultSettings() { if (MApplication::VApp()->IsAppInGUIMode()) { VTapeSettings *settings = MApplication::VApp()->TapeSettings(); if (not settings->IsLocaleSelected()) { QGuiApplication::setOverrideCursor(QCursor(Qt::WaitCursor)); DialogSelectLanguage dialog(this); QGuiApplication::restoreOverrideCursor(); dialog.setWindowModality(Qt::WindowModal); if (dialog.exec() == QDialog::Accepted) { QString locale = dialog.Locale(); settings->SetLocale(locale); VAbstractApplication::VApp()->LoadTranslation(locale); } } if (settings->IsAskCollectStatistic()) { DialogAskCollectStatistic dialog(this); if (dialog.exec() == QDialog::Accepted) { settings->SetCollectStatistic(dialog.CollectStatistic()); } settings->SetAskCollectStatistic(false); } if (settings->IsCollectStatistic()) { auto *statistic = VGAnalytics::Instance(); statistic->SetGUILanguage(settings->GetLocale()); QString clientID = settings->GetClientID(); bool freshID = false; if (clientID.isEmpty()) { clientID = QUuid::createUuid().toString(); settings->SetClientID(clientID); statistic->SetClientID(clientID); freshID = true; } statistic->Enable(true); const qint64 uptime = MApplication::VApp()->AppUptime(); freshID ? statistic->SendAppFreshInstallEvent(uptime) : statistic->SendAppStartEvent(uptime); } } } //--------------------------------------------------------------------------------------------------------------------- void TMainWindow::SetupMenu() { // File connect(ui->actionNew, &QAction::triggered, this, &TMainWindow::FileNew); ui->actionNew->setShortcuts(QKeySequence::New); connect(ui->actionOpenIndividual, &QAction::triggered, this, &TMainWindow::OpenIndividual); connect(ui->actionOpenMultisize, &QAction::triggered, this, &TMainWindow::OpenMultisize); connect(ui->actionOpenTemplate, &QAction::triggered, this, &TMainWindow::OpenTemplate); connect(ui->actionCreateFromExisting, &QAction::triggered, this, &TMainWindow::CreateFromExisting); connect(ui->actionSave, &QAction::triggered, this, &TMainWindow::FileSave); ui->actionSave->setShortcuts(QKeySequence::Save); connect(ui->actionSaveAs, &QAction::triggered, this, &TMainWindow::FileSaveAs); ui->actionSaveAs->setShortcuts(QKeySequence::SaveAs); connect(ui->actionExportToCSV, &QAction::triggered, this, &TMainWindow::ExportDataToCSV); connect(ui->actionImportFromCSV, &QAction::triggered, this, &TMainWindow::ImportDataFromCSV); connect(ui->actionReadOnly, &QAction::triggered, this, [this](bool ro) { if (not m_mIsReadOnly) { m_m->SetReadOnly(ro); MeasurementsWereSaved(false); UpdatePadlock(ro); UpdateWindowTitle(); } else { if (auto *action = qobject_cast(this->sender())) { action->setChecked(true); } } }); connect(ui->actionPreferences, &QAction::triggered, this, &TMainWindow::Preferences); for (auto &recentFileAct : m_recentFileActs) { auto *action = new QAction(this); recentFileAct = action; connect(action, &QAction::triggered, this, [this]() { if (auto *senderAction = qobject_cast(sender())) { const QString filePath = senderAction->data().toString(); if (not filePath.isEmpty()) { LoadFile(filePath); } } }); ui->menuFile->insertAction(ui->actionPreferences, recentFileAct); recentFileAct->setVisible(false); } m_separatorAct = new QAction(this); m_separatorAct->setSeparator(true); m_separatorAct->setVisible(false); ui->menuFile->insertAction(ui->actionPreferences, m_separatorAct); connect(ui->actionQuit, &QAction::triggered, this, &TMainWindow::close); ui->actionQuit->setShortcuts(QKeySequence::Quit); // Measurements connect(ui->actionAddCustom, &QAction::triggered, this, &TMainWindow::AddCustom); connect(ui->actionAddKnown, &QAction::triggered, this, &TMainWindow::AddKnown); connect(ui->actionAddSeparator, &QAction::triggered, this, &TMainWindow::AddSeparator); connect(ui->actionDatabase, &QAction::triggered, MApplication::VApp(), &MApplication::ShowDataBase); connect(ui->actionImportFromPattern, &QAction::triggered, this, &TMainWindow::ImportFromPattern); // Window connect(ui->menuWindow, &QMenu::aboutToShow, this, &TMainWindow::AboutToShowWindowMenu); AboutToShowWindowMenu(); // Help connect(ui->actionAboutQt, &QAction::triggered, this, [this]() { QMessageBox::aboutQt(this, tr("About Qt")); }); connect(ui->actionAboutTape, &QAction::triggered, this, [this]() { auto *aboutDialog = new DialogAboutTape(this); aboutDialog->setAttribute(Qt::WA_DeleteOnClose, true); aboutDialog->show(); }); // Actions for recent files loaded by a tape window application. UpdateRecentFileActions(); } //--------------------------------------------------------------------------------------------------------------------- void TMainWindow::InitWindow() { SCASSERT(m_m != nullptr) ui->labelToolTip->setVisible(false); ui->tabWidget->setVisible(true); ui->tabWidget->setCurrentIndex(0); // measurements ui->plainTextEditNotes->setEnabled(true); ui->toolBarGradation->setVisible(true); if (m_mType == MeasurementsType::Multisize) { InitMenu(); ui->labelMType->setText(tr("Multisize measurements")); InitDimensionsBaseValue(); HackDimensionBaseValue(); // Because Qt Designer doesn't know about our deleting we will create empty objects for correct // working the retranslation UI // Tab Measurements HackWidget(&ui->horizontalLayoutValue); HackWidget(&ui->plainTextEditFormula); HackWidget(&ui->toolButtonExpr); HackWidget(&ui->labelFormula); HackWidget(&ui->pushButtonGrow); HackWidget(&ui->labelDimension); HackWidget(&ui->comboBoxDimension); // Tab Information HackWidget(&ui->lineEditCustomerName); HackWidget(&ui->comboBoxGender); HackWidget(&ui->lineEditEmail); HackWidget(&ui->labelCustomerName); HackWidget(&ui->labelBirthDate); HackWidget(&ui->dateEditBirthDate); HackWidget(&ui->labelGender); HackWidget(&ui->labelEmail); InitDimensionControls(); ShowDimensionControls(); SetDimensionBases(); InitDimesionShifts(); connect(ui->doubleSpinBoxBaseValue, QOverload::of(&QDoubleSpinBox::valueChanged), this, &TMainWindow::SaveMBaseValue); connect(ui->doubleSpinBoxCorrection, QOverload::of(&QDoubleSpinBox::valueChanged), this, &TMainWindow::SaveMCorrectionValue); connect(ui->doubleSpinBoxShiftA, QOverload::of(&QDoubleSpinBox::valueChanged), this, &TMainWindow::SaveMShiftA); connect(ui->doubleSpinBoxShiftB, QOverload::of(&QDoubleSpinBox::valueChanged), this, &TMainWindow::SaveMShiftB); connect(ui->doubleSpinBoxShiftC, QOverload::of(&QDoubleSpinBox::valueChanged), this, &TMainWindow::SaveMShiftC); HackDimensionShifts(); SetDecimals(); } else { ui->labelMType->setText(tr("Individual measurements")); ui->lineEditCustomerName->setEnabled(true); ui->dateEditBirthDate->setEnabled(true); ui->comboBoxGender->setEnabled(true); ui->lineEditEmail->setEnabled(true); // Tab Measurements HackWidget(&ui->doubleSpinBoxBaseValue); HackWidget(&ui->doubleSpinBoxCorrection); HackWidget(&ui->labelBaseValue); HackWidget(&ui->labelCorrection); HackDimensionShifts(); // Tab Information HackDimensionBaseValue(); HackWidget(&ui->frameBaseValue); HackWidget(&ui->labelBaseValues); ui->lineEditCustomerName->setText(m_m->Customer()); ui->comboBoxGender->clear(); InitGender(ui->comboBoxGender); const qint32 index = ui->comboBoxGender->findData(static_cast(m_m->Gender())); ui->comboBoxGender->setCurrentIndex(index); { const QLocale dateLocale = QLocale(VAbstractApplication::VApp()->Settings()->GetLocale()); ui->dateEditBirthDate->setLocale(dateLocale); ui->dateEditBirthDate->setDisplayFormat(dateLocale.dateFormat()); ui->dateEditBirthDate->setDate(m_m->BirthDate()); } ui->lineEditEmail->setText(m_m->Email()); connect(ui->lineEditCustomerName, &QLineEdit::editingFinished, this, &TMainWindow::SaveCustomerName); connect(ui->lineEditEmail, &QLineEdit::editingFinished, this, &TMainWindow::SaveEmail); connect(ui->comboBoxGender, QOverload::of(&QComboBox::currentIndexChanged), this, &TMainWindow::SaveGender); connect(ui->dateEditBirthDate, &QDateEdit::dateChanged, this, &TMainWindow::SaveBirthDate); connect(ui->pushButtonGrow, &QPushButton::clicked, this, &TMainWindow::DeployFormula); this->m_formulaBaseHeight = ui->plainTextEditFormula->height(); connect(ui->plainTextEditFormula, &QPlainTextEdit::textChanged, this, &TMainWindow::SaveMValue, Qt::UniqueConnection); connect(ui->toolButtonExpr, &QToolButton::clicked, this, &TMainWindow::Fx); InitMeasurementDimension(); connect(ui->comboBoxDimension, QOverload::of(&QComboBox::currentIndexChanged), this, &TMainWindow::SaveMDimension); } ui->comboBoxPMSystem->setEnabled(true); ui->comboBoxPMSystem->clear(); InitPMSystems(ui->comboBoxPMSystem); const qint32 index = ui->comboBoxPMSystem->findData(QLatin1Char('p') + m_m->PMSystem()); ui->comboBoxPMSystem->setCurrentIndex(index); connect(ui->comboBoxPMSystem, QOverload::of(&QComboBox::currentIndexChanged), this, &TMainWindow::SavePMSystem); InitSearch(); ui->plainTextEditNotes->setPlainText(m_m->Notes()); connect(ui->plainTextEditNotes, &QPlainTextEdit::textChanged, this, &TMainWindow::SaveNotes); ui->actionAddCustom->setEnabled(true); ui->actionAddKnown->setEnabled(true); ui->actionAddSeparator->setEnabled(true); ui->actionImportFromPattern->setEnabled(true); ui->actionSaveAs->setEnabled(true); ui->lineEditName->setValidator( new QRegularExpressionValidator(QRegularExpression(QStringLiteral("^$|") + NameRegExp()), this)); connect(ui->toolButtonRemove, &QToolButton::clicked, this, &TMainWindow::Remove); connect(ui->toolButtonTop, &QToolButton::clicked, this, &TMainWindow::MoveTop); connect(ui->toolButtonUp, &QToolButton::clicked, this, &TMainWindow::MoveUp); connect(ui->toolButtonDown, &QToolButton::clicked, this, &TMainWindow::MoveDown); connect(ui->toolButtonBottom, &QToolButton::clicked, this, &TMainWindow::MoveBottom); connect(ui->lineEditName, &QLineEdit::textEdited, this, &TMainWindow::SaveMName); connect(ui->plainTextEditDescription, &QPlainTextEdit::textChanged, this, &TMainWindow::SaveMDescription); connect(ui->lineEditFullName, &QLineEdit::textEdited, this, &TMainWindow::SaveMFullName); connect(ui->pushButtonShowInExplorer, &QPushButton::clicked, this, [this]() { ShowInGraphicalShell(m_curFile); }); InitPatternUnits(); InitMeasurementUnits(); ui->comboBoxMUnits->blockSignals(true); ui->comboBoxMUnits->setCurrentIndex(-1); ui->comboBoxMUnits->blockSignals(false); connect(ui->comboBoxMUnits, QOverload::of(&QComboBox::currentIndexChanged), this, &TMainWindow::SaveMUnits); InitTable(); } //--------------------------------------------------------------------------------------------------------------------- void TMainWindow::InitMenu() { if (m_mType == MeasurementsType::Multisize) { // Measurements ui->actionUseFullCircumference->setVisible(true); ui->actionUseFullCircumference->setEnabled(true); ui->actionUseFullCircumference->setChecked(m_m->IsFullCircumference()); connect(ui->actionUseFullCircumference, &QAction::triggered, this, &TMainWindow::FullCircumferenceChanged); auto *separator = new QAction(this); separator->setSeparator(true); ui->menuMeasurements->insertAction(ui->actionUseFullCircumference, separator); const QList dimensions = m_m->Dimensions().values(); if (not dimensions.empty()) { ui->actionRestrictFirstDimension->setVisible(true); ui->actionRestrictFirstDimension->setEnabled(true); connect(ui->actionRestrictFirstDimension, &QAction::triggered, this, &TMainWindow::RestrictFirstDimesion); if (dimensions.size() > 1) { ui->actionRestrictSecondDimension->setVisible(true); ui->actionRestrictSecondDimension->setEnabled(true); connect(ui->actionRestrictSecondDimension, &QAction::triggered, this, &TMainWindow::RestrictSecondDimesion); if (dimensions.size() > 2) { ui->actionRestrictThirdDimension->setVisible(true); ui->actionRestrictThirdDimension->setEnabled(true); connect(ui->actionRestrictThirdDimension, &QAction::triggered, this, &TMainWindow::RestrictThirdDimesion); } } } ui->actionDimensionLabels->setVisible(true); ui->actionDimensionLabels->setEnabled(true); connect(ui->actionDimensionLabels, &QAction::triggered, this, &TMainWindow::EditDimensionLabels); ui->actionDimensionCustomNames->setVisible(true); ui->actionDimensionCustomNames->setEnabled(true); connect(ui->actionDimensionCustomNames, &QAction::triggered, this, &TMainWindow::DimensionCustomNames); // File ui->actionExportToIndividual->setVisible(true); ui->actionExportToIndividual->setEnabled(true); connect(ui->actionExportToIndividual, &QAction::triggered, this, &TMainWindow::ExportToIndividual); } } //--------------------------------------------------------------------------------------------------------------------- void TMainWindow::InitDimensionsBaseValue() { const QList dimensions = m_m->Dimensions().values(); const QString unit = UnitsToStr(m_m->Units(), true); const bool fc = m_m->IsFullCircumference(); auto DimensionsBaseValue = [this, dimensions, unit, fc](int index, QLabel *name, QLabel *base) { SCASSERT(name != nullptr) SCASSERT(base != nullptr) if (dimensions.size() > index) { const MeasurementDimension_p &dimension = dimensions.at(index); name->setText(dimension->Name() + QChar(':')); name->setToolTip(VAbstartMeasurementDimension::DimensionToolTip(dimension, m_m->IsFullCircumference())); DimesionLabels labels = dimension->Labels(); if (VFuzzyContains(labels, dimension->BaseValue()) && not VFuzzyValue(labels, dimension->BaseValue()).isEmpty()) { base->setText(VFuzzyValue(labels, dimension->BaseValue())); } else { if (dimension->IsBodyMeasurement() || dimension->Type() == MeasurementDimension::X) { if (dimension->Type() != MeasurementDimension::X && fc) { base->setText(QStringLiteral("%1 %2").arg(dimension->BaseValue() * 2).arg(unit)); } else { base->setText(QStringLiteral("%1 %2").arg(dimension->BaseValue()).arg(unit)); } } else { base->setText(QString::number(dimension->BaseValue())); } } } }; DimensionsBaseValue(0, ui->labelDimensionA, ui->labelDimensionABase); DimensionsBaseValue(1, ui->labelDimensionB, ui->labelDimensionBBase); DimensionsBaseValue(2, ui->labelDimensionC, ui->labelDimensionCBase); } //--------------------------------------------------------------------------------------------------------------------- void TMainWindow::InitDimensionGradation(int index, const MeasurementDimension_p &dimension, QComboBox *control) { SCASSERT(control != nullptr) const bool fc = m_m->IsFullCircumference(); const QString unit = UnitsToStr(m_m->Units(), true); qreal current = -1; if (control->currentIndex() != -1) { current = control->currentData().toDouble(); } control->blockSignals(true); control->clear(); const QVector bases = DimensionRestrictedValues(index, dimension); const DimesionLabels labels = dimension->Labels(); if (dimension->Type() == MeasurementDimension::X) { InitDimensionXItems(bases, labels, control, unit); } else if (dimension->Type() == MeasurementDimension::Y || dimension->Type() == MeasurementDimension::W || dimension->Type() == MeasurementDimension::Z) { InitDimensionYWZItems(bases, labels, control, unit, dimension->IsBodyMeasurement(), fc); } // Calculate the width of the largest item using QFontMetrics QFontMetrics fontMetrics(control->font()); int maxWidth = 0; for (int i = 0; i < control->count(); ++i) { int itemWidth = TextWidth(fontMetrics, control->itemText(i)); if (itemWidth > maxWidth) { maxWidth = itemWidth; } } // Set the minimum width of the view to the largest item width control->view()->setMinimumWidth(maxWidth); // after initialization the current index is 0. The signal for changing the index will not be triggered if not make // it invalid first control->setCurrentIndex(-1); int i = control->findData(current); if (i != -1) { control->setCurrentIndex(i); control->blockSignals(false); } else { control->blockSignals(false); control->setCurrentIndex(0); } } //--------------------------------------------------------------------------------------------------------------------- void TMainWindow::InitDimensionControls() { const QList dimensions = m_m->Dimensions().values(); auto InitControl = [this, dimensions](int index, QLabel *&name, QComboBox *&control) { if (dimensions.size() > index) { const MeasurementDimension_p &dimension = dimensions.at(index); if (name == nullptr) { name = new QLabel(dimension->Name() + QChar(':')); } else { name->setText(dimension->Name() + QChar(':')); } name->setToolTip(VAbstartMeasurementDimension::DimensionToolTip(dimension, m_m->IsFullCircumference())); if (control == nullptr) { control = new QComboBox; control->setSizeAdjustPolicy(QComboBox::AdjustToContents); } InitDimensionGradation(index, dimension, control); } }; InitControl(0, m_labelGradationDimensionA, m_gradationDimensionA); InitControl(1, m_labelGradationDimensionB, m_gradationDimensionB); InitControl(2, m_labelGradationDimensionC, m_gradationDimensionC); } //--------------------------------------------------------------------------------------------------------------------- void TMainWindow::InitDimesionShifts() { const QList dimensions = m_m->Dimensions().values(); auto InitShift = [this, dimensions](int index, QLabel *name) { if (dimensions.size() > index) { const MeasurementDimension_p &dimension = dimensions.at(index); name->setText(tr("Shift (%1):").arg(dimension->Name())); name->setToolTip(VAbstartMeasurementDimension::DimensionToolTip(dimension, m_m->IsFullCircumference())); } }; InitShift(0, ui->labelShiftA); InitShift(1, ui->labelShiftB); InitShift(2, ui->labelShiftC); } //--------------------------------------------------------------------------------------------------------------------- void TMainWindow::InitTable() { if (m_mType == MeasurementsType::Multisize) { ui->tableWidget->setColumnHidden(ColumnFormula, true); // formula RetranslateTableHeaders(); } else { ui->tableWidget->setColumnHidden(ColumnBaseValue, true); // base value ui->tableWidget->setColumnHidden(ColumnShiftA, true); ui->tableWidget->setColumnHidden(ColumnShiftB, true); ui->tableWidget->setColumnHidden(ColumnShiftC, true); ui->tableWidget->setColumnHidden(ColumnCorrection, true); } connect(ui->tableWidget, &QTableWidget::itemSelectionChanged, this, &TMainWindow::ShowMData); ShowUnits(); ui->tableWidget->resizeColumnsToContents(); ui->tableWidget->resizeRowsToContents(); ui->tableWidget->horizontalHeader()->setStretchLastSection(true); } //--------------------------------------------------------------------------------------------------------------------- void TMainWindow::ShowUnits() { const QString unit = UnitsToStr(m_mUnit); ShowHeaderUnits(ui->tableWidget, ColumnCalcValue, UnitsToStr(m_pUnit)); // calculated value ShowHeaderUnits(ui->tableWidget, ColumnFormula, unit); // formula ShowHeaderUnits(ui->tableWidget, ColumnBaseValue, unit); // base value ShowHeaderUnits(ui->tableWidget, ColumnShiftA, unit); ShowHeaderUnits(ui->tableWidget, ColumnShiftB, unit); ShowHeaderUnits(ui->tableWidget, ColumnShiftC, unit); ShowHeaderUnits(ui->tableWidget, ColumnCorrection, unit); } //--------------------------------------------------------------------------------------------------------------------- void TMainWindow::ShowHeaderUnits(QTableWidget *table, int column, const QString &unit) { SCASSERT(table != nullptr) QString header = table->horizontalHeaderItem(column)->text(); const auto index = header.indexOf(QLatin1String("(")); if (index != -1) { header.remove(index - 1, 100); } const QString unitHeader = QStringLiteral("%1 (%2)").arg(header, unit); table->horizontalHeaderItem(column)->setText(unitHeader); } //--------------------------------------------------------------------------------------------------------------------- void TMainWindow::MeasurementsWereSaved(bool saved) { setWindowModified(!saved); not m_mIsReadOnly ? ui->actionSave->setEnabled(!saved) : ui->actionSave->setEnabled(false); } //--------------------------------------------------------------------------------------------------------------------- void TMainWindow::SetCurrentFile(const QString &fileName) { m_curFile = fileName; if (m_curFile.isEmpty()) { ui->lineEditPathToFile->setText(QChar('<') + tr("Empty") + QChar('>')); ui->lineEditPathToFile->setToolTip(tr("File was not saved yet.")); ui->lineEditPathToFile->setCursorPosition(0); ui->pushButtonShowInExplorer->setEnabled(false); } else { ui->lineEditPathToFile->setText(QDir::toNativeSeparators(m_curFile)); ui->lineEditPathToFile->setToolTip(QDir::toNativeSeparators(m_curFile)); ui->lineEditPathToFile->setCursorPosition(0); ui->pushButtonShowInExplorer->setEnabled(true); auto *settings = MApplication::VApp()->TapeSettings(); QStringList files = settings->GetRecentFileList(); files.removeAll(fileName); files.prepend(fileName); while (files.size() > MaxRecentFiles) { files.removeLast(); } settings->SetRecentFileList(files); UpdateRecentFileActions(); } UpdateWindowTitle(); } //--------------------------------------------------------------------------------------------------------------------- auto TMainWindow::SaveMeasurements(const QString &fileName, QString &error) -> bool { const bool result = m_m->SaveDocument(fileName, error); if (result) { SetCurrentFile(fileName); MeasurementsWereSaved(result); } return result; } //--------------------------------------------------------------------------------------------------------------------- auto TMainWindow::MaybeSave() -> bool { if (this->isWindowModified()) { if (m_curFile.isEmpty() && ui->tableWidget->rowCount() == 0) { return true; // Don't ask if file was created without modifications. } QScopedPointer messageBox(new QMessageBox( tr("Unsaved changes"), tr("Measurements have been modified. 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, m_curFile.isEmpty() || m_mIsReadOnly ? 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 (m_mIsReadOnly) { return FileSaveAs(); } else { return FileSave(); } case QMessageBox::No: return true; case QMessageBox::Cancel: return false; default: break; } } return true; } //--------------------------------------------------------------------------------------------------------------------- auto TMainWindow::AddCell(const QString &text, int row, int column, int aligment, bool ok) -> QTableWidgetItem * { auto *item = new QTableWidgetItem(text); item->setTextAlignment(aligment); item->setToolTip(text); // set the item non-editable (view only), and non-selectable Qt::ItemFlags flags = item->flags(); flags &= ~(Qt::ItemIsEditable); // reset/clear the flag item->setFlags(flags); if (not ok) { QBrush brush = item->foreground(); brush.setColor(Qt::red); item->setForeground(brush); } ui->tableWidget->setItem(row, column, item); return item; } //--------------------------------------------------------------------------------------------------------------------- auto TMainWindow::AddSeparatorCell(const QString &text, int row, int column, int aligment, bool ok) -> QTableWidgetItem * { QTableWidgetItem *item = AddCell(text, row, column, aligment, ok); QFont itemFont = item->font(); itemFont.setBold(true); itemFont.setItalic(true); item->setFont(itemFont); return item; } //--------------------------------------------------------------------------------------------------------------------- void TMainWindow::RefreshData(bool freshCall) { QGuiApplication::setOverrideCursor(QCursor(Qt::WaitCursor)); m_data->ClearUniqueNames(); m_data->ClearVariables({VarType::Measurement, VarType::MeasurementSeparator}); m_m->ReadMeasurements(m_currentDimensionA, m_currentDimensionB, m_currentDimensionC); RefreshTable(freshCall); QGuiApplication::restoreOverrideCursor(); } //--------------------------------------------------------------------------------------------------------------------- void TMainWindow::RefreshTable(bool freshCall) { ui->tableWidget->blockSignals(true); ui->tableWidget->clearContents(); ShowUnits(); const QMap> orderedTable = OrderedMeasurments(); qint32 currentRow = -1; ui->tableWidget->setRowCount(static_cast(orderedTable.size())); for (auto iMap = orderedTable.constBegin(); iMap != orderedTable.constEnd(); ++iMap) { const QSharedPointer &meash = iMap.value(); currentRow++; if (meash->GetType() == VarType::Measurement) { RefreshMeasurementData(meash, currentRow); } else if (meash->GetType() == VarType::MeasurementSeparator) { QTableWidgetItem *item = AddSeparatorCell(VAbstractApplication::VApp()->TrVars()->MToUser(meash->GetName()), currentRow, ColumnName, Qt::AlignVCenter); // name item->setData(Qt::UserRole, meash->GetName()); AddCell(meash->GetDescription(), currentRow, ColumnFullName, Qt::AlignVCenter); // description if (m_mType == MeasurementsType::Individual) { ui->tableWidget->setSpan(currentRow, 1, 1, 3); } else if (m_mType == MeasurementsType::Multisize) { ui->tableWidget->setSpan(currentRow, 1, 1, 8); } } } Q_UNUSED(freshCall) // if (freshCall) // { // ui->tableWidget->resizeColumnsToContents(); // ui->tableWidget->resizeRowsToContents(); // } // ui->tableWidget->horizontalHeader()->setStretchLastSection(true); ui->tableWidget->blockSignals(false); ui->actionExportToCSV->setEnabled(ui->tableWidget->rowCount() > 0); } //--------------------------------------------------------------------------------------------------------------------- void TMainWindow::RefreshMeasurementData(const QSharedPointer &meash, qint32 currentRow) { if (m_mType == MeasurementsType::Individual) { QTableWidgetItem *item = AddCell(VAbstractApplication::VApp()->TrVars()->MToUser(meash->GetName()), currentRow, ColumnName, Qt::AlignVCenter); // name item->setData(Qt::UserRole, meash->GetName()); if (meash->IsCustom()) { AddCell(meash->GetGuiText(), currentRow, ColumnFullName, Qt::AlignVCenter); } else { AddCell(VAbstractApplication::VApp()->TrVars()->GuiText(meash->GetName()), currentRow, ColumnFullName, Qt::AlignVCenter); } QString calculatedValue; if (meash->IsSpecialUnits()) { calculatedValue = locale().toString(*meash->GetValue()) + degreeSymbol; } else { calculatedValue = locale().toString(UnitConvertor(*meash->GetValue(), m_mUnit, m_pUnit)); } AddCell(calculatedValue, currentRow, ColumnCalcValue, Qt::AlignHCenter | Qt::AlignVCenter, meash->IsFormulaOk()); // calculated value QString formula = VTranslateVars::TryFormulaToUser(meash->GetFormula(), VAbstractApplication::VApp()->Settings()->GetOsSeparator()); AddCell(formula, currentRow, ColumnFormula, Qt::AlignVCenter); // formula } else { QTableWidgetItem *item = AddCell(VAbstractApplication::VApp()->TrVars()->MToUser(meash->GetName()), currentRow, 0, Qt::AlignVCenter); // name item->setData(Qt::UserRole, meash->GetName()); if (meash->IsCustom()) { AddCell(meash->GetGuiText(), currentRow, ColumnFullName, Qt::AlignVCenter); } else { AddCell(VAbstractApplication::VApp()->TrVars()->GuiText(meash->GetName()), currentRow, ColumnFullName, Qt::AlignVCenter); } QString calculatedValue; if (meash->IsSpecialUnits()) { const qreal value = *m_data->DataVariables()->value(meash->GetName())->GetValue(); calculatedValue = locale().toString(value) + degreeSymbol; } else { const qreal value = UnitConvertor(*m_data->DataVariables()->value(meash->GetName())->GetValue(), m_mUnit, m_pUnit); calculatedValue = locale().toString(value); } AddCell(calculatedValue, currentRow, ColumnCalcValue, Qt::AlignHCenter | Qt::AlignVCenter, meash->IsFormulaOk()); // calculated value AddCell(locale().toString(meash->GetBase()), currentRow, ColumnBaseValue, Qt::AlignHCenter | Qt::AlignVCenter); // base value AddCell(locale().toString(meash->GetShiftA()), currentRow, ColumnShiftA, Qt::AlignHCenter | Qt::AlignVCenter); AddCell(locale().toString(meash->GetShiftB()), currentRow, ColumnShiftB, Qt::AlignHCenter | Qt::AlignVCenter); AddCell(locale().toString(meash->GetShiftC()), currentRow, ColumnShiftC, Qt::AlignHCenter | Qt::AlignVCenter); AddCell(locale().toString(meash->GetCorrection(m_currentDimensionA, m_currentDimensionB, m_currentDimensionC)), currentRow, ColumnCorrection, Qt::AlignHCenter | Qt::AlignVCenter); } } //--------------------------------------------------------------------------------------------------------------------- auto TMainWindow::GetCustomName() const -> QString { qint32 num = 1; QString name; do { name = CustomMSign + VAbstractApplication::VApp()->TrVars()->InternalVarToUser(measurement_) + QString::number(num); num++; } while (not m_data->IsUnique(name)); return name; } //--------------------------------------------------------------------------------------------------------------------- void TMainWindow::Controls() { if (ui->tableWidget->rowCount() > 0) { ui->toolButtonRemove->setEnabled(true); } else { ui->toolButtonRemove->setEnabled(false); } if (ui->tableWidget->rowCount() >= 2) { if (ui->tableWidget->currentRow() == 0) { ui->toolButtonTop->setEnabled(false); ui->toolButtonUp->setEnabled(false); ui->toolButtonDown->setEnabled(true); ui->toolButtonBottom->setEnabled(true); } else if (ui->tableWidget->currentRow() == ui->tableWidget->rowCount() - 1) { ui->toolButtonTop->setEnabled(true); ui->toolButtonUp->setEnabled(true); ui->toolButtonDown->setEnabled(false); ui->toolButtonBottom->setEnabled(false); } else { ui->toolButtonTop->setEnabled(true); ui->toolButtonUp->setEnabled(true); ui->toolButtonDown->setEnabled(true); ui->toolButtonBottom->setEnabled(true); } } else { ui->toolButtonTop->setEnabled(false); ui->toolButtonUp->setEnabled(false); ui->toolButtonDown->setEnabled(false); ui->toolButtonBottom->setEnabled(false); } } //--------------------------------------------------------------------------------------------------------------------- void TMainWindow::MFields(bool enabled) { ui->lineEditName->setEnabled(enabled); ui->plainTextEditDescription->setEnabled(enabled); ui->lineEditFullName->setEnabled(enabled); ui->comboBoxMUnits->setEnabled(enabled); if (m_mType == MeasurementsType::Multisize) { ui->doubleSpinBoxBaseValue->setEnabled(enabled); ui->doubleSpinBoxShiftA->setEnabled(enabled); ui->doubleSpinBoxShiftB->setEnabled(enabled); ui->doubleSpinBoxShiftC->setEnabled(enabled); ui->doubleSpinBoxCorrection->setEnabled(enabled); } else { ui->plainTextEditFormula->setEnabled(enabled); ui->pushButtonGrow->setEnabled(enabled); ui->toolButtonExpr->setEnabled(enabled); ui->comboBoxDimension->setEnabled(enabled); } ui->lineEditFind->setEnabled(enabled); if (enabled && not ui->lineEditFind->text().isEmpty()) { ui->toolButtonFindPrevious->setEnabled(enabled); ui->toolButtonFindNext->setEnabled(enabled); } else { ui->toolButtonFindPrevious->setEnabled(false); ui->toolButtonFindNext->setEnabled(false); } Controls(); } //--------------------------------------------------------------------------------------------------------------------- void TMainWindow::UpdateWindowTitle() { QString showName; bool isFileWritable = true; if (not m_curFile.isEmpty()) { // #ifdef Q_OS_WIN32 // qt_ntfs_permission_lookup++; // turn checking on // #endif /*Q_OS_WIN32*/ isFileWritable = QFileInfo(m_curFile).isWritable(); // #ifdef Q_OS_WIN32 // qt_ntfs_permission_lookup--; // turn it off again // #endif /*Q_OS_WIN32*/ showName = StrippedName(m_curFile); } else { auto index = MApplication::VApp()->MainWindows().indexOf(this); if (index != -1) { showName = tr("untitled %1").arg(index + 1); } else { showName = tr("untitled"); } m_mType == MeasurementsType::Multisize ? showName += QLatin1String(".vst") : showName += QLatin1String(".vit"); } showName += QLatin1String("[*]"); if (m_mIsReadOnly || not isFileWritable) { showName += QStringLiteral(" (") + tr("read only") + QChar(')'); } setWindowTitle(showName); setWindowFilePath(m_curFile); #if defined(Q_OS_MAC) static QIcon fileIcon = QIcon(QCoreApplication::applicationDirPath() + QLatin1String("/../Resources/measurements.icns")); QIcon icon; if (not m_curFile.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) } //--------------------------------------------------------------------------------------------------------------------- auto TMainWindow::ClearCustomName(const QString &name) -> QString { QString clear = name; const auto index = clear.indexOf(CustomMSign); if (index == 0) { clear.remove(0, 1); } return clear; } //--------------------------------------------------------------------------------------------------------------------- auto TMainWindow::EvalFormula(const QString &formula, bool fromUser, VContainer *data, QLabel *label, bool specialUnits) -> bool { const QString postfix = specialUnits ? degreeSymbol : UnitsToStr(m_pUnit); if (formula.isEmpty()) { label->setText(tr("Error") + QStringLiteral(" (") + postfix + QStringLiteral("). ") + tr("Empty field.")); label->setToolTip(tr("Empty field")); return false; } try { // Replace line return character with spaces for calc if exist QString f; if (fromUser) { f = VAbstractApplication::VApp()->TrVars()->FormulaFromUser( formula, VAbstractApplication::VApp()->Settings()->GetOsSeparator()); } else { f = formula; } QScopedPointer cal(new Calculator()); qreal result = cal->EvalFormula(data->DataVariables(), f); if (qIsInf(result) || qIsNaN(result)) { label->setText(tr("Error") + " (" + postfix + ")."); label->setToolTip(tr("Invalid result. Value is infinite or NaN. Please, check your calculations.")); return false; } if (not specialUnits) { result = UnitConvertor(result, m_mUnit, m_pUnit); } label->setText(VAbstractApplication::VApp()->LocaleToString(result) + QChar(QChar::Space) + postfix); label->setToolTip(tr("Value")); return true; } catch (qmu::QmuParserError &e) { label->setText(tr("Error") + " (" + postfix + "). " + tr("Parser error: %1").arg(e.GetMsg())); label->setToolTip(tr("Parser error: %1").arg(e.GetMsg())); return false; } } //--------------------------------------------------------------------------------------------------------------------- void TMainWindow::Open(const QString &pathTo, const QString &filter) { const QString mPath = QFileDialog::getOpenFileName(this, tr("Open file"), pathTo, filter, nullptr, VAbstractApplication::VApp()->NativeFileDialog()); if (not mPath.isEmpty()) { if (m_m == nullptr) { LoadFile(mPath); } else { MApplication::VApp()->NewMainWindow()->LoadFile(mPath); } } } //--------------------------------------------------------------------------------------------------------------------- void TMainWindow::UpdatePadlock(bool ro) { ui->actionReadOnly->setChecked(ro); ui->actionReadOnly->setIcon(ro ? QIcon("://tapeicon/24x24/padlock_locked.png") : QIcon("://tapeicon/24x24/padlock_opened.png")); ui->actionReadOnly->setDisabled(m_mIsReadOnly); } //--------------------------------------------------------------------------------------------------------------------- void TMainWindow::MeasurementGUI() { if (const QTableWidgetItem *nameField = ui->tableWidget->item(ui->tableWidget->currentRow(), ColumnName)) { const bool isCustom = not(nameField->text().indexOf(CustomMSign) == 0); ui->lineEditName->setReadOnly(isCustom); ui->plainTextEditDescription->setReadOnly(isCustom); ui->lineEditFullName->setReadOnly(isCustom); // Need to block signals for QLineEdit in readonly mode because it still emits // QLineEdit::editingFinished signal. ui->lineEditName->blockSignals(isCustom); ui->lineEditFullName->blockSignals(isCustom); Controls(); // Buttons remove, up, down } } //--------------------------------------------------------------------------------------------------------------------- void TMainWindow::ReadSettings() { const VTapeSettings *settings = MApplication::VApp()->TapeSettings(); if (settings->status() == QSettings::NoError) { restoreGeometry(settings->GetGeometry()); restoreState(settings->GetToolbarsState(), static_cast(AppVersion())); // Text under tool buton icon ToolBarStyles(); // Stack limit // VAbstractApplication::VApp()->getUndoStack()->setUndoLimit(settings->GetUndoCount()); } else { qWarning() << tr("Cannot read settings from a malformed .INI file."); } } //--------------------------------------------------------------------------------------------------------------------- void TMainWindow::WriteSettings() { VTapeSettings *settings = MApplication::VApp()->TapeSettings(); settings->SetGeometry(saveGeometry()); settings->SetToolbarsState(saveState(static_cast(AppVersion()))); settings->SetTapeSearchOptionMatchCase(m_search->IsMatchCase()); settings->SetTapeSearchOptionWholeWord(m_search->IsMatchWord()); settings->SetTapeSearchOptionRegexp(m_search->IsMatchRegexp()); settings->SetTapeSearchOptionUseUnicodeProperties(m_search->IsUseUnicodePreperties()); settings->sync(); if (settings->status() == QSettings::AccessError) { qWarning() << tr("Cannot save settings. Access denied."); } } //--------------------------------------------------------------------------------------------------------------------- auto TMainWindow::FilterMeasurements(const QStringList &mNew, const QStringList &mFilter) -> QStringList { return ConvertToList(ConvertToSet(mNew).subtract(ConvertToSet(mFilter))); } //--------------------------------------------------------------------------------------------------------------------- void TMainWindow::UpdatePatternUnit() { const int row = ui->tableWidget->currentRow(); if (row == -1) { return; } RefreshTable(); m_search->RefreshList(ui->lineEditFind->text()); ui->tableWidget->selectRow(row); } //--------------------------------------------------------------------------------------------------------------------- auto TMainWindow::LoadFromExistingFile(const QString &path) -> bool { if (m_m != nullptr) { return MApplication::VApp()->NewMainWindow()->LoadFile(path); } if (not QFileInfo::exists(path)) { qCCritical(tMainWindow, "%s", qUtf8Printable(tr("File '%1' doesn't exist!").arg(path))); if (MApplication::VApp()->IsTestMode()) { QCoreApplication::exit(V_EX_NOINPUT); } return false; } // Check if file already opened const QList list = MApplication::VApp()->MainWindows(); auto w = std::find_if(list.begin(), list.end(), [path](TMainWindow *window) { return window->CurrentFile() == path; }); if (w != list.end()) { (*w)->activateWindow(); close(); return false; } VlpCreateLock(m_lock, path); if (not m_lock->IsLocked()) { if (not IgnoreLocking(m_lock->GetLockError(), path, MApplication::VApp()->IsAppInGUIMode())) { return false; } } try { m_data = new VContainer(VAbstractApplication::VApp()->TrVars(), &m_mUnit, VContainer::UniqueNamespace()); m_m = new VMeasurements(m_data); m_m->setXMLContent(path); m_mType = m_m->Type(); if (m_mType == MeasurementsType::Unknown) { throw VException(tr("File has unknown format.")); } QScopedPointer converter( (m_mType == MeasurementsType::Individual) ? static_cast(new VVITConverter(path)) : static_cast(new VVSTConverter(path))); m_curFileFormatVersion = converter->GetCurrentFormatVersion(); m_curFileFormatVersionStr = converter->GetFormatVersionStr(); m_m->setXMLContent(converter->Convert()); // Read again after conversion if (not m_m->IsDefinedKnownNamesValid()) { throw VException(tr("File contains invalid known measurement(s).")); } m_mUnit = m_m->Units(); m_pUnit = m_mUnit; m_currentDimensionA = m_m->DimensionABase(); m_currentDimensionB = m_m->DimensionBBase(); m_currentDimensionC = m_m->DimensionCBase(); ui->labelToolTip->setVisible(false); ui->tabWidget->setVisible(true); InitWindow(); m_m->ClearForExport(); const bool freshCall = true; RefreshData(freshCall); if (ui->tableWidget->rowCount() > 0) { ui->tableWidget->selectRow(0); } m_lock.reset(); // Now we can unlock the file m_mIsReadOnly = m_m->IsReadOnly(); UpdatePadlock(m_mIsReadOnly); MeasurementGUI(); } catch (VException &e) { qCCritical(tMainWindow, "%s\n\n%s\n\n%s", qUtf8Printable(tr("File error.")), qUtf8Printable(e.ErrorMessage()), qUtf8Printable(e.DetailedInformation())); ui->labelToolTip->setVisible(true); ui->tabWidget->setVisible(false); delete m_m; m_m = nullptr; delete m_data; m_data = nullptr; m_lock.reset(); if (MApplication::VApp()->IsTestMode()) { QCoreApplication::exit(V_EX_NOINPUT); } return false; } return true; } //--------------------------------------------------------------------------------------------------------------------- void TMainWindow::CreateWindowMenu(QMenu *menu) { SCASSERT(menu != nullptr) QAction *action = menu->addAction(tr("&New Window")); connect(action, &QAction::triggered, this, []() { MApplication::VApp()->NewMainWindow()->activateWindow(); }); action->setMenuRole(QAction::NoRole); menu->addSeparator(); const QList windows = MApplication::VApp()->MainWindows(); for (int i = 0; i < windows.count(); ++i) { TMainWindow *window = windows.at(i); QString title = QStringLiteral("%1. %2").arg(i + 1).arg(window->windowTitle()); const auto index = title.lastIndexOf(QLatin1String("[*]")); if (index != -1) { window->isWindowModified() ? title.replace(index, 3, QChar('*')) : title.replace(index, 3, QString()); } QAction *action = menu->addAction(title, this, &TMainWindow::ShowWindow); action->setData(i); action->setCheckable(true); action->setMenuRole(QAction::NoRole); if (window->isActiveWindow()) { action->setChecked(true); } } } //--------------------------------------------------------------------------------------------------------------------- void TMainWindow::HackDimensionBaseValue() { const QList dimensions = m_m->Dimensions().values(); auto DimensionsBaseValue = [this, dimensions](int index, QLabel *name, QLabel *base) { SCASSERT(name != nullptr) SCASSERT(base != nullptr) if (dimensions.size() <= index) { HackWidget(&name); HackWidget(&base); } }; DimensionsBaseValue(0, ui->labelDimensionA, ui->labelDimensionABase); DimensionsBaseValue(1, ui->labelDimensionB, ui->labelDimensionBBase); DimensionsBaseValue(2, ui->labelDimensionC, ui->labelDimensionCBase); } //--------------------------------------------------------------------------------------------------------------------- void TMainWindow::HackDimensionShifts() { const QList dimensions = m_m->Dimensions().values(); auto HackShift = [this, dimensions](int index, QLabel *&name, QDoubleSpinBox *&shift) { SCASSERT(name != nullptr) SCASSERT(shift != nullptr) if (dimensions.size() <= index) { HackWidget(&name); HackWidget(&shift); } }; HackShift(0, ui->labelShiftA, ui->doubleSpinBoxShiftA); HackShift(1, ui->labelShiftB, ui->doubleSpinBoxShiftB); HackShift(2, ui->labelShiftC, ui->doubleSpinBoxShiftC); } //--------------------------------------------------------------------------------------------------------------------- auto TMainWindow::CheckMName(const QString &name, const QSet &importedNames) const -> QString { if (name.isEmpty()) { throw VException(tr("Measurement name is empty.")); } if (importedNames.contains(name)) { throw VException(tr("Imported file must not contain the same name twice.")); } if (name.indexOf(CustomMSign) == 0) { QRegularExpression rx(NameRegExp()); if (not rx.match(name).hasMatch()) { throw VException(tr("Measurement '%1' doesn't match regex pattern.").arg(name)); } if (not m_data->IsUnique(name)) { throw VException(tr("Measurement '%1' already used in the file.").arg(name)); } } else { if (not ConvertToSet(AllGroupNames()).contains(name)) { throw VException(tr("Measurement '%1' is not one of the known measurements.").arg(name)); } if (not m_data->IsUnique(name)) { throw VException(tr("Measurement '%1' already used in file.").arg(name)); } } return name; } //--------------------------------------------------------------------------------------------------------------------- void TMainWindow::ShowError(const QString &text) { QMessageBox messageBox(this); messageBox.setIcon(QMessageBox::Critical); messageBox.setText(text); messageBox.setStandardButtons(QMessageBox::Ok); messageBox.setDefaultButton(QMessageBox::Ok); messageBox.exec(); } //--------------------------------------------------------------------------------------------------------------------- void TMainWindow::RefreshDataAfterImport() { const int currentRow = ui->tableWidget->currentRow(); m_search->AddRow(currentRow); RefreshData(); m_search->RefreshList(ui->lineEditFind->text()); ui->tableWidget->selectRow(currentRow); ui->actionExportToCSV->setEnabled(true); MeasurementsWereSaved(false); } //--------------------------------------------------------------------------------------------------------------------- void TMainWindow::ImportIndividualMeasurements(const QxtCsvModel &csv, const QVector &map, bool withHeader) { const int columns = csv.columnCount(); const int rows = csv.rowCount(); if (columns < 2) { ShowError(tr("Individual measurements require at least 2 columns.")); return; } QVector measurements; QSet importedNames; for (int i = 0; i < rows; ++i) { try { const int nameColumn = map.at(static_cast(IndividualMeasurementsColumns::Name)); const QString name = csv.text(i, nameColumn).simplified(); if (name.isEmpty()) { ShowError(tr("Error in row %1. The measurement name is empty.").arg(i)); continue; } IndividualMeasurement measurement; const QString mName = CheckMName(VAbstractApplication::VApp()->TrVars()->MFromUser(name), importedNames); importedNames.insert(mName); measurement.name = mName; const int valueColumn = map.at(static_cast(IndividualMeasurementsColumns::Value)); measurement.value = VTranslateVars::TryFormulaFromUser( csv.text(i, valueColumn), VAbstractApplication::VApp()->Settings()->GetOsSeparator()); SetIndividualMeasurementFullName(i, name, csv, map, measurement); SetIndividualMeasurementDescription(i, name, csv, map, measurement); measurements.append(measurement); } catch (VException &e) { int rowIndex = i + 1; if (withHeader) { ++rowIndex; } ShowError(tr("Error in row %1. %2").arg(rowIndex).arg(e.ErrorMessage())); return; } } for (const auto &im : qAsConst(measurements)) { m_m->AddEmpty(im.name, im.value); if (not im.fullName.isEmpty()) { m_m->SetMFullName(im.name, im.fullName); } if (not im.description.isEmpty()) { m_m->SetMDescription(im.name, im.description); } } RefreshDataAfterImport(); } //--------------------------------------------------------------------------------------------------------------------- void TMainWindow::ImportMultisizeMeasurements(const QxtCsvModel &csv, const QVector &map, bool withHeader) { const int columns = csv.columnCount(); if (columns < 4) { ShowError(tr("Multisize measurements require at least 4 columns.")); return; } QVector measurements; QSet importedNames; const QMap dimensions = m_m->Dimensions(); const int rows = csv.rowCount(); for (int i = 0; i < rows; ++i) { try { measurements.append(ImportMultisizeMeasurement(csv, i, map, dimensions.size(), importedNames)); } catch (VException &e) { int rowIndex = i + 1; if (withHeader) { ++rowIndex; } ShowError(tr("Error in row %1. %2").arg(rowIndex).arg(e.ErrorMessage())); return; } } for (const auto &mm : qAsConst(measurements)) { m_m->AddEmpty(mm.name); m_m->SetMBaseValue(mm.name, mm.base); m_m->SetMShiftA(mm.name, mm.shiftA); if (dimensions.size() > 1) { m_m->SetMShiftB(mm.name, mm.shiftB); } if (dimensions.size() > 2) { m_m->SetMShiftC(mm.name, mm.shiftC); } if (not mm.fullName.isEmpty()) { m_m->SetMFullName(mm.name, mm.fullName); } if (not mm.description.isEmpty()) { m_m->SetMDescription(mm.name, mm.description); } } RefreshDataAfterImport(); } //--------------------------------------------------------------------------------------------------------------------- auto TMainWindow::ImportMultisizeMeasurement(const QxtCsvModel &csv, int i, const QVector &map, vsizetype dimensionsCount, QSet &importedNames) -> TMainWindow::MultisizeMeasurement { const auto nameColumn = map.at(static_cast(MultisizeMeasurementsColumns::Name)); const QString name = csv.text(i, nameColumn).simplified(); if (name.isEmpty()) { throw VException(tr("The measurement name is empty.")); } MultisizeMeasurement measurement; const QString mName = CheckMName(VAbstractApplication::VApp()->TrVars()->MFromUser(name), importedNames); importedNames.insert(mName); measurement.name = mName; const auto baseValueColumn = map.at(static_cast(MultisizeMeasurementsColumns::BaseValue)); measurement.base = ConverToDouble(csv.text(i, baseValueColumn), tr("Cannot convert base value to double in column 2.")); const auto shiftAColumn = map.at(static_cast(MultisizeMeasurementsColumns::ShiftA)); measurement.shiftA = ConverToDouble(csv.text(i, shiftAColumn), tr("Cannot convert shift value to double in column %1.").arg(shiftAColumn)); if (dimensionsCount > 1) { const auto shiftBColumn = map.at(static_cast(MultisizeMeasurementsColumns::ShiftB)); measurement.shiftB = ConverToDouble(csv.text(i, shiftBColumn), tr("Cannot convert shift value to double in column %1.").arg(shiftBColumn)); } if (dimensionsCount > 2) { const auto shiftCColumn = map.at(static_cast(MultisizeMeasurementsColumns::ShiftC)); measurement.shiftC = ConverToDouble(csv.text(i, shiftCColumn), tr("Cannot convert shift value to double in column %1.").arg(shiftCColumn)); } const int columns = csv.columnCount(); const bool custom = name.startsWith(CustomMSign); if (columns > 4 && custom) { const auto fullNameColumn = map.at(static_cast(MultisizeMeasurementsColumns::FullName)); if (fullNameColumn >= 0) { measurement.fullName = csv.text(i, fullNameColumn).simplified(); } } if (columns > 5 && custom) { const auto descriptionColumn = map.at(static_cast(MultisizeMeasurementsColumns::Description)); if (descriptionColumn >= 0) { measurement.description = csv.text(i, descriptionColumn).simplified(); } } return measurement; } //--------------------------------------------------------------------------------------------------------------------- void TMainWindow::SetCurrentPatternUnit() { if (m_comboBoxUnits) { m_comboBoxUnits->blockSignals(true); const qint32 indexUnit = m_comboBoxUnits->findData(static_cast(m_pUnit)); if (indexUnit != -1) { m_comboBoxUnits->setCurrentIndex(indexUnit); } m_comboBoxUnits->blockSignals(false); } } //--------------------------------------------------------------------------------------------------------------------- void TMainWindow::ShowDimensionControls() { const QList dimensions = m_m->Dimensions().values(); auto ShowControl = [this, dimensions](int index, QLabel *name, QComboBox *control) { if (dimensions.size() > index) { SCASSERT(name != nullptr) SCASSERT(control != nullptr) ui->toolBarGradation->addWidget(name); ui->toolBarGradation->addWidget(control); } }; ShowControl(0, m_labelGradationDimensionA, m_gradationDimensionA); ShowControl(1, m_labelGradationDimensionB, m_gradationDimensionB); ShowControl(2, m_labelGradationDimensionC, m_gradationDimensionC); if (m_gradationDimensionA) { connect(m_gradationDimensionA, QOverload::of(&QComboBox::currentIndexChanged), this, &TMainWindow::DimensionABaseChanged); } if (m_gradationDimensionB) { connect(m_gradationDimensionB, QOverload::of(&QComboBox::currentIndexChanged), this, &TMainWindow::DimensionBBaseChanged); } if (m_gradationDimensionC) { connect(m_gradationDimensionC, QOverload::of(&QComboBox::currentIndexChanged), this, &TMainWindow::DimensionCBaseChanged); } } //--------------------------------------------------------------------------------------------------------------------- void TMainWindow::SetDimensionBases() { const QList dimensions = m_m->Dimensions().values(); if (not dimensions.empty()) { const MeasurementDimension_p &dimension = dimensions.at(0); m_currentDimensionA = dimension->BaseValue(); } if (dimensions.size() > 1) { const MeasurementDimension_p &dimension = dimensions.at(1); m_currentDimensionB = dimension->BaseValue(); } if (dimensions.size() > 2) { const MeasurementDimension_p &dimension = dimensions.at(2); m_currentDimensionC = dimension->BaseValue(); } auto SetBase = [dimensions](int index, QComboBox *control, qreal &value) { if (dimensions.size() > index) { SCASSERT(control != nullptr) const qint32 i = control->findData(value); if (i != -1) { control->setCurrentIndex(i); } else { value = control->currentData().toDouble(); } } }; SetBase(0, m_gradationDimensionA, m_currentDimensionA); SetBase(1, m_gradationDimensionB, m_currentDimensionB); SetBase(2, m_gradationDimensionC, m_currentDimensionC); } //--------------------------------------------------------------------------------------------------------------------- void TMainWindow::SetCurrentDimensionValues() { const QList dimensions = m_m->Dimensions().values(); auto SetDimensionValue = [dimensions](int index, qreal &value) { if (dimensions.size() > index) { const MeasurementDimension_p &dimension = dimensions.at(index); value = dimension->BaseValue(); } }; SetDimensionValue(0, m_currentDimensionA); SetDimensionValue(1, m_currentDimensionB); SetDimensionValue(2, m_currentDimensionC); } //--------------------------------------------------------------------------------------------------------------------- auto TMainWindow::DimensionRestrictedValues(int index, const MeasurementDimension_p &dimension) -> QVector { VDimensionRestriction restriction; if (index == 0) { restriction = m_m->Restriction(0); } else if (index == 1) { restriction = m_m->Restriction(m_currentDimensionA); } else { restriction = m_m->Restriction(m_currentDimensionA, m_currentDimensionB); } const QVector bases = dimension->ValidBases(); qreal min = VFuzzyIndexOf(bases, restriction.GetMin()) != -1 ? restriction.GetMin() : dimension->MinValue(); qreal max = VFuzzyIndexOf(bases, restriction.GetMax()) != -1 ? restriction.GetMax() : dimension->MaxValue(); if (min > max) { min = dimension->MinValue(); max = dimension->MaxValue(); } return VAbstartMeasurementDimension::ValidBases(min, max, dimension->Step(), restriction.GetExcludeValues()); } //--------------------------------------------------------------------------------------------------------------------- auto TMainWindow::OrderedMeasurments() const -> QMap> { const QMap> table = m_data->DataMeasurementsWithSeparators(); QMap> orderedTable; QMap>::const_iterator iterMap; for (iterMap = table.constBegin(); iterMap != table.constEnd(); ++iterMap) { const QSharedPointer &meash = iterMap.value(); orderedTable.insert(meash->Index(), meash); } return orderedTable; } //--------------------------------------------------------------------------------------------------------------------- void TMainWindow::InitIcons() { QString iconResource = QStringLiteral("icon"); ui->toolButtonExpr->setIcon(VTheme::GetIconResource(iconResource, QStringLiteral("24x24/fx.png"))); QString tapeIconResource = QStringLiteral("tapeicon"); ui->actionMeasurementDiagram->setIcon( VTheme::GetIconResource(tapeIconResource, QStringLiteral("24x24/mannequin.png"))); } //--------------------------------------------------------------------------------------------------------------------- void TMainWindow::SetDecimals() { switch (m_mUnit) { case Unit::Cm: ui->doubleSpinBoxBaseValue->setDecimals(2); ui->doubleSpinBoxBaseValue->setSingleStep(0.1); ui->doubleSpinBoxCorrection->setDecimals(2); ui->doubleSpinBoxCorrection->setSingleStep(0.1); ui->doubleSpinBoxShiftA->setDecimals(2); ui->doubleSpinBoxShiftA->setSingleStep(0.1); ui->doubleSpinBoxShiftB->setDecimals(2); ui->doubleSpinBoxShiftB->setSingleStep(0.1); ui->doubleSpinBoxShiftC->setDecimals(2); ui->doubleSpinBoxShiftC->setSingleStep(0.1); break; case Unit::Mm: ui->doubleSpinBoxBaseValue->setDecimals(1); ui->doubleSpinBoxBaseValue->setSingleStep(1); ui->doubleSpinBoxCorrection->setDecimals(1); ui->doubleSpinBoxCorrection->setSingleStep(1); ui->doubleSpinBoxShiftA->setDecimals(1); ui->doubleSpinBoxShiftA->setSingleStep(1); ui->doubleSpinBoxShiftB->setDecimals(1); ui->doubleSpinBoxShiftB->setSingleStep(1); ui->doubleSpinBoxShiftC->setDecimals(1); ui->doubleSpinBoxShiftC->setSingleStep(1); break; case Unit::Inch: ui->doubleSpinBoxBaseValue->setDecimals(5); ui->doubleSpinBoxBaseValue->setSingleStep(0.00001); ui->doubleSpinBoxCorrection->setDecimals(5); ui->doubleSpinBoxCorrection->setSingleStep(0.00001); ui->doubleSpinBoxShiftA->setDecimals(5); ui->doubleSpinBoxShiftA->setSingleStep(0.00001); ui->doubleSpinBoxShiftB->setDecimals(5); ui->doubleSpinBoxShiftB->setSingleStep(0.00001); ui->doubleSpinBoxShiftC->setDecimals(5); ui->doubleSpinBoxShiftC->setSingleStep(0.00001); break; default: break; } } //--------------------------------------------------------------------------------------------------------------------- void TMainWindow::InitPatternUnits() { m_labelPatternUnit = new QLabel(tr("Pattern unit:")); ui->toolBarGradation->addWidget(m_labelPatternUnit); m_comboBoxUnits = new QComboBox(this); InitComboBoxUnits(); SetCurrentPatternUnit(); connect(m_comboBoxUnits, QOverload::of(&QComboBox::currentIndexChanged), this, [this](int index) { m_pUnit = static_cast(m_comboBoxUnits->itemData(index).toInt()); UpdatePatternUnit(); }); ui->toolBarGradation->addWidget(m_comboBoxUnits); } //--------------------------------------------------------------------------------------------------------------------- void TMainWindow::InitComboBoxUnits() { SCASSERT(m_comboBoxUnits != nullptr) m_comboBoxUnits->addItem(UnitsToStr(Unit::Cm, true), QVariant(static_cast(Unit::Cm))); m_comboBoxUnits->addItem(UnitsToStr(Unit::Mm, true), QVariant(static_cast(Unit::Mm))); m_comboBoxUnits->addItem(UnitsToStr(Unit::Inch, true), QVariant(static_cast(Unit::Inch))); } //--------------------------------------------------------------------------------------------------------------------- void TMainWindow::InitMeasurementUnits() { if (not m_m) { return; } ui->comboBoxMUnits->blockSignals(true); int current = -1; if (ui->comboBoxMUnits->currentIndex() != -1) { current = ui->comboBoxMUnits->currentData().toInt(); } QString units; switch (m_m->Units()) { case Unit::Mm: units = tr("Millimeters"); break; case Unit::Inch: units = tr("Inches"); break; case Unit::Cm: units = tr("Centimeters"); break; default: units = QStringLiteral(""); break; } ui->comboBoxMUnits->clear(); ui->comboBoxMUnits->addItem(units, QVariant(static_cast(MUnits::Table))); ui->comboBoxMUnits->addItem(tr("Degrees"), QVariant(static_cast(MUnits::Degrees))); int i = ui->comboBoxMUnits->findData(current); if (i != -1) { ui->comboBoxMUnits->setCurrentIndex(i); } ui->comboBoxMUnits->blockSignals(false); } //--------------------------------------------------------------------------------------------------------------------- void TMainWindow::InitGender(QComboBox *gender) { SCASSERT(gender != nullptr) gender->addItem(tr("unknown", "gender"), QVariant(static_cast(GenderType::Unknown))); gender->addItem(tr("male", "gender"), QVariant(static_cast(GenderType::Male))); gender->addItem(tr("female", "gender"), QVariant(static_cast(GenderType::Female))); } //--------------------------------------------------------------------------------------------------------------------- void TMainWindow::InitMeasurementDimension() { ui->comboBoxDimension->blockSignals(true); int current = -1; if (ui->comboBoxDimension->currentIndex() != -1) { current = ui->comboBoxDimension->currentData().toInt(); } ui->comboBoxDimension->clear(); ui->comboBoxDimension->addItem(VMeasurements::IMDName(IMD::N), QVariant(static_cast(IMD::N))); ui->comboBoxDimension->addItem(VMeasurements::IMDName(IMD::X), QVariant(static_cast(IMD::X))); ui->comboBoxDimension->addItem(VMeasurements::IMDName(IMD::Y), QVariant(static_cast(IMD::Y))); ui->comboBoxDimension->addItem(VMeasurements::IMDName(IMD::W), QVariant(static_cast(IMD::W))); ui->comboBoxDimension->addItem(VMeasurements::IMDName(IMD::Z), QVariant(static_cast(IMD::Z))); int i = ui->comboBoxDimension->findData(current); if (i != -1) { ui->comboBoxDimension->setCurrentIndex(i); } ui->comboBoxDimension->blockSignals(false); } //--------------------------------------------------------------------------------------------------------------------- void TMainWindow::InitSearch() { VTapeSettings *settings = MApplication::VApp()->TapeSettings(); m_search->SetUseUnicodePreperties(settings->GetTapeSearchOptionUseUnicodeProperties()); m_search->SetMatchWord(settings->GetTapeSearchOptionWholeWord()); m_search->SetMatchRegexp(settings->GetTapeSearchOptionRegexp()); m_search->SetMatchCase(settings->GetTapeSearchOptionMatchCase()); ui->lineEditFind->setPlaceholderText(m_search->SearchPlaceholder()); UpdateSearchControlsTooltips(); connect(ui->lineEditFind, &QLineEdit::textChanged, this, [this](const QString &term) { m_search->Find(term); }); connect(ui->lineEditFind, &QLineEdit::editingFinished, this, [this]() { SaveSearchRequest(); InitSearchHistory(); m_search->Find(ui->lineEditFind->text()); }); connect(ui->toolButtonFindPrevious, &QToolButton::clicked, this, [this]() { SaveSearchRequest(); InitSearchHistory(); m_search->FindPrevious(); ui->labelResults->setText( QStringLiteral("%1/%2").arg(m_search->MatchIndex() + 1).arg(m_search->MatchCount())); }); connect(ui->toolButtonFindNext, &QToolButton::clicked, this, [this]() { SaveSearchRequest(); InitSearchHistory(); m_search->FindNext(); ui->labelResults->setText( QStringLiteral("%1/%2").arg(m_search->MatchIndex() + 1).arg(m_search->MatchCount())); }); connect(m_search.data(), &VTableSearch::HasResult, this, [this](bool state) { ui->toolButtonFindPrevious->setEnabled(state); ui->toolButtonFindNext->setEnabled(state); if (state) { ui->labelResults->setText( QStringLiteral("%1/%2").arg(m_search->MatchIndex() + 1).arg(m_search->MatchCount())); } else { ui->labelResults->setText(tr("0 results")); } QPalette palette; if (not state && not ui->lineEditFind->text().isEmpty()) { palette.setColor(QPalette::Text, Qt::red); ui->lineEditFind->setPalette(palette); palette.setColor(QPalette::Active, ui->labelResults->foregroundRole(), Qt::red); palette.setColor(QPalette::Inactive, ui->labelResults->foregroundRole(), Qt::red); ui->labelResults->setPalette(palette); } else { ui->lineEditFind->setPalette(palette); ui->labelResults->setPalette(palette); } }); connect(ui->toolButtonCaseSensitive, &QToolButton::toggled, this, [this](bool checked) { m_search->SetMatchCase(checked); m_search->Find(ui->lineEditFind->text()); ui->lineEditFind->setPlaceholderText(m_search->SearchPlaceholder()); }); connect(ui->toolButtonWholeWord, &QToolButton::toggled, this, [this](bool checked) { m_search->SetMatchWord(checked); m_search->Find(ui->lineEditFind->text()); ui->lineEditFind->setPlaceholderText(m_search->SearchPlaceholder()); }); connect(ui->toolButtonRegexp, &QToolButton::toggled, this, [this](bool checked) { m_search->SetMatchRegexp(checked); if (checked) { ui->toolButtonWholeWord->blockSignals(true); ui->toolButtonWholeWord->setChecked(false); ui->toolButtonWholeWord->blockSignals(false); ui->toolButtonWholeWord->setEnabled(false); ui->toolButtonUseUnicodeProperties->setEnabled(true); } else { ui->toolButtonWholeWord->setEnabled(true); ui->toolButtonUseUnicodeProperties->blockSignals(true); ui->toolButtonUseUnicodeProperties->setChecked(false); ui->toolButtonUseUnicodeProperties->blockSignals(false); ui->toolButtonUseUnicodeProperties->setEnabled(false); } m_search->Find(ui->lineEditFind->text()); ui->lineEditFind->setPlaceholderText(m_search->SearchPlaceholder()); }); connect(ui->toolButtonUseUnicodeProperties, &QToolButton::toggled, this, [this](bool checked) { m_search->SetUseUnicodePreperties(checked); m_search->Find(ui->lineEditFind->text()); }); m_searchHistory->setStyleSheet(QStringLiteral("QMenu { menu-scrollable: 1; }")); InitSearchHistory(); ui->pushButtonSearch->setMenu(m_searchHistory); } //--------------------------------------------------------------------------------------------------------------------- void TMainWindow::InitSearchHistory() { QStringList searchHistory = MApplication::VApp()->TapeSettings()->GetTapeSearchHistory(); m_searchHistory->clear(); for (const auto &term : searchHistory) { QAction *action = m_searchHistory->addAction(term); action->setData(term); connect(action, &QAction::triggered, this, [this]() { auto *action = qobject_cast(sender()); if (action != nullptr) { QString term = action->data().toString(); ui->lineEditFind->setText(term); m_search->Find(term); ui->lineEditFind->setFocus(); } }); } } //--------------------------------------------------------------------------------------------------------------------- void TMainWindow::SaveSearchRequest() { QStringList searchHistory = MApplication::VApp()->TapeSettings()->GetTapeSearchHistory(); QString term = ui->lineEditFind->text(); if (term.isEmpty()) { return; } searchHistory.removeAll(term); searchHistory.prepend(term); while (searchHistory.size() > VTableSearch::MaxHistoryRecords) { searchHistory.removeLast(); } MApplication::VApp()->TapeSettings()->SetTapeSearchHistory(searchHistory); } //--------------------------------------------------------------------------------------------------------------------- void TMainWindow::UpdateSearchControlsTooltips() { auto UpdateToolTip = [](QAbstractButton *button) { if (button->toolTip().contains(QLatin1String("%1"))) { button->setToolTip(button->toolTip().arg(button->shortcut().toString(QKeySequence::NativeText))); } }; UpdateToolTip(ui->toolButtonCaseSensitive); UpdateToolTip(ui->toolButtonWholeWord); UpdateToolTip(ui->toolButtonRegexp); UpdateToolTip(ui->toolButtonUseUnicodeProperties); UpdateToolTip(ui->pushButtonSearch); UpdateToolTip(ui->toolButtonFindPrevious); UpdateToolTip(ui->toolButtonFindNext); } //--------------------------------------------------------------------------------------------------------------------- void TMainWindow::RetranslateTableHeaders() { if (m_mType == MeasurementsType::Multisize) { const QList dimensions = m_m->Dimensions().values(); if (not dimensions.isEmpty()) { const MeasurementDimension_p &dimension = dimensions.at(0); ui->tableWidget->horizontalHeaderItem(ColumnShiftA)->setText(tr("%1 shift").arg(dimension->Name())); } if (dimensions.size() < 2) { ui->tableWidget->setColumnHidden(ColumnShiftB, true); } else { const MeasurementDimension_p &dimension = dimensions.at(1); ui->tableWidget->horizontalHeaderItem(ColumnShiftB)->setText(tr("%1 shift").arg(dimension->Name())); } if (dimensions.size() < 3) { ui->tableWidget->setColumnHidden(ColumnShiftC, true); } else { const MeasurementDimension_p &dimension = dimensions.at(2); ui->tableWidget->horizontalHeaderItem(ColumnShiftC)->setText(tr("%1 shift").arg(dimension->Name())); } } } //--------------------------------------------------------------------------------------------------------------------- template void TMainWindow::HackWidget(T **widget) { delete *widget; *widget = new T(); m_hackedWidgets.append(*widget); }