///////////////////////////////////////////////////////////////////////////////// // // 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 "PlotBuilderCommands/RicSummaryPlotBuilder.h" #include "RicPasteAsciiDataToSummaryPlotFeatureUi.h" #include "RiaLogging.h" #include "RimAsciiDataCurve.h" #include "RimSummaryCurveAppearanceCalculator.h" #include "RimSummaryMultiPlot.h" #include "RimSummaryPlot.h" #include "cafPdmDefaultObjectFactory.h" #include "cafPdmDocument.h" #include "cafPdmObjectGroup.h" #include "cafPdmSettings.h" #include "cafPdmUiPropertyViewDialog.h" #include "cafSelectionManager.h" #include "cvfAssert.h" #include "cvfColor3.h" #include #include #include #include CAF_CMD_SOURCE_INIT( RicPasteAsciiDataToSummaryPlotFeature, "RicPasteAsciiDataToSummaryPlotFeature" ); //-------------------------------------------------------------------------------------------------- /// //-------------------------------------------------------------------------------------------------- bool RicPasteAsciiDataToSummaryPlotFeature::isCommandEnabled() const { caf::PdmObjectHandle* destinationObject = dynamic_cast( caf::SelectionManager::instance()->selectedItem() ); if ( !destinationObject ) return false; RimSummaryMultiPlot* multiPlot = destinationObject->firstAncestorOrThisOfType(); if ( !multiPlot ) { return false; } return hasPastedText(); } //-------------------------------------------------------------------------------------------------- /// //-------------------------------------------------------------------------------------------------- void RicPasteAsciiDataToSummaryPlotFeature::onActionTriggered( bool isChecked ) { caf::PdmObjectHandle* destinationObject = dynamic_cast( caf::SelectionManager::instance()->selectedItem() ); RimSummaryPlot* summaryPlot = destinationObject->firstAncestorOrThisOfType(); QString text = getPastedData(); RicPasteAsciiDataToSummaryPlotFeatureUi pasteOptions; caf::PdmSettings::readFieldsFromApplicationStore( &pasteOptions, pasteOptions.contextString() ); if ( !summaryPlot ) pasteOptions.setCreateNewPlot(); pasteOptions.setUiModePasteText( text ); caf::PdmUiPropertyViewDialog propertyDialog( nullptr, &pasteOptions, "Set Paste Options", "" ); if ( propertyDialog.exec() != QDialog::Accepted ) return; std::vector curves = parseCurves( text, pasteOptions ); if ( curves.size() > 0 ) { if ( !summaryPlot ) { summaryPlot = new RimSummaryPlot(); summaryPlot->enableAutoPlotTitle( true ); RicSummaryPlotBuilder::createAndAppendSingleSummaryMultiPlot( summaryPlot ); } caf::PdmSettings::writeFieldsToApplicationStore( &pasteOptions, pasteOptions.contextString() ); for ( RimAsciiDataCurve* curve : curves ) { summaryPlot->addAsciiDataCruve( curve ); } summaryPlot->updateConnectedEditors(); summaryPlot->loadDataAndUpdate(); } } //-------------------------------------------------------------------------------------------------- /// //-------------------------------------------------------------------------------------------------- void RicPasteAsciiDataToSummaryPlotFeature::setupActionLook( QAction* actionToSetup ) { actionToSetup->setText( "Paste Excel Data to Summary Plot" ); 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(); if ( mimeData->hasText() && mimeData->text().size() > 12 ) { QString text = mimeData->text(); if ( text.size() > 12 ) return true; } return false; } //-------------------------------------------------------------------------------------------------- /// //-------------------------------------------------------------------------------------------------- std::vector RicPasteAsciiDataToSummaryPlotFeature::parseCurves( QString& data, const RicPasteAsciiDataToSummaryPlotFeatureUi& settings ) { std::vector curves; const AsciiDataParseOptions& parseOptions = settings.parseOptions(); RifCsvUserDataPastedTextParser parser = RifCsvUserDataPastedTextParser( data ); if ( !parser.parse( parseOptions ) ) { return curves; } if ( parser.tableData().columnInfos().empty() || !parser.dateTimeColumn() ) { return curves; } std::map> curveToTypeMap; QString curvePrefix = parseOptions.curvePrefix; for ( size_t i = 0; i < parser.tableData().columnInfos().size(); i++ ) { const Column* col = parser.columnInfo( i ); if ( col->dataType != Column::NUMERIC ) continue; RimAsciiDataCurve* curve = new RimAsciiDataCurve(); curve->setTimeSteps( parser.dateTimeColumn()->qDateTimeValues() ); curve->setValues( parser.columnInfo( i )->values ); if ( curvePrefix.isEmpty() ) { curve->setTitle( QString::fromStdString( col->columnName() ) ); } else { curve->setTitle( QString( "%1: %2" ).arg( curvePrefix ).arg( QString::fromStdString( col->columnName() ) ) ); } // Appearance curve->setSymbol( parseOptions.curveSymbol ); curve->setLineStyle( parseOptions.curveLineStyle ); curve->setSymbolSkipDistance( parseOptions.curveSymbolSkipDistance ); curveToTypeMap[guessCurveType( QString::fromStdString( col->columnName() ) )].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; }