2021-09-06 02:32:24 -05:00
|
|
|
/////////////////////////////////////////////////////////////////////////////////
|
|
|
|
//
|
|
|
|
// Copyright (C) 2021- Equinor 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 "RicCreateEnsembleWellLogFeature.h"
|
|
|
|
|
|
|
|
#include "RiaApplication.h"
|
|
|
|
#include "RiaColorTables.h"
|
|
|
|
#include "RiaImportEclipseCaseTools.h"
|
|
|
|
#include "RiaLogging.h"
|
2024-07-04 05:38:41 -05:00
|
|
|
#include "RiaPreferencesGrid.h"
|
2021-09-06 02:32:24 -05:00
|
|
|
|
|
|
|
#include "ExportCommands/RicExportToLasFileFeature.h"
|
2021-09-15 04:46:24 -05:00
|
|
|
#include "RicCloseCaseFeature.h"
|
2021-09-06 02:32:24 -05:00
|
|
|
#include "RicCreateEnsembleWellLogUi.h"
|
|
|
|
#include "RicImportEnsembleWellLogsFeature.h"
|
2021-09-23 07:19:55 -05:00
|
|
|
#include "RicImportGeneralDataFeature.h"
|
2021-09-06 02:32:24 -05:00
|
|
|
#include "RicRecursiveFileSearchDialog.h"
|
2021-09-27 05:14:06 -05:00
|
|
|
#include "RifReaderSettings.h"
|
2021-09-06 02:32:24 -05:00
|
|
|
#include "WellPathCommands/RicImportWellPaths.h"
|
|
|
|
|
|
|
|
#include "RimDialogData.h"
|
|
|
|
#include "RimEclipseCase.h"
|
|
|
|
#include "RimEnsembleWellLogCurveSet.h"
|
|
|
|
#include "RimMainPlotCollection.h"
|
|
|
|
#include "RimProject.h"
|
|
|
|
#include "RimWellLogTrack.h"
|
2021-09-21 08:33:18 -05:00
|
|
|
#include "RimWellPath.h"
|
2021-09-06 02:32:24 -05:00
|
|
|
#include "RimcWellLogPlot.h"
|
|
|
|
#include "RimcWellLogPlotCollection.h"
|
|
|
|
#include "RimcWellLogTrack.h"
|
|
|
|
|
|
|
|
#include "Riu3DMainWindowTools.h"
|
|
|
|
#include "RiuPlotMainWindowTools.h"
|
|
|
|
#include "RiuPropertyViewTabWidget.h"
|
|
|
|
|
|
|
|
#include "cafCmdFeatureManager.h"
|
|
|
|
#include "cafPdmSettings.h"
|
|
|
|
#include "cafPdmUiPropertyViewDialog.h"
|
|
|
|
#include "cafProgressInfo.h"
|
|
|
|
#include "cafSelectionManager.h"
|
|
|
|
|
|
|
|
#include <QAction>
|
|
|
|
#include <QDir>
|
|
|
|
#include <QFileInfo>
|
|
|
|
|
|
|
|
CAF_CMD_SOURCE_INIT( RicCreateEnsembleWellLogFeature, "RicCreateEnsembleWellLogFeature" );
|
|
|
|
|
|
|
|
//--------------------------------------------------------------------------------------------------
|
|
|
|
///
|
|
|
|
//--------------------------------------------------------------------------------------------------
|
|
|
|
void RicCreateEnsembleWellLogFeature::openDialogAndExecuteCommand()
|
|
|
|
{
|
|
|
|
// Get the list of egrid files
|
|
|
|
RiaApplication* app = RiaApplication::instance();
|
|
|
|
QString defaultDir = app->lastUsedDialogDirectory( "BINARY_GRID" );
|
|
|
|
|
|
|
|
QString pathFilter( "*" );
|
|
|
|
QString fileNameFilter( "*" );
|
|
|
|
|
2023-04-20 03:16:43 -05:00
|
|
|
RicRecursiveFileSearchDialogResult result =
|
|
|
|
RicRecursiveFileSearchDialog::runRecursiveSearchDialog( nullptr,
|
|
|
|
"Choose Eclipse Cases",
|
|
|
|
defaultDir,
|
|
|
|
pathFilter,
|
|
|
|
fileNameFilter,
|
|
|
|
{ RicRecursiveFileSearchDialog::FileType::GRDECL,
|
|
|
|
RicRecursiveFileSearchDialog::FileType::EGRID } );
|
2021-09-06 02:32:24 -05:00
|
|
|
if ( !result.ok || result.files.isEmpty() )
|
|
|
|
{
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Use case data from first case
|
|
|
|
RimEclipseCase* eclipseCase = loadEclipseCase( result.files[0] );
|
2021-09-23 07:19:55 -05:00
|
|
|
if ( !eclipseCase ) return;
|
2021-09-06 02:32:24 -05:00
|
|
|
|
|
|
|
RicCreateEnsembleWellLogUi* ui = RimProject::current()->dialogData()->createEnsembleWellLogUi();
|
|
|
|
ui->setCaseData( eclipseCase->eclipseCaseData() );
|
|
|
|
|
2021-09-21 08:33:18 -05:00
|
|
|
// Automatically selected the well path the dialog was triggered on (if any)
|
2021-09-23 07:19:55 -05:00
|
|
|
auto* wellPath = caf::SelectionManager::instance()->selectedItemOfType<RimWellPath>();
|
2021-09-21 08:33:18 -05:00
|
|
|
if ( wellPath )
|
|
|
|
{
|
|
|
|
ui->setWellPathSource( RicCreateEnsembleWellLogUi::WellPathSource::PROJECT_WELLS );
|
|
|
|
ui->setWellPathFromProject( wellPath );
|
|
|
|
}
|
|
|
|
|
2023-02-26 03:48:40 -06:00
|
|
|
RiuPropertyViewTabWidget propertyDialog( Riu3DMainWindowTools::mainWindowWidget(), ui, "Create Ensemble Well Log", ui->tabNames() );
|
2021-09-06 02:32:24 -05:00
|
|
|
|
2021-09-21 08:33:18 -05:00
|
|
|
if ( propertyDialog.exec() == QDialog::Accepted && !ui->properties().empty() )
|
2021-09-06 02:32:24 -05:00
|
|
|
{
|
2022-03-11 06:24:01 -06:00
|
|
|
auto stdList = std::list<QString>( result.files.begin(), result.files.end() );
|
|
|
|
executeCommand( *ui, stdList );
|
2021-09-06 02:32:24 -05:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
//--------------------------------------------------------------------------------------------------
|
|
|
|
///
|
|
|
|
//--------------------------------------------------------------------------------------------------
|
2023-02-26 03:48:40 -06:00
|
|
|
void RicCreateEnsembleWellLogFeature::executeCommand( const RicCreateEnsembleWellLogUi& ui, const std::list<QString>& fileNames )
|
2021-09-06 02:32:24 -05:00
|
|
|
{
|
|
|
|
caf::ProgressInfo progress( fileNames.size(), "Creating ensemble well log" );
|
|
|
|
|
|
|
|
std::vector<std::pair<QString, RiaDefines::ResultCatType>> properties = ui.properties();
|
|
|
|
|
2022-08-19 07:30:25 -05:00
|
|
|
RimWellLogPlotCollection* plotCollection = RimMainPlotCollection::current()->wellLogPlotCollection();
|
2021-09-06 02:32:24 -05:00
|
|
|
|
2021-09-21 08:33:18 -05:00
|
|
|
RimWellPath* wellPath = nullptr;
|
|
|
|
|
|
|
|
if ( ui.wellPathSource() == RicCreateEnsembleWellLogUi::WellPathSource::FILE )
|
|
|
|
{
|
|
|
|
if ( ui.wellPathFilePath().isEmpty() ) return;
|
|
|
|
|
|
|
|
// Load well path from file
|
|
|
|
QStringList wellPathFilePaths;
|
|
|
|
wellPathFilePaths << ui.wellPathFilePath();
|
|
|
|
QStringList errorMessages;
|
2023-03-07 05:51:14 -06:00
|
|
|
std::vector<RimWellPath*> wellPaths = RicImportWellPaths::importWellPaths( wellPathFilePaths, &errorMessages );
|
2021-09-21 08:33:18 -05:00
|
|
|
if ( wellPaths.empty() ) return;
|
|
|
|
|
|
|
|
wellPath = wellPaths[0];
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
CAF_ASSERT( ui.wellPathSource() == RicCreateEnsembleWellLogUi::WellPathSource::PROJECT_WELLS );
|
|
|
|
wellPath = ui.wellPathFromProject();
|
|
|
|
}
|
2021-09-06 02:32:24 -05:00
|
|
|
|
2021-09-21 08:33:18 -05:00
|
|
|
if ( !wellPath ) return;
|
2021-09-06 02:32:24 -05:00
|
|
|
|
2021-09-24 05:21:16 -05:00
|
|
|
std::vector<RimWellLogPlot*> tmpPlotsToDelete;
|
|
|
|
QStringList allLasFileNames;
|
2021-09-15 04:46:24 -05:00
|
|
|
for ( const auto& fileName : fileNames )
|
2021-09-06 02:32:24 -05:00
|
|
|
{
|
|
|
|
auto task = progress.task( QString( "Extracting well log for %1" ).arg( fileName ) );
|
|
|
|
|
|
|
|
// Load eclipse case
|
|
|
|
RimEclipseCase* eclipseCase = loadEclipseCase( fileName );
|
|
|
|
if ( !eclipseCase )
|
|
|
|
{
|
|
|
|
RiaLogging::error( QString( "Failed to load model from file: " ).arg( fileName ) );
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Create the well log plot
|
2023-02-26 03:48:40 -06:00
|
|
|
RimWellLogPlot* wellLogPlot = RimcWellLogPlotCollection_newWellLogPlot::createWellLogPlot( plotCollection, wellPath, eclipseCase );
|
2021-09-06 02:32:24 -05:00
|
|
|
|
|
|
|
// Create well log track
|
2023-02-26 03:48:40 -06:00
|
|
|
QString title = "Track";
|
|
|
|
RimWellLogTrack* wellLogTrack = RimcWellLogPlot_newWellLogTrack::createWellLogTrack( wellLogPlot, eclipseCase, wellPath, title );
|
2021-09-06 02:32:24 -05:00
|
|
|
|
|
|
|
// Create a well log curve for each property
|
2021-09-15 04:46:24 -05:00
|
|
|
for ( const auto& property : properties )
|
2021-09-06 02:32:24 -05:00
|
|
|
{
|
|
|
|
QString propertyName = property.first;
|
|
|
|
RiaDefines::ResultCatType resultCategoryType = property.second;
|
|
|
|
int timeStep = ui.timeStep();
|
2023-02-26 03:48:40 -06:00
|
|
|
RimcWellLogTrack_addExtractionCurve::addExtractionCurve( wellLogTrack, eclipseCase, wellPath, propertyName, resultCategoryType, timeStep );
|
2021-09-06 02:32:24 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
{
|
|
|
|
// Create missing directories
|
|
|
|
QString wellLogExportDirName = "lasexport";
|
|
|
|
QFileInfo fi( fileName );
|
|
|
|
QString exportFolder = fi.absoluteDir().absolutePath() + QString( "/%1/" ).arg( wellLogExportDirName );
|
|
|
|
|
|
|
|
if ( !fi.absoluteDir().exists( wellLogExportDirName ) )
|
|
|
|
{
|
|
|
|
if ( !fi.absoluteDir().mkpath( wellLogExportDirName ) )
|
|
|
|
{
|
|
|
|
RiaLogging::error( QString( "Unable to create directory for well log export: " ).arg( exportFolder ) );
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Export to las file
|
|
|
|
QString filePrefix = "";
|
|
|
|
bool exportTvdRkb = false;
|
|
|
|
bool capitalizeFileNames = false;
|
|
|
|
bool alwaysOverwrite = true;
|
|
|
|
double resampleInterval = 0.0;
|
|
|
|
bool convertCurveUnits = false;
|
|
|
|
|
|
|
|
std::vector<QString> lasFiles = RicExportToLasFileFeature::exportToLasFiles( exportFolder,
|
|
|
|
filePrefix,
|
|
|
|
wellLogPlot,
|
|
|
|
exportTvdRkb,
|
|
|
|
capitalizeFileNames,
|
|
|
|
alwaysOverwrite,
|
|
|
|
resampleInterval,
|
|
|
|
convertCurveUnits );
|
2021-09-15 04:46:24 -05:00
|
|
|
for ( const auto& lasFile : lasFiles )
|
2021-09-06 02:32:24 -05:00
|
|
|
allLasFileNames << lasFile;
|
|
|
|
}
|
|
|
|
|
2021-09-24 05:21:16 -05:00
|
|
|
tmpPlotsToDelete.push_back( wellLogPlot );
|
2021-09-15 04:46:24 -05:00
|
|
|
|
|
|
|
RicCloseCaseFeature::deleteEclipseCase( eclipseCase );
|
2021-09-06 02:32:24 -05:00
|
|
|
}
|
|
|
|
|
2021-09-24 05:21:16 -05:00
|
|
|
for ( auto wlp : tmpPlotsToDelete )
|
|
|
|
{
|
|
|
|
// Hide window to avoid flickering
|
|
|
|
wlp->setShowWindow( false );
|
|
|
|
wlp->updateMdiWindowVisibility();
|
|
|
|
|
|
|
|
plotCollection->removePlot( wlp );
|
|
|
|
delete wlp;
|
|
|
|
}
|
|
|
|
|
2021-09-06 02:32:24 -05:00
|
|
|
if ( ui.autoCreateEnsembleWellLogs() )
|
|
|
|
{
|
2021-09-17 07:17:58 -05:00
|
|
|
std::vector<RimEnsembleWellLogs*> ensembleWellLogs =
|
2021-09-30 02:08:09 -05:00
|
|
|
RicImportEnsembleWellLogsFeature::createEnsembleWellLogsFromFiles( allLasFileNames,
|
|
|
|
RiaEnsembleNameTools::EnsembleGroupingMode::FMU_FOLDER_STRUCTURE );
|
2021-09-17 07:17:58 -05:00
|
|
|
|
|
|
|
for ( auto ensembleWellLog : ensembleWellLogs )
|
2021-09-06 02:32:24 -05:00
|
|
|
{
|
2021-09-17 07:17:58 -05:00
|
|
|
if ( ensembleWellLog )
|
|
|
|
{
|
|
|
|
RimEclipseCase* eclipseCase = nullptr;
|
2021-09-06 02:32:24 -05:00
|
|
|
|
2021-09-17 07:17:58 -05:00
|
|
|
// Create the well log plot
|
|
|
|
RimWellLogPlot* wellLogPlot =
|
|
|
|
RimcWellLogPlotCollection_newWellLogPlot::createWellLogPlot( plotCollection, wellPath, eclipseCase );
|
2021-09-06 02:32:24 -05:00
|
|
|
|
2021-09-17 07:17:58 -05:00
|
|
|
// Create a track per property
|
2021-10-08 02:11:20 -05:00
|
|
|
int trackNum = 0;
|
2021-09-17 07:17:58 -05:00
|
|
|
for ( const auto& property : properties )
|
|
|
|
{
|
|
|
|
// Create well log track
|
2021-10-08 02:11:20 -05:00
|
|
|
cvf::Color3f color = RiaColorTables::normalPaletteColors().cycledColor3f( trackNum );
|
|
|
|
QString title = QString( "Track %1" ).arg( trackNum );
|
2021-09-17 07:17:58 -05:00
|
|
|
RimWellLogTrack* wellLogTrack =
|
|
|
|
RimcWellLogPlot_newWellLogTrack::createWellLogTrack( wellLogPlot, eclipseCase, wellPath, title );
|
2021-09-23 07:19:55 -05:00
|
|
|
auto* ensembleWellLogCurveSet = new RimEnsembleWellLogCurveSet();
|
2021-09-17 07:17:58 -05:00
|
|
|
ensembleWellLogCurveSet->setEnsembleWellLogs( ensembleWellLog );
|
|
|
|
ensembleWellLogCurveSet->setColor( color );
|
|
|
|
ensembleWellLogCurveSet->setWellLogChannelName( property.first );
|
|
|
|
wellLogTrack->setEnsembleWellLogCurveSet( ensembleWellLogCurveSet );
|
|
|
|
ensembleWellLogCurveSet->loadDataAndUpdate( true );
|
2021-10-08 02:11:20 -05:00
|
|
|
trackNum++;
|
2021-09-17 07:17:58 -05:00
|
|
|
}
|
2021-09-06 02:32:24 -05:00
|
|
|
|
2021-09-17 07:17:58 -05:00
|
|
|
wellLogPlot->updateConnectedEditors();
|
2021-09-06 02:32:24 -05:00
|
|
|
|
2021-09-17 07:17:58 -05:00
|
|
|
RiuPlotMainWindowTools::showPlotMainWindow();
|
2022-08-18 05:37:51 -05:00
|
|
|
RiuPlotMainWindowTools::onObjectAppended( wellLogPlot );
|
2021-09-17 07:17:58 -05:00
|
|
|
}
|
2021-09-06 02:32:24 -05:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
//--------------------------------------------------------------------------------------------------
|
|
|
|
///
|
|
|
|
//--------------------------------------------------------------------------------------------------
|
|
|
|
RimEclipseCase* RicCreateEnsembleWellLogFeature::loadEclipseCase( const QString& fileName )
|
|
|
|
{
|
|
|
|
QString absolutePath = fileName;
|
|
|
|
QFileInfo projectPathInfo( absolutePath );
|
|
|
|
if ( !projectPathInfo.exists() )
|
|
|
|
{
|
|
|
|
QDir startDir( RiaApplication::instance()->startDir() );
|
|
|
|
absolutePath = startDir.absoluteFilePath( fileName );
|
|
|
|
}
|
2021-09-23 07:19:55 -05:00
|
|
|
bool createView = false;
|
|
|
|
bool createPlot = false;
|
2021-09-27 05:14:06 -05:00
|
|
|
|
2024-07-04 05:38:41 -05:00
|
|
|
RifReaderSettings readerSettings = RiaPreferencesGrid::current()->gridOnlyReaderSettings();
|
|
|
|
auto openResult =
|
2023-02-26 03:48:40 -06:00
|
|
|
RicImportGeneralDataFeature::openEclipseFilesFromFileNames( QStringList( { absolutePath } ), createPlot, createView, readerSettings );
|
2021-09-06 02:32:24 -05:00
|
|
|
|
2021-09-23 07:19:55 -05:00
|
|
|
if ( !openResult.createdCaseIds.empty() )
|
|
|
|
{
|
|
|
|
return RimProject::current()->eclipseCaseFromCaseId( openResult.createdCaseIds.front() );
|
|
|
|
}
|
2021-09-06 02:32:24 -05:00
|
|
|
|
2021-09-23 07:19:55 -05:00
|
|
|
return nullptr;
|
2021-09-06 02:32:24 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
//--------------------------------------------------------------------------------------------------
|
|
|
|
///
|
|
|
|
//--------------------------------------------------------------------------------------------------
|
|
|
|
void RicCreateEnsembleWellLogFeature::onActionTriggered( bool isChecked )
|
|
|
|
{
|
|
|
|
openDialogAndExecuteCommand();
|
|
|
|
}
|
|
|
|
|
|
|
|
//--------------------------------------------------------------------------------------------------
|
|
|
|
///
|
|
|
|
//--------------------------------------------------------------------------------------------------
|
|
|
|
void RicCreateEnsembleWellLogFeature::setupActionLook( QAction* actionToSetup )
|
|
|
|
{
|
|
|
|
actionToSetup->setText( "Create Ensemble Well Log..." );
|
|
|
|
}
|