///////////////////////////////////////////////////////////////////////////////// // // Copyright (C) 2017 Statoil ASA // // ResInsight 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. // // ResInsight 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 at // for more details. // ///////////////////////////////////////////////////////////////////////////////// #include "RicPasteAsciiDataToSummaryPlotFeature.h" #include "OperationsUsingObjReferences/RicPasteFeatureImpl.h" #include "RicNewSummaryPlotFeature.h" #include "RicPasteAsciiDataToSummaryPlotFeatureUi.h" #include "RiaLogging.h" #include "RimSummaryPlot.h" #include "RimSummaryPlotCollection.h" #include "RimAsciiDataCurve.h" #include "RimSummaryCurveAppearanceCalculator.h" #include "cafPdmDefaultObjectFactory.h" #include "cafPdmDocument.h" #include "cafPdmObjectGroup.h" #include "cafSelectionManager.h" #include "cafPdmUiPropertyViewDialog.h" #include "cafPdmSettings.h" #include "cvfAssert.h" #include "cvfColor3.h" #include #include #include #include CAF_CMD_SOURCE_INIT(RicPasteAsciiDataToSummaryPlotFeature, "RicPasteAsciiDataToSummaryPlotFeature"); //-------------------------------------------------------------------------------------------------- /// //-------------------------------------------------------------------------------------------------- bool RicPasteAsciiDataToSummaryPlotFeature::isCommandEnabled() { caf::PdmObjectHandle* destinationObject = dynamic_cast(caf::SelectionManager::instance()->selectedItem()); RimSummaryPlot* summaryPlot = nullptr; destinationObject->firstAncestorOrThisOfType(summaryPlot); if (!summaryPlot) { RimSummaryPlotCollection* summaryPlotCollection = nullptr; destinationObject->firstAncestorOrThisOfType(summaryPlotCollection); if (!summaryPlotCollection) { return false; } } return hasPastedText(); } //-------------------------------------------------------------------------------------------------- /// //-------------------------------------------------------------------------------------------------- void RicPasteAsciiDataToSummaryPlotFeature::onActionTriggered(bool isChecked) { caf::PdmObjectHandle* destinationObject = dynamic_cast(caf::SelectionManager::instance()->selectedItem()); RimSummaryPlot* summaryPlot = nullptr; destinationObject->firstAncestorOrThisOfType(summaryPlot); RicPasteAsciiDataToSummaryPlotFeatureUi pasteOptions; if (!summaryPlot) pasteOptions.createNewPlot(); caf::PdmSettings::readFieldsFromApplicationStore(&pasteOptions); caf::PdmUiPropertyViewDialog propertyDialog(NULL, &pasteOptions, "Set Paste Options", ""); if (propertyDialog.exec() != QDialog::Accepted) return; if (!summaryPlot) { RimSummaryPlotCollection* summaryPlotCollection = nullptr; destinationObject->firstAncestorOrThisOfType(summaryPlotCollection); if (!summaryPlotCollection) { return; } summaryPlot = RicNewSummaryPlotFeature::createNewSummaryPlot(summaryPlotCollection, nullptr); if (!summaryPlot) { return; } summaryPlot->setDescription(pasteOptions.plotTitle()); } caf::PdmSettings::writeFieldsToApplicationStore(&pasteOptions); QString text = getPastedData(); std::vector curves = parseCurves(text, pasteOptions); for (RimAsciiDataCurve* curve : curves) { summaryPlot->addAsciiDataCruve(curve); } summaryPlot->updateConnectedEditors(); summaryPlot->loadDataAndUpdate(); } //-------------------------------------------------------------------------------------------------- /// //-------------------------------------------------------------------------------------------------- void RicPasteAsciiDataToSummaryPlotFeature::setupActionLook(QAction* actionToSetup) { actionToSetup->setText("Paste Excel Data to Summary Curve"); RicPasteFeatureImpl::setIconAndShortcuts(actionToSetup); } //-------------------------------------------------------------------------------------------------- /// //-------------------------------------------------------------------------------------------------- QString RicPasteAsciiDataToSummaryPlotFeature::getPastedData() { if (hasPastedText()) { QClipboard* clipboard = QApplication::clipboard(); return clipboard->text(); } return QString(); } //-------------------------------------------------------------------------------------------------- /// //-------------------------------------------------------------------------------------------------- bool RicPasteAsciiDataToSummaryPlotFeature::hasPastedText() { QClipboard* clipboard = QApplication::clipboard(); const QMimeData* mimeData = clipboard->mimeData(); return mimeData->hasText(); } //-------------------------------------------------------------------------------------------------- /// //-------------------------------------------------------------------------------------------------- std::vector RicPasteAsciiDataToSummaryPlotFeature::parseCurves(QString& data, const RicPasteAsciiDataToSummaryPlotFeatureUi& settings) { std::vector curves; std::vector headers; QTextStream tableData(&data); { QString header; do { header = tableData.readLine(); } while (header.isEmpty() && !tableData.atEnd()); // No header row found if (header.isEmpty()) return curves; QStringList columnHeaders = header.split('\t'); for (int i = 1; i < columnHeaders.size(); ++i) { headers.push_back(columnHeaders[i]); } // No columns found if (headers.empty()) return curves; } int numColumns = static_cast(headers.size()); std::vector timeSteps; std::vector< std::vector > values; values.resize(numColumns); size_t row = 0; while (!tableData.atEnd()) { ++row; QString line = tableData.readLine(); // Skip empty lines if (line.isEmpty()) continue; QStringList columns = line.split(settings.cellSeparator()); if (columns.size() != numColumns + 1) { RiaLogging::warning(QString("Invalid number of columns in row %1").arg(row)); continue; } QDateTime date = QDateTime::fromString(columns[0], settings.dateFormat()); if (!date.isValid()) { RiaLogging::warning(QString("First column of row %1 could not be parsed as a date: %2").arg(row).arg(columns[0])); continue; } timeSteps.push_back(date); for (int col = 1; col < columns.size(); ++col) { bool ok; values[col - 1].push_back(settings.decimalLocale().toDouble(columns[col], &ok)); if (!ok) { RiaLogging::warning(QString("Could not parse value at row %1 column %2 as double: %3. Defaulting to 0.0").arg(row).arg(col).arg(columns[col])); } } } std::map< CurveType, std::vector > curveToTypeMap; QString curvePrefix = settings.curvePrefix(); for (size_t i = 0; i < values.size(); ++i) { RimAsciiDataCurve* curve = new RimAsciiDataCurve(); curve->setTimeSteps(timeSteps); curve->setValues(values[i]); if (curvePrefix.isEmpty()) { curve->setTitle(headers[i]); } else { curve->setTitle(QString("%1: %2").arg(curvePrefix).arg(headers[i])); } // Appearance curve->setSymbol(settings.pointSymbol()); curve->setLineStyle(settings.lineStyle()); curve->setSymbolSkipDinstance(settings.symbolSkipDinstance()); curveToTypeMap[guessCurveType(headers[i])].push_back(curve); curves.push_back(curve); } for (auto& it : curveToTypeMap) { for (int i = 0; i < static_cast(it.second.size()); ++i) { cvf::Color3f color; switch (it.first) { case CURVE_GAS: color = RimSummaryCurveAppearanceCalculator::cycledGreenColor(i); break; case CURVE_OIL: color = RimSummaryCurveAppearanceCalculator::cycledRedColor(i); break; case CURVE_WAT: color = RimSummaryCurveAppearanceCalculator::cycledBlueColor(i); break; default: color = RimSummaryCurveAppearanceCalculator::cycledNoneRGBBrColor(i); break; } it.second[i]->setColor(color); } } return curves; } //-------------------------------------------------------------------------------------------------- /// //-------------------------------------------------------------------------------------------------- RicPasteAsciiDataToSummaryPlotFeature::CurveType RicPasteAsciiDataToSummaryPlotFeature::guessCurveType(const QString& curveName) { if (curveName.contains("SW") || curveName.contains("water", Qt::CaseInsensitive)) { return CURVE_WAT; } else if (curveName.contains("oil", Qt::CaseInsensitive)) { return CURVE_OIL; } else if (curveName.contains("gas", Qt::CaseInsensitive)) { return CURVE_GAS; } return CURVE_UNKNOWN; }