///////////////////////////////////////////////////////////////////////////////// // // 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 // for more details. // ///////////////////////////////////////////////////////////////////////////////// #include "RigSimulationWellCoordsAndMD.h" #include "cvfGeometryTools.h" #include "cvfMath.h" //-------------------------------------------------------------------------------------------------- /// //-------------------------------------------------------------------------------------------------- RigSimulationWellCoordsAndMD::RigSimulationWellCoordsAndMD( const std::vector& wellPathPoints ) : m_wellPathPoints( wellPathPoints ) { computeMeasuredDepths(); } //-------------------------------------------------------------------------------------------------- /// //-------------------------------------------------------------------------------------------------- const std::vector& RigSimulationWellCoordsAndMD::wellPathPoints() const { return m_wellPathPoints; } //-------------------------------------------------------------------------------------------------- /// //-------------------------------------------------------------------------------------------------- const std::vector& RigSimulationWellCoordsAndMD::measuredDepths() const { return m_measuredDepths; } //-------------------------------------------------------------------------------------------------- /// //-------------------------------------------------------------------------------------------------- cvf::Vec3d RigSimulationWellCoordsAndMD::interpolatedPointAlongWellPath( double measuredDepth ) const { cvf::Vec3d wellPathPoint = cvf::Vec3d::ZERO; size_t i = 0; while ( i < m_measuredDepths.size() && m_measuredDepths.at( i ) < measuredDepth ) { i++; } if ( m_measuredDepths.size() > i ) { if ( i == 0 ) { // For measuredDepth same or lower than first point, use this first point wellPathPoint = m_wellPathPoints.at( 0 ); } else { // Do interpolation double stepsize = ( measuredDepth - m_measuredDepths.at( i - 1 ) ) / ( m_measuredDepths.at( i ) - m_measuredDepths.at( i - 1 ) ); wellPathPoint = m_wellPathPoints.at( i - 1 ) + stepsize * ( m_wellPathPoints.at( i ) - m_wellPathPoints.at( i - 1 ) ); } } else { // Use endpoint if measuredDepth same or higher than last point wellPathPoint = m_wellPathPoints.at( i - 1 ); } return wellPathPoint; } //-------------------------------------------------------------------------------------------------- /// //-------------------------------------------------------------------------------------------------- double RigSimulationWellCoordsAndMD::locationAlongWellCoords( const cvf::Vec3d& position ) const { double location = 0.0; size_t closestIndex = findClosestIndex( position ); if ( closestIndex != cvf::UNDEFINED_SIZE_T ) { cvf::Vec3d p1 = m_wellPathPoints[closestIndex - 1]; cvf::Vec3d p2 = m_wellPathPoints[closestIndex - 0]; double intersection = 0.0; cvf::GeometryTools::projectPointOnLine( p1, p2, position, &intersection ); location = m_measuredDepths[closestIndex - 1]; location += intersection * ( p1 - p2 ).length(); } return location; } //-------------------------------------------------------------------------------------------------- /// //-------------------------------------------------------------------------------------------------- size_t RigSimulationWellCoordsAndMD::findClosestIndex( const cvf::Vec3d& position ) const { size_t closestIndex = cvf::UNDEFINED_SIZE_T; double closestDistance = cvf::UNDEFINED_DOUBLE; for ( size_t i = 1; i < m_wellPathPoints.size(); i++ ) { cvf::Vec3d p1 = m_wellPathPoints[i - 1]; cvf::Vec3d p2 = m_wellPathPoints[i - 0]; double candidateDistance = cvf::GeometryTools::linePointSquareDist( p1, p2, position ); if ( candidateDistance < closestDistance ) { closestDistance = candidateDistance; closestIndex = i; } } return closestIndex; } //-------------------------------------------------------------------------------------------------- /// //-------------------------------------------------------------------------------------------------- double RigSimulationWellCoordsAndMD::simWellAzimuthAngle( const cvf::Vec3d& position ) const { size_t closestIndex = findClosestIndex( position ); // For vertical well (x-component of direction = 0) returned angle will be 90. double azimuthAngle = 90.0; if ( closestIndex != cvf::UNDEFINED_DOUBLE ) { cvf::Vec3d p1; cvf::Vec3d p2; if ( closestIndex > 0 ) { p1 = m_wellPathPoints[closestIndex - 1]; p2 = m_wellPathPoints[closestIndex - 0]; } else { p1 = m_wellPathPoints[closestIndex + 1]; p2 = m_wellPathPoints[closestIndex + 0]; } cvf::Vec3d direction = p2 - p1; if ( fabs( direction.y() ) > 1e-5 ) { double atanValue = direction.x() / direction.y(); azimuthAngle = atan( atanValue ); azimuthAngle = cvf::Math::toDegrees( azimuthAngle ); } } return azimuthAngle; } //-------------------------------------------------------------------------------------------------- /// //-------------------------------------------------------------------------------------------------- double RigSimulationWellCoordsAndMD::simWellDipAngle( const cvf::Vec3d& position ) const { size_t closestIndex = findClosestIndex( position ); double dipAngle = 0.0; if ( closestIndex != cvf::UNDEFINED_DOUBLE ) { cvf::Vec3d p1; cvf::Vec3d p2; if ( closestIndex > 0 ) { p1 = m_wellPathPoints[closestIndex - 1]; p2 = m_wellPathPoints[closestIndex - 0]; } else { p1 = m_wellPathPoints[closestIndex + 1]; p2 = m_wellPathPoints[closestIndex + 0]; } cvf::Vec3d direction = p1 - p2; double horizonal = sqrt( pow( direction.x(), 2 ) + pow( direction.y(), 2 ) ); double vertical = direction.z(); if ( fabs( vertical ) > 1e-5 ) { double atanValue = vertical / horizonal; dipAngle = atan( atanValue ); dipAngle = cvf::Math::toDegrees( dipAngle ); } } return dipAngle; } //-------------------------------------------------------------------------------------------------- /// //-------------------------------------------------------------------------------------------------- void RigSimulationWellCoordsAndMD::computeMeasuredDepths() { cvf::Vec3d prev = cvf::Vec3d::UNDEFINED; double accumulatedMD = 0; for ( const auto& point : m_wellPathPoints ) { if ( !prev.isUndefined() ) { accumulatedMD += point.pointDistance( prev ); } m_measuredDepths.push_back( accumulatedMD ); prev = point; } }