///////////////////////////////////////////////////////////////////////////////// // // Copyright (C) 2018- 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 "RimWellPathTieIn.h" #include "RigWellPath.h" #include "RimFileWellPath.h" #include "RimModeledWellPath.h" #include "RimTools.h" #include "RimWellPathCollection.h" #include "RimWellPathGeometryDef.h" #include "RimWellPathTarget.h" #include "RimWellPathValve.h" #include "RiuMainWindow.h" #include "cafPdmFieldScriptingCapability.h" #include "cafPdmObjectScriptingCapability.h" #include "cafPdmUiDoubleValueEditor.h" CAF_PDM_SOURCE_INIT( RimWellPathTieIn, "RimWellPathTieIn" ); //-------------------------------------------------------------------------------------------------- /// //-------------------------------------------------------------------------------------------------- RimWellPathTieIn::RimWellPathTieIn() { CAF_PDM_InitObject( "Well Path Tie In", ":/NotDefined.png", "", "Well Path Tie In description" ); CAF_PDM_InitFieldNoDefault( &m_parentWell, "ParentWellPath", "Parent Well Path", "", "", "" ); CAF_PDM_InitFieldNoDefault( &m_childWell, "ChildWellPath", "ChildWellPath", "", "", "" ); CAF_PDM_InitFieldNoDefault( &m_tieInMeasuredDepth, "TieInMeasuredDepth", "Tie In Measured Depth", "", "", "" ); m_tieInMeasuredDepth.uiCapability()->setUiEditorTypeName( caf::PdmUiDoubleValueEditor::uiEditorTypeName() ); CAF_PDM_InitScriptableField( &m_addValveAtConnection, "AddValveAtConnection", false, "Add Outlet Valve for Branches", "", "", "" ); CAF_PDM_InitScriptableFieldNoDefault( &m_valve, "Valve", "Branch Outlet Valve", "", "", "" ); m_valve = new RimWellPathValve; } //-------------------------------------------------------------------------------------------------- /// //-------------------------------------------------------------------------------------------------- void RimWellPathTieIn::connectWellPaths( RimWellPath* parentWell, RimWellPath* childWell, double tieInMeasuredDepth ) { m_parentWell = parentWell; m_childWell = childWell; m_tieInMeasuredDepth = tieInMeasuredDepth; } //-------------------------------------------------------------------------------------------------- /// //-------------------------------------------------------------------------------------------------- RimWellPath* RimWellPathTieIn::parentWell() const { return m_parentWell(); } //-------------------------------------------------------------------------------------------------- /// //-------------------------------------------------------------------------------------------------- double RimWellPathTieIn::tieInMeasuredDepth() const { return m_tieInMeasuredDepth(); } //-------------------------------------------------------------------------------------------------- /// //-------------------------------------------------------------------------------------------------- RimWellPath* RimWellPathTieIn::childWell() const { return m_childWell(); } //-------------------------------------------------------------------------------------------------- /// //-------------------------------------------------------------------------------------------------- void RimWellPathTieIn::updateChildWellGeometry() { auto modeledWellPath = dynamic_cast( m_childWell() ); if ( modeledWellPath ) { modeledWellPath->updateTieInLocationFromParentWell(); } } //-------------------------------------------------------------------------------------------------- /// //-------------------------------------------------------------------------------------------------- void RimWellPathTieIn::updateFirstTargetFromParentWell() { auto parentWellPath = m_parentWell(); if ( !parentWellPath ) return; auto modeledWellPath = dynamic_cast( m_childWell() ); if ( modeledWellPath && modeledWellPath->geometryDefinition() && parentWellPath->wellPathGeometry() && parentWellPath->wellPathGeometry()->measuredDepths().size() > 2 ) { auto [pointVector, measuredDepths] = parentWellPath->wellPathGeometry() ->clippedPointSubset( parentWellPath->wellPathGeometry()->measuredDepths().front(), m_tieInMeasuredDepth ); if ( pointVector.size() < 2u ) return; RimWellPathTarget* newTarget = nullptr; if ( modeledWellPath->geometryDefinition()->activeWellTargets().empty() ) { newTarget = modeledWellPath->geometryDefinition()->appendTarget(); } else { newTarget = modeledWellPath->geometryDefinition()->activeWellTargets().front(); } auto lastPoint = pointVector.back(); auto tangent = lastPoint - pointVector[pointVector.size() - 2]; newTarget->setAsPointXYZAndTangentTarget( { lastPoint[0], lastPoint[1], lastPoint[2] }, tangent ); } } //-------------------------------------------------------------------------------------------------- /// //-------------------------------------------------------------------------------------------------- const RimWellPathValve* RimWellPathTieIn::outletValve() const { return m_addValveAtConnection() && m_valve() && m_valve->valveTemplate() ? m_valve() : nullptr; } //-------------------------------------------------------------------------------------------------- /// //-------------------------------------------------------------------------------------------------- void RimWellPathTieIn::defineUiOrdering( QString uiConfigName, caf::PdmUiOrdering& uiOrdering ) { auto tieInGroup = uiOrdering.addNewGroup( "Tie In Settings" ); tieInGroup->add( &m_parentWell ); if ( m_parentWell() != nullptr ) { tieInGroup->add( &m_tieInMeasuredDepth ); tieInGroup->add( &m_addValveAtConnection ); bool isFileWellPath = dynamic_cast( m_childWell() ); m_tieInMeasuredDepth.uiCapability()->setUiReadOnly( isFileWellPath ); // Display only ICV valves m_valve->setComponentTypeFilter( { RiaDefines::WellPathComponentType::ICV } ); if ( m_addValveAtConnection ) { m_valve->uiOrdering( "TemplateOnly", *tieInGroup ); } } uiOrdering.skipRemainingFields(); } //-------------------------------------------------------------------------------------------------- /// //-------------------------------------------------------------------------------------------------- void RimWellPathTieIn::fieldChangedByUi( const caf::PdmFieldHandle* changedField, const QVariant& oldValue, const QVariant& newValue ) { if ( changedField == &m_parentWell ) { updateFirstTargetFromParentWell(); RimTools::wellPathCollection()->rebuildWellPathNodes(); } updateChildWellGeometry(); // Update all well paths to make sure the visibility of completion settings is updated // Completions settings is only visible for top-level wells, not for tie-in wells RimTools::wellPathCollection()->updateAllRequiredEditors(); if ( changedField == &m_parentWell && m_childWell ) { RiuMainWindow::instance()->setExpanded( m_childWell ); RiuMainWindow::instance()->selectAsCurrentItem( m_childWell ); } } //-------------------------------------------------------------------------------------------------- /// //-------------------------------------------------------------------------------------------------- QList RimWellPathTieIn::calculateValueOptions( const caf::PdmFieldHandle* fieldNeedingOptions, bool* useOptionsOnly ) { QList options; if ( fieldNeedingOptions == &m_parentWell ) { std::vector wellPathsToExclude = { m_childWell() }; RimTools::wellPathOptionItemsSubset( wellPathsToExclude, &options ); options.push_front( caf::PdmOptionItemInfo( "None", nullptr ) ); } return options; }