mirror of
https://github.com/OPM/ResInsight.git
synced 2025-01-24 07:16:53 -06:00
296 lines
9.8 KiB
C++
296 lines
9.8 KiB
C++
/////////////////////////////////////////////////////////////////////////////////
|
|
//
|
|
// 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 <http://www.gnu.org/licenses/gpl.html>
|
|
// 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 <QApplication>
|
|
#include <QAction>
|
|
#include <QClipboard>
|
|
#include <QMimeData>
|
|
|
|
|
|
CAF_CMD_SOURCE_INIT(RicPasteAsciiDataToSummaryPlotFeature, "RicPasteAsciiDataToSummaryPlotFeature");
|
|
|
|
//--------------------------------------------------------------------------------------------------
|
|
///
|
|
//--------------------------------------------------------------------------------------------------
|
|
bool RicPasteAsciiDataToSummaryPlotFeature::isCommandEnabled()
|
|
{
|
|
caf::PdmObjectHandle* destinationObject = dynamic_cast<caf::PdmObjectHandle*>(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::PdmObjectHandle*>(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<RimAsciiDataCurve*> 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<RimAsciiDataCurve*> RicPasteAsciiDataToSummaryPlotFeature::parseCurves(QString& data, const RicPasteAsciiDataToSummaryPlotFeatureUi& settings)
|
|
{
|
|
std::vector<RimAsciiDataCurve*> curves;
|
|
std::vector<QString> 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<int>(headers.size());
|
|
std::vector<QDateTime> timeSteps;
|
|
std::vector< std::vector<double> > 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<RimAsciiDataCurve*> > 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<int>(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;
|
|
}
|