ResInsight/ApplicationLibCode/Commands/ApplicationCommands/RicShowPlotDataFeature.cpp
Magne Sjaastad 34e30b38d3 #11372 Make sure regression curves can be resampled
Make sure a regression curve based on observed data can be resampled. Split display of curves, one section for resampled curves and one section for observed curves without resampling.
2024-07-25 10:29:16 +02:00

380 lines
13 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 "RicShowPlotDataFeature.h"
#include "RiaFeatureCommandContext.h"
#include "RiaGuiApplication.h"
#include "RiaPreferencesSummary.h"
#include "RiaQDateTimeTools.h"
#include "RimAnalysisPlot.h"
#include "RimCorrelationMatrixPlot.h"
#include "RimCorrelationPlot.h"
#include "RimCorrelationReportPlot.h"
#include "RimGridCrossPlot.h"
#include "RimGridCrossPlotCurve.h"
#include "RimParameterResultCrossPlot.h"
#include "RimPlotWindow.h"
#include "RimProject.h"
#include "RimSummaryPlot.h"
#include "RimWellAllocationOverTimePlot.h"
#include "RimWellLogPlot.h"
#include "RimWellLogTrack.h"
#include "VerticalFlowPerformance/RimCustomVfpPlot.h"
#include "RiuPlotMainWindow.h"
#include "RiuTextDialog.h"
#include "cafPdmPointer.h"
#include "cafProgressInfo.h"
#include "cafSelectionManagerTools.h"
#include <QAction>
CAF_CMD_SOURCE_INIT( RicShowPlotDataFeature, "RicShowPlotDataFeature" );
//--------------------------------------------------------------------------------------------------
/// Private text provider class for summary plots
//--------------------------------------------------------------------------------------------------
class RiuTabbedSummaryPlotTextProvider : public RiuTabbedTextProvider
{
public:
RiuTabbedSummaryPlotTextProvider( RimSummaryPlot* summaryPlot )
: m_summaryPlot( summaryPlot )
{
}
bool isValid() const override { return m_summaryPlot.notNull(); }
QString description() const override
{
CVF_ASSERT( m_summaryPlot.notNull() && "Need to check that provider is valid" );
return m_summaryPlot->description();
}
QString tabTitle( int tabIndex ) const override
{
auto allTabs = tabs();
CVF_ASSERT( tabIndex < (int)allTabs.size() );
RiaDefines::DateTimePeriod timePeriod = allTabs[tabIndex];
if ( timePeriod == RiaDefines::DateTimePeriod::NONE )
{
return "No Resampling";
}
return QString( "%1" ).arg( RiaQDateTimeTools::dateTimePeriodName( timePeriod ) );
}
QString tabText( int tabIndex ) const override
{
CVF_ASSERT( m_summaryPlot.notNull() && "Need to check that provider is valid" );
RiaDefines::DateTimePeriod timePeriod = indexToPeriod( tabIndex );
RiaPreferencesSummary* prefs = RiaPreferencesSummary::current();
return m_summaryPlot->asciiDataForSummaryPlotExport( timePeriod, prefs->showSummaryTimeAsLongString() );
}
int tabCount() const override { return (int)tabs().size(); }
private:
static RiaDefines::DateTimePeriod indexToPeriod( int tabIndex )
{
auto allTabs = tabs();
CVF_ASSERT( tabIndex < (int)allTabs.size() );
RiaDefines::DateTimePeriod timePeriod = allTabs[tabIndex];
return timePeriod;
}
static std::vector<RiaDefines::DateTimePeriod> tabs()
{
std::vector<RiaDefines::DateTimePeriod> dateTimePeriods;
dateTimePeriods.emplace_back( RiaDefines::DateTimePeriod::NONE );
dateTimePeriods.emplace_back( RiaDefines::DateTimePeriod::YEAR );
dateTimePeriods.emplace_back( RiaDefines::DateTimePeriod::HALFYEAR );
dateTimePeriods.emplace_back( RiaDefines::DateTimePeriod::QUARTER );
dateTimePeriods.emplace_back( RiaDefines::DateTimePeriod::MONTH );
dateTimePeriods.emplace_back( RiaDefines::DateTimePeriod::WEEK );
dateTimePeriods.emplace_back( RiaDefines::DateTimePeriod::DAY );
dateTimePeriods.emplace_back( RiaDefines::DateTimePeriod::HOUR );
dateTimePeriods.emplace_back( RiaDefines::DateTimePeriod::MINUTE );
return dateTimePeriods;
}
private:
caf::PdmPointer<RimSummaryPlot> m_summaryPlot;
};
//--------------------------------------------------------------------------------------------------
/// Private text provider class for grid cross plots
//--------------------------------------------------------------------------------------------------
class RiuTabbedGridCrossPlotTextProvider : public RiuTabbedTextProvider
{
public:
RiuTabbedGridCrossPlotTextProvider( RimGridCrossPlot* crossPlot )
: m_crossPlot( crossPlot )
{
}
bool isValid() const override { return m_crossPlot.notNull(); }
QString description() const override
{
CVF_ASSERT( m_crossPlot.notNull() && "Need to check that provider is valid" );
return m_crossPlot->createAutoName();
}
QString tabTitle( int tabIndex ) const override
{
CVF_ASSERT( m_crossPlot.notNull() && "Need to check that provider is valid" );
return m_crossPlot->asciiTitleForPlotExport( tabIndex );
}
QString tabText( int tabIndex ) const override
{
CVF_ASSERT( m_crossPlot.notNull() && "Need to check that provider is valid" );
return m_crossPlot->asciiDataForGridCrossPlotExport( tabIndex );
}
int tabCount() const override
{
CVF_ASSERT( m_crossPlot.notNull() && "Need to check that provider is valid" );
return (int)m_crossPlot->dataSets().size();
}
private:
caf::PdmPointer<RimGridCrossPlot> m_crossPlot;
};
//--------------------------------------------------------------------------------------------------
///
///
/// RicShowPlotDataFeature
///
///
//--------------------------------------------------------------------------------------------------
bool RicShowPlotDataFeature::isCommandEnabled() const
{
QString content = RiaFeatureCommandContext::instance()->contentString();
if ( !content.isEmpty() )
{
return true;
}
std::vector<RimPlotWindow*> selection;
getSelection( selection );
int validPlots = 0;
for ( auto plot : selection )
{
if ( dynamic_cast<RimSummaryPlot*>( plot ) || dynamic_cast<RimWellLogPlot*>( plot ) || dynamic_cast<RimWellLogTrack*>( plot ) ||
dynamic_cast<RimGridCrossPlot*>( plot ) || dynamic_cast<RimCustomVfpPlot*>( plot ) ||
dynamic_cast<RimWellAllocationOverTimePlot*>( plot ) || dynamic_cast<RimAnalysisPlot*>( plot ) ||
dynamic_cast<RimCorrelationMatrixPlot*>( plot ) || dynamic_cast<RimAbstractCorrelationPlot*>( plot ) ||
dynamic_cast<RimCorrelationReportPlot*>( plot ) )
{
validPlots++;
}
}
return ( validPlots > 0 );
}
//--------------------------------------------------------------------------------------------------
///
//--------------------------------------------------------------------------------------------------
void RicShowPlotDataFeature::onActionTriggered( bool isChecked )
{
QString content = RiaFeatureCommandContext::instance()->contentString();
if ( !content.isEmpty() )
{
QString title = "Data Content";
{
QString titleCandidate = RiaFeatureCommandContext::instance()->titleString();
if ( !titleCandidate.isEmpty() ) title = titleCandidate;
}
RicShowPlotDataFeature::showTextWindow( title, content );
return;
}
disableModelChangeContribution();
std::vector<RimPlotWindow*> selection;
getSelection( selection );
// Using RiuTabbedSummaryPlotTextProvider
std::vector<RimSummaryPlot*> summaryPlots;
// Using RiuTabbedGridCrossPlotTextProvider
std::vector<RimGridCrossPlot*> crossPlots;
// Special handling for well log plots
std::vector<RimWellLogPlot*> wellLogPlots;
// Show content using RimPlot::description() and RimPlot::asciiDataForPlotExport()
std::vector<RimPlot*> rimPlots;
for ( auto plot : selection )
{
if ( auto sumPlot = dynamic_cast<RimSummaryPlot*>( plot ) )
{
summaryPlots.push_back( sumPlot );
continue;
}
if ( auto xPlot = dynamic_cast<RimGridCrossPlot*>( plot ) )
{
crossPlots.push_back( xPlot );
continue;
}
if ( auto wellLogPlot = dynamic_cast<RimWellLogPlot*>( plot ) )
{
wellLogPlots.push_back( wellLogPlot );
continue;
}
if ( auto correlationReportPlot = dynamic_cast<RimCorrelationReportPlot*>( plot ) )
{
// A correlation report plot contains three plots. Add them as individual plots to rimPlots to make the data available in three
// individual text dialogs.
rimPlots.push_back( correlationReportPlot->matrixPlot() );
rimPlots.push_back( correlationReportPlot->correlationPlot() );
rimPlots.push_back( correlationReportPlot->crossPlot() );
continue;
}
if ( auto rimPlot = dynamic_cast<RimPlot*>( plot ) )
{
rimPlots.push_back( rimPlot );
}
}
for ( RimSummaryPlot* summaryPlot : summaryPlots )
{
auto textProvider = new RiuTabbedSummaryPlotTextProvider( summaryPlot );
RicShowPlotDataFeature::showTabbedTextWindow( textProvider );
}
for ( RimGridCrossPlot* crossPlot : crossPlots )
{
auto textProvider = new RiuTabbedGridCrossPlotTextProvider( crossPlot );
RicShowPlotDataFeature::showTabbedTextWindow( textProvider );
}
for ( auto rimPlot : rimPlots )
{
QString title = rimPlot->description();
QString text = title;
text += "\n";
text += "\n";
text += rimPlot->asciiDataForPlotExport();
RicShowPlotDataFeature::showTextWindow( title, text );
}
for ( auto wellLogPlot : wellLogPlots )
{
QString title = wellLogPlot->description();
QString text = title;
text += "\n";
text += "\n";
text += wellLogPlot->asciiDataForPlotExport();
RicShowPlotDataFeature::showTextWindow( title, text );
}
}
//--------------------------------------------------------------------------------------------------
///
//--------------------------------------------------------------------------------------------------
void RicShowPlotDataFeature::setupActionLook( QAction* actionToSetup )
{
actionToSetup->setText( "Show Plot Data" );
actionToSetup->setIcon( QIcon( ":/PlotWindow.svg" ) );
}
//--------------------------------------------------------------------------------------------------
///
//--------------------------------------------------------------------------------------------------
void RicShowPlotDataFeature::showTabbedTextWindow( RiuTabbedTextProvider* textProvider )
{
RiuPlotMainWindow* plotwindow = RiaGuiApplication::instance()->mainPlotWindow();
CVF_ASSERT( plotwindow );
auto* textWidget = new RiuTabbedTextDialog( textProvider );
textWidget->setMinimumSize( 800, 600 );
plotwindow->addToTemporaryWidgets( textWidget );
textWidget->show();
textWidget->redrawText();
}
//--------------------------------------------------------------------------------------------------
///
//--------------------------------------------------------------------------------------------------
void RicShowPlotDataFeature::showTextWindow( const QString& title, const QString& text )
{
RiuPlotMainWindow* plotwindow = RiaGuiApplication::instance()->mainPlotWindow();
CVF_ASSERT( plotwindow );
auto* textWiget = new RiuTextDialog();
textWiget->setMinimumSize( 400, 600 );
textWiget->setWindowTitle( title );
textWiget->setText( text );
textWiget->show();
plotwindow->addToTemporaryWidgets( textWiget );
}
//--------------------------------------------------------------------------------------------------
///
//--------------------------------------------------------------------------------------------------
void RicShowPlotDataFeature::getSelection( std::vector<RimPlotWindow*>& selection ) const
{
if ( sender() )
{
QVariant userData = this->userData();
if ( !userData.isNull() && userData.canConvert<void*>() )
{
auto* plot = static_cast<RimPlot*>( userData.value<void*>() );
if ( plot ) selection.push_back( plot );
}
}
if ( selection.empty() )
{
auto selectedObjects = caf::selectedObjectsByType<caf::PdmObject*>();
for ( auto obj : selectedObjects )
{
if ( !obj ) continue;
if ( auto plotWindow = obj->firstAncestorOrThisOfType<RimPlotWindow>() )
{
selection.push_back( plotWindow );
}
}
}
}