///////////////////////////////////////////////////////////////////////////////// // // 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 // for more details. // ///////////////////////////////////////////////////////////////////////////////// #include "RimcWellPath.h" #include "FractureCommands/RicPlaceThermalFractureUsingTemplateDataFeature.h" #include "RiaLogging.h" #include "RimEclipseCase.h" #include "RimEclipseCaseTools.h" #include "RimPerforationCollection.h" #include "RimPerforationInterval.h" #include "RimStimPlanFractureTemplate.h" #include "RimStimPlanModel.h" #include "RimThermalFractureTemplate.h" #include "RimTools.h" #include "RimWellPath.h" #include "RimWellPathCollection.h" #include "RimWellPathFracture.h" #include "RigStimPlanModelTools.h" #include "FractureCommands/RicNewWellPathFractureFeature.h" #include "cafPdmAbstractFieldScriptingCapability.h" #include "cafPdmFieldScriptingCapability.h" CAF_PDM_OBJECT_METHOD_SOURCE_INIT( RimWellPath, RimcWellPath_addFracture, "AddFracture" ); //-------------------------------------------------------------------------------------------------- /// //-------------------------------------------------------------------------------------------------- RimcWellPath_addFracture::RimcWellPath_addFracture( caf::PdmObjectHandle* self ) : caf::PdmObjectMethod( self ) { CAF_PDM_InitObject( "Add StimPlan Fracture", "", "", "Add StimPlan Fracture" ); CAF_PDM_InitScriptableField( &m_md, "MeasuredDepth", 0.0, "Measured Depth" ); CAF_PDM_InitScriptableFieldNoDefault( &m_stimPlanFractureTemplate, "StimPlanFractureTemplate", "", "", "", "StimPlan Fracture Template" ); CAF_PDM_InitScriptableField( &m_alignDip, "AlignDip", false, "Align Dip" ); CAF_PDM_InitScriptableFieldNoDefault( &m_eclipseCase, "EclipseCase", "", "", "", "Eclipse Case" ); } //-------------------------------------------------------------------------------------------------- /// //-------------------------------------------------------------------------------------------------- caf::PdmObjectHandle* RimcWellPath_addFracture::execute() { auto wellPath = self(); RimWellPathFracture* wellPathFracture = RicNewWellPathFractureFeature::addFracture( wellPath, m_md() ); if ( m_stimPlanFractureTemplate ) wellPathFracture->setFractureTemplate( m_stimPlanFractureTemplate() ); if ( m_alignDip ) { if ( m_eclipseCase && m_eclipseCase->eclipseCaseData() ) { RiaLogging::info( "Computing formation dip for fracture alignment" ); double boundingBoxHorizontal = 50.0; double boundingBoxVertical = 100.0; cvf::Vec3d position = wellPathFracture->anchorPosition(); cvf::Vec3d direction = RigStimPlanModelTools::calculateTSTDirection( m_eclipseCase->eclipseCaseData(), position, boundingBoxHorizontal, boundingBoxVertical ); RiaLogging::info( QString( "Direction: %1 %2 %3" ).arg( direction.x() ).arg( direction.y() ).arg( direction.z() ) ); cvf::Vec3d fractureDirectionNormal = wellPathFracture->computeFractureDirectionNormal(); cvf::Vec3d formationDirection = RimStimPlanModel::projectVectorIntoFracturePlane( position, fractureDirectionNormal, direction ); if ( !formationDirection.isUndefined() ) { double formationDip = RigStimPlanModelTools::calculateFormationDip( formationDirection ) - 90.0; RiaLogging::info( QString( "Computed formation dip: %1" ).arg( formationDip ) ); wellPathFracture->setDip( formationDip ); } } else { RiaLogging::error( "No eclipse case found. Fracture not aligned with formation dip." ); } } return wellPathFracture; } //-------------------------------------------------------------------------------------------------- /// //-------------------------------------------------------------------------------------------------- bool RimcWellPath_addFracture::resultIsPersistent() const { return true; } //-------------------------------------------------------------------------------------------------- /// //-------------------------------------------------------------------------------------------------- std::unique_ptr RimcWellPath_addFracture::defaultResult() const { return std::unique_ptr( new RimWellPathFracture ); } CAF_PDM_OBJECT_METHOD_SOURCE_INIT( RimWellPath, RimcWellPath_addThermalFracture, "AddThermalFracture" ); //-------------------------------------------------------------------------------------------------- /// //-------------------------------------------------------------------------------------------------- RimcWellPath_addThermalFracture::RimcWellPath_addThermalFracture( caf::PdmObjectHandle* self ) : caf::PdmObjectMethod( self ) { CAF_PDM_InitObject( "Add Thermal Fracture", "", "", "Add Thermal Fracture" ); CAF_PDM_InitScriptableField( &m_md, "MeasuredDepth", 0.0, "Measured Depth" ); CAF_PDM_InitScriptableFieldNoDefault( &m_fractureTemplate, "FractureTemplate", "", "", "", "Thermal Fracture Template" ); CAF_PDM_InitScriptableField( &m_placeUsingTemplateData, "PlaceUsingTemplateData", true, "Place using template data" ); } //-------------------------------------------------------------------------------------------------- /// //-------------------------------------------------------------------------------------------------- caf::PdmObjectHandle* RimcWellPath_addThermalFracture::execute() { auto wellPath = self(); RimWellPathFracture* wellPathFracture = RicNewWellPathFractureFeature::addFracture( wellPath, m_md() ); if ( m_fractureTemplate ) { wellPathFracture->setFractureTemplate( m_fractureTemplate() ); } if ( m_placeUsingTemplateData ) { RicPlaceThermalFractureUsingTemplateDataFeature::placeUsingTemplateData( wellPathFracture ); } return wellPathFracture; } //-------------------------------------------------------------------------------------------------- /// //-------------------------------------------------------------------------------------------------- bool RimcWellPath_addThermalFracture::resultIsPersistent() const { return true; } //-------------------------------------------------------------------------------------------------- /// //-------------------------------------------------------------------------------------------------- std::unique_ptr RimcWellPath_addThermalFracture::defaultResult() const { return std::unique_ptr( new RimWellPathFracture ); } CAF_PDM_OBJECT_METHOD_SOURCE_INIT( RimWellPath, RimcWellPath_appendPerforationInterval, "AppendPerforationInterval" ); //-------------------------------------------------------------------------------------------------- /// //-------------------------------------------------------------------------------------------------- RimcWellPath_appendPerforationInterval::RimcWellPath_appendPerforationInterval( caf::PdmObjectHandle* self ) : caf::PdmObjectMethod( self ) { CAF_PDM_InitObject( "Append Perforation Interval", "", "", "Append Perforation Interval" ); CAF_PDM_InitScriptableField( &m_startMD, "StartMd", 0.0, "", "", "", "Start Measured Depth" ); CAF_PDM_InitScriptableField( &m_endMD, "EndMd", 0.0, "", "", "", "End Measured Depth" ); CAF_PDM_InitScriptableField( &m_diameter, "Diameter", 0.0, "", "", "", "Diameter" ); CAF_PDM_InitScriptableField( &m_skinFactor, "SkinFactor", 0.0, "", "", "", "Skin Factor" ); } //-------------------------------------------------------------------------------------------------- /// //-------------------------------------------------------------------------------------------------- caf::PdmObjectHandle* RimcWellPath_appendPerforationInterval::execute() { auto wellPath = self(); auto perforationInterval = new RimPerforationInterval; perforationInterval->setStartAndEndMD( m_startMD, m_endMD ); perforationInterval->setSkinFactor( m_skinFactor ); perforationInterval->setDiameter( m_diameter ); wellPath->perforationIntervalCollection()->appendPerforation( perforationInterval ); auto* wellPathCollection = RimTools::wellPathCollection(); wellPathCollection->uiCapability()->updateConnectedEditors(); wellPathCollection->scheduleRedrawAffectedViews(); return perforationInterval; } //-------------------------------------------------------------------------------------------------- /// //-------------------------------------------------------------------------------------------------- bool RimcWellPath_appendPerforationInterval::resultIsPersistent() const { return true; } //-------------------------------------------------------------------------------------------------- /// //-------------------------------------------------------------------------------------------------- std::unique_ptr RimcWellPath_appendPerforationInterval::defaultResult() const { return std::unique_ptr( new RimPerforationInterval ); }