///////////////////////////////////////////////////////////////////////////////// // // Copyright (C) 2011- Statoil ASA // Copyright (C) 2013- Ceetron Solutions AS // Copyright (C) 2011-2012 Ceetron AS // // 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 "RimEclipseInputCase.h" #include "RiaDefines.h" #include "RiaFieldHandleTools.h" #include "RiaLogging.h" #include "RiaPreferences.h" #include "RifEclipseInputFileTools.h" #include "RifInputPropertyLoader.h" #include "RifReaderEclipseInput.h" #include "RifReaderInterface.h" #include "RifReaderMockModel.h" #include "RifReaderSettings.h" #include "RigActiveCellInfo.h" #include "RigCaseCellResultsData.h" #include "RigEclipseCaseData.h" #include "RigMainGrid.h" #include "RimEclipseInputProperty.h" #include "RimEclipseInputPropertyCollection.h" #include "RimReservoirCellResultsStorage.h" #include "RimTools.h" #include "cafProgressInfo.h" #include #include CAF_PDM_SOURCE_INIT( RimEclipseInputCase, "RimInputReservoir" ); //-------------------------------------------------------------------------------------------------- /// //-------------------------------------------------------------------------------------------------- RimEclipseInputCase::RimEclipseInputCase() : RimEclipseCase() { CAF_PDM_InitObject( "RimInputCase", ":/EclipseInput48x48.png" ); CAF_PDM_InitFieldNoDefault( &m_additionalFiles, "AdditionalFileNamesProxy", "Additional Files" ); m_additionalFiles.registerGetMethod( this, &RimEclipseInputCase::additionalFiles ); m_additionalFiles.uiCapability()->setUiReadOnly( true ); m_additionalFiles.xmlCapability()->setIOWritable( false ); } //-------------------------------------------------------------------------------------------------- /// //-------------------------------------------------------------------------------------------------- RimEclipseInputCase::~RimEclipseInputCase() { } //-------------------------------------------------------------------------------------------------- /// Import ascii properties. If no grid data has been read, it will first find the possible /// grid data among the files then read all supported properties from the files matching the grid //-------------------------------------------------------------------------------------------------- bool RimEclipseInputCase::importAsciiInputProperties( const QStringList& fileNames ) { return openDataFileSet( fileNames ); } //-------------------------------------------------------------------------------------------------- /// Open the supplied file set. If no grid data has been read, it will first find the possible /// grid data among the files then read all supported properties from the files matching the grid //-------------------------------------------------------------------------------------------------- bool RimEclipseInputCase::openDataFileSet( const QStringList& fileNames ) { if ( fileNames.contains( RiaDefines::mockModelBasicInputCase() ) ) { cvf::ref readerInterface = this->createMockModel( fileNames[0] ); results( RiaDefines::PorosityModelType::MATRIX_MODEL )->setReaderInterface( readerInterface.p() ); results( RiaDefines::PorosityModelType::FRACTURE_MODEL )->setReaderInterface( readerInterface.p() ); eclipseCaseData()->activeCellInfo( RiaDefines::PorosityModelType::MATRIX_MODEL )->computeDerivedData(); eclipseCaseData()->activeCellInfo( RiaDefines::PorosityModelType::FRACTURE_MODEL )->computeDerivedData(); QFileInfo gridFileName( fileNames[0] ); QString caseName = gridFileName.completeBaseName(); this->setCaseUserDescription( caseName ); computeCachedData(); return true; } if ( this->eclipseCaseData() == nullptr ) { this->setReservoirData( new RigEclipseCaseData( this ) ); } bool importFaults = RiaPreferences::current()->readerSettings()->importFaults(); std::vector allErrorMessages; QString gridFileName; // First find and read the grid data if ( this->eclipseCaseData()->mainGrid()->gridPointDimensions() == cvf::Vec3st( 0, 0, 0 ) ) { for ( int i = 0; i < fileNames.size(); i++ ) { QString errorMessages; if ( RifEclipseInputFileTools::openGridFile( fileNames[i], this->eclipseCaseData(), importFaults, &errorMessages ) ) { setGridFileName( fileNames[i] ); gridFileName = fileNames[i]; QFileInfo gridFileName( fileNames[i] ); QString caseName = gridFileName.completeBaseName(); this->setCaseUserDescription( caseName ); this->eclipseCaseData()->mainGrid()->setFlipAxis( m_flipXAxis, m_flipYAxis ); computeCachedData(); break; } else { allErrorMessages.push_back( errorMessages ); } } } if ( this->eclipseCaseData()->mainGrid()->gridPointDimensions() == cvf::Vec3st( 0, 0, 0 ) ) { if ( !allErrorMessages.empty() ) { for ( QString errorMessages : allErrorMessages ) { RiaLogging::error( errorMessages ); } } return false; // No grid present } std::vector filesToRead; for ( const QString& filename : fileNames ) { if ( filename == gridFileName ) continue; bool exists = false; for ( const QString& currentFileName : additionalFiles() ) { if ( filename == currentFileName ) { exists = true; break; } } if ( !exists ) { filesToRead.push_back( filename ); } } RifInputPropertyLoader::loadAndSynchronizeInputProperties( m_inputPropertyCollection, this->eclipseCaseData(), filesToRead, importFaults ); if ( importFaults ) { this->ensureFaultDataIsComputed(); } results( RiaDefines::PorosityModelType::MATRIX_MODEL )->createPlaceholderResultEntries(); return true; } //-------------------------------------------------------------------------------------------------- /// //-------------------------------------------------------------------------------------------------- bool RimEclipseInputCase::openEclipseGridFile() { // Early exit if reservoir data is created if ( this->eclipseCaseData() == nullptr ) { cvf::ref readerInterface; if ( gridFileName().contains( RiaDefines::mockModelBasicInputCase() ) ) { readerInterface = this->createMockModel( gridFileName() ); } else { readerInterface = new RifReaderEclipseInput; cvf::ref eclipseCase = new RigEclipseCaseData( this ); if ( !readerInterface->open( gridFileName(), eclipseCase.p() ) ) { return false; } this->setReservoirData( eclipseCase.p() ); } CVF_ASSERT( this->eclipseCaseData() ); CVF_ASSERT( readerInterface.notNull() ); results( RiaDefines::PorosityModelType::MATRIX_MODEL )->setReaderInterface( readerInterface.p() ); results( RiaDefines::PorosityModelType::FRACTURE_MODEL )->setReaderInterface( readerInterface.p() ); this->eclipseCaseData()->mainGrid()->setFlipAxis( m_flipXAxis, m_flipYAxis ); loadAndSynchronizeInputProperties( true ); computeCachedData(); } if ( RiaPreferences::current()->autocomputeDepthRelatedProperties ) { results( RiaDefines::PorosityModelType::MATRIX_MODEL )->computeDepthRelatedResults(); results( RiaDefines::PorosityModelType::FRACTURE_MODEL )->computeDepthRelatedResults(); } results( RiaDefines::PorosityModelType::MATRIX_MODEL )->computeCellVolumes(); return true; } //-------------------------------------------------------------------------------------------------- /// //-------------------------------------------------------------------------------------------------- void RimEclipseInputCase::reloadEclipseGridFile() { setReservoirData( nullptr ); openReserviorCase(); } //-------------------------------------------------------------------------------------------------- /// //-------------------------------------------------------------------------------------------------- cvf::ref RimEclipseInputCase::createMockModel( QString modelName ) { cvf::ref reservoir = new RigEclipseCaseData( this ); cvf::ref mockFileInterface = new RifReaderMockModel; if ( modelName == RiaDefines::mockModelBasicInputCase() ) { setGridFileName( modelName ); // Create the mock file interface and and RigSerervoir and set them up. mockFileInterface->setWorldCoordinates( cvf::Vec3d( 10, 10, 10 ), cvf::Vec3d( 20, 20, 20 ) ); mockFileInterface->setGridPointDimensions( cvf::Vec3st( 4, 5, 6 ) ); mockFileInterface->addLocalGridRefinement( cvf::Vec3st( 0, 2, 2 ), cvf::Vec3st( 0, 2, 2 ), cvf::Vec3st( 3, 3, 3 ) ); mockFileInterface->setResultInfo( 3, 10 ); mockFileInterface->open( "", reservoir.p() ); { // size_t idx = reservoir->mainGrid()->cellIndexFromIJK(1, 3, 4); // TODO: Rewrite active cell info in mock models // reservoir->mainGrid()->cell(idx).setActiveIndexInMatrixModel(cvf::UNDEFINED_SIZE_T); } { // size_t idx = reservoir->mainGrid()->cellIndexFromIJK(2, 2, 3); // TODO: Rewrite active cell info in mock models // reservoir->mainGrid()->cell(idx).setActiveIndexInMatrixModel(cvf::UNDEFINED_SIZE_T); } // Add a property RimEclipseInputProperty* inputProperty = new RimEclipseInputProperty; inputProperty->resultName = "PORO"; inputProperty->eclipseKeyword = "PORO"; inputProperty->fileName = QString( "PORO.prop" ); m_inputPropertyCollection->inputProperties.push_back( inputProperty ); } this->setReservoirData( reservoir.p() ); return mockFileInterface.p(); } //-------------------------------------------------------------------------------------------------- /// //-------------------------------------------------------------------------------------------------- void RimEclipseInputCase::defineUiOrdering( QString uiConfigName, caf::PdmUiOrdering& uiOrdering ) { uiOrdering.add( &m_caseUserDescription ); uiOrdering.add( &m_displayNameOption ); uiOrdering.add( &m_caseId ); uiOrdering.add( &m_caseFileName ); uiOrdering.add( &m_additionalFiles ); auto group = uiOrdering.addNewGroup( "Case Options" ); group->add( &m_activeFormationNames ); group->add( &m_flipXAxis ); group->add( &m_flipYAxis ); } //-------------------------------------------------------------------------------------------------- /// //-------------------------------------------------------------------------------------------------- QString RimEclipseInputCase::locationOnDisc() const { if ( gridFileName().isEmpty() ) return QString(); QFileInfo fi( gridFileName() ); return fi.absolutePath(); } //-------------------------------------------------------------------------------------------------- /// //-------------------------------------------------------------------------------------------------- void RimEclipseInputCase::updateAdditionalFileFolder( const QString& newFolder ) { QDir newDir( newFolder ); for ( RimEclipseInputProperty* inputProperty : m_inputPropertyCollection()->inputProperties() ) { if ( inputProperty->fileName == gridFileName() ) continue; QFileInfo oldFilePath( inputProperty->fileName().path() ); QFileInfo newFilePath( newDir, oldFilePath.fileName() ); inputProperty->fileName = newFilePath.absoluteFilePath(); } }