2017-02-07 02:09:00 -06:00
|
|
|
/////////////////////////////////////////////////////////////////////////////////
|
|
|
|
//
|
|
|
|
// 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 <http://www.gnu.org/licenses/gpl.html>
|
|
|
|
// for more details.
|
|
|
|
//
|
|
|
|
/////////////////////////////////////////////////////////////////////////////////
|
|
|
|
|
|
|
|
#include "RigStimPlanFractureDefinition.h"
|
|
|
|
|
2018-02-06 05:42:13 -06:00
|
|
|
#include "RiaFractureDefines.h"
|
2017-06-13 07:44:04 -05:00
|
|
|
#include "RiaLogging.h"
|
2018-02-14 06:34:38 -06:00
|
|
|
|
2018-01-19 02:10:30 -06:00
|
|
|
#include "RigFractureCell.h"
|
2017-06-13 07:44:04 -05:00
|
|
|
#include "RigFractureGrid.h"
|
2018-01-19 02:10:30 -06:00
|
|
|
#include "RigStatisticsMath.h"
|
|
|
|
|
|
|
|
#include "RivWellFracturePartMgr.h"
|
|
|
|
|
|
|
|
#include "cvfMath.h"
|
|
|
|
|
2017-06-26 04:05:46 -05:00
|
|
|
#include <cmath>
|
2017-02-20 02:31:24 -06:00
|
|
|
|
2017-02-17 05:17:57 -06:00
|
|
|
//--------------------------------------------------------------------------------------------------
|
2018-02-08 04:05:01 -06:00
|
|
|
/// Internal functions
|
2017-02-17 05:17:57 -06:00
|
|
|
//--------------------------------------------------------------------------------------------------
|
2018-02-08 04:05:01 -06:00
|
|
|
size_t findMirrorXIndex(std::vector<double> xs);
|
2017-02-17 05:17:57 -06:00
|
|
|
|
2018-02-08 04:05:01 -06:00
|
|
|
//--------------------------------------------------------------------------------------------------
|
|
|
|
///
|
|
|
|
//--------------------------------------------------------------------------------------------------
|
|
|
|
const double RigStimPlanFractureDefinition::THRESHOLD_VALUE = 1e-5;
|
2017-02-17 05:17:57 -06:00
|
|
|
|
2017-02-07 02:09:00 -06:00
|
|
|
//--------------------------------------------------------------------------------------------------
|
2018-08-02 12:22:03 -05:00
|
|
|
///
|
2017-02-07 02:09:00 -06:00
|
|
|
//--------------------------------------------------------------------------------------------------
|
2018-08-02 12:22:03 -05:00
|
|
|
RigStimPlanFractureDefinition::RigStimPlanFractureDefinition()
|
|
|
|
: m_unitSet(RiaEclipseUnitTools::UNITS_UNKNOWN)
|
|
|
|
, m_topPerfTvd(HUGE_VAL)
|
|
|
|
, m_bottomPerfTvd(HUGE_VAL)
|
|
|
|
, m_xMirrorMode(false)
|
2017-02-07 02:09:00 -06:00
|
|
|
{
|
|
|
|
}
|
|
|
|
|
|
|
|
//--------------------------------------------------------------------------------------------------
|
|
|
|
///
|
|
|
|
//--------------------------------------------------------------------------------------------------
|
|
|
|
RigStimPlanFractureDefinition::~RigStimPlanFractureDefinition()
|
|
|
|
{
|
|
|
|
|
|
|
|
}
|
2017-02-15 03:32:34 -06:00
|
|
|
|
2018-02-14 06:34:38 -06:00
|
|
|
//--------------------------------------------------------------------------------------------------
|
|
|
|
///
|
|
|
|
//--------------------------------------------------------------------------------------------------
|
|
|
|
RiaEclipseUnitTools::UnitSystem RigStimPlanFractureDefinition::unitSet() const
|
|
|
|
{
|
|
|
|
return m_unitSet;
|
|
|
|
}
|
|
|
|
|
|
|
|
//--------------------------------------------------------------------------------------------------
|
|
|
|
///
|
|
|
|
//--------------------------------------------------------------------------------------------------
|
|
|
|
size_t RigStimPlanFractureDefinition::xCount() const
|
|
|
|
{
|
|
|
|
return m_Xs.size();
|
|
|
|
}
|
|
|
|
|
|
|
|
//--------------------------------------------------------------------------------------------------
|
|
|
|
///
|
|
|
|
//--------------------------------------------------------------------------------------------------
|
|
|
|
size_t RigStimPlanFractureDefinition::yCount() const
|
|
|
|
{
|
|
|
|
return m_Ys.size();
|
|
|
|
}
|
|
|
|
|
|
|
|
//--------------------------------------------------------------------------------------------------
|
|
|
|
///
|
|
|
|
//--------------------------------------------------------------------------------------------------
|
|
|
|
double RigStimPlanFractureDefinition::minDepth() const
|
|
|
|
{
|
|
|
|
return -minY();
|
|
|
|
}
|
|
|
|
|
|
|
|
//--------------------------------------------------------------------------------------------------
|
|
|
|
///
|
|
|
|
//--------------------------------------------------------------------------------------------------
|
|
|
|
double RigStimPlanFractureDefinition::maxDepth() const
|
|
|
|
{
|
|
|
|
return -maxY();
|
|
|
|
}
|
|
|
|
|
|
|
|
//--------------------------------------------------------------------------------------------------
|
|
|
|
///
|
|
|
|
//--------------------------------------------------------------------------------------------------
|
|
|
|
double RigStimPlanFractureDefinition::topPerfTvd() const
|
|
|
|
{
|
|
|
|
return m_topPerfTvd;
|
|
|
|
}
|
|
|
|
|
|
|
|
//--------------------------------------------------------------------------------------------------
|
|
|
|
///
|
|
|
|
//--------------------------------------------------------------------------------------------------
|
|
|
|
double RigStimPlanFractureDefinition::bottomPerfTvd() const
|
|
|
|
{
|
|
|
|
return m_bottomPerfTvd;
|
|
|
|
}
|
|
|
|
|
|
|
|
//--------------------------------------------------------------------------------------------------
|
|
|
|
///
|
|
|
|
//--------------------------------------------------------------------------------------------------
|
|
|
|
double RigStimPlanFractureDefinition::minY() const
|
|
|
|
{
|
|
|
|
return m_Ys[0];
|
|
|
|
}
|
|
|
|
|
|
|
|
//--------------------------------------------------------------------------------------------------
|
|
|
|
///
|
|
|
|
//--------------------------------------------------------------------------------------------------
|
|
|
|
double RigStimPlanFractureDefinition::maxY() const
|
|
|
|
{
|
|
|
|
return m_Ys.back();
|
|
|
|
}
|
|
|
|
|
2018-03-01 07:16:08 -06:00
|
|
|
//--------------------------------------------------------------------------------------------------
|
|
|
|
///
|
|
|
|
//--------------------------------------------------------------------------------------------------
|
|
|
|
void RigStimPlanFractureDefinition::scaleXs(double scaleFactor)
|
|
|
|
{
|
|
|
|
// Scale using 0 as scaling anchor
|
|
|
|
for (double& x : m_Xs)
|
|
|
|
{
|
|
|
|
x *= scaleFactor;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
//--------------------------------------------------------------------------------------------------
|
|
|
|
///
|
|
|
|
//--------------------------------------------------------------------------------------------------
|
|
|
|
void RigStimPlanFractureDefinition::scaleYs(double scaleFactor, double wellPathIntersectionY)
|
|
|
|
{
|
|
|
|
// Scale using wellPathIntersectionY as scaling anchor
|
|
|
|
for (double& y : m_Ys)
|
|
|
|
{
|
|
|
|
y = (y - wellPathIntersectionY) * scaleFactor + wellPathIntersectionY;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-01-31 05:25:58 -06:00
|
|
|
//--------------------------------------------------------------------------------------------------
|
2018-03-05 02:43:06 -06:00
|
|
|
///
|
2018-01-31 05:25:58 -06:00
|
|
|
//--------------------------------------------------------------------------------------------------
|
2018-02-20 08:35:08 -06:00
|
|
|
void RigStimPlanFractureDefinition::setTvdToTopPerf(double topPerfTvd)
|
2018-01-31 05:25:58 -06:00
|
|
|
{
|
2018-02-20 08:35:08 -06:00
|
|
|
m_topPerfTvd = topPerfTvd;
|
2018-01-31 05:25:58 -06:00
|
|
|
}
|
|
|
|
|
|
|
|
//--------------------------------------------------------------------------------------------------
|
|
|
|
///
|
|
|
|
//--------------------------------------------------------------------------------------------------
|
2018-02-20 08:35:08 -06:00
|
|
|
void RigStimPlanFractureDefinition::setTvdToBottomPerf(double bottomPerfTvd)
|
2018-01-31 05:25:58 -06:00
|
|
|
{
|
2018-02-20 08:35:08 -06:00
|
|
|
m_bottomPerfTvd = bottomPerfTvd;
|
2018-01-31 05:25:58 -06:00
|
|
|
}
|
|
|
|
|
2017-06-13 08:04:57 -05:00
|
|
|
//--------------------------------------------------------------------------------------------------
|
|
|
|
///
|
|
|
|
//--------------------------------------------------------------------------------------------------
|
2018-02-08 04:05:01 -06:00
|
|
|
void RigStimPlanFractureDefinition::generateXsFromFileXs(bool xMirrorMode)
|
2017-06-13 08:04:57 -05:00
|
|
|
{
|
2018-02-08 04:05:01 -06:00
|
|
|
m_xMirrorMode = xMirrorMode;
|
|
|
|
m_Xs.clear();
|
|
|
|
|
|
|
|
if (m_xMirrorMode)
|
2017-06-13 08:04:57 -05:00
|
|
|
{
|
2018-02-08 04:05:01 -06:00
|
|
|
size_t mirrorIndex = findMirrorXIndex(m_fileXs);
|
|
|
|
std::list<double> xs;
|
|
|
|
|
|
|
|
// Mirror positive X values
|
|
|
|
xs.push_back(m_fileXs[mirrorIndex]);
|
|
|
|
for (size_t i = mirrorIndex + 1; i < m_fileXs.size(); i++)
|
2017-06-13 08:04:57 -05:00
|
|
|
{
|
2018-02-08 04:05:01 -06:00
|
|
|
xs.push_front(-m_fileXs[i]);
|
|
|
|
xs.push_back(m_fileXs[i]);
|
2017-06-13 08:04:57 -05:00
|
|
|
}
|
2018-02-08 04:05:01 -06:00
|
|
|
m_Xs = std::vector<double>(xs.begin(), xs.end());
|
2017-06-13 08:04:57 -05:00
|
|
|
}
|
2018-02-08 04:05:01 -06:00
|
|
|
else
|
2017-06-13 08:04:57 -05:00
|
|
|
{
|
2018-02-08 04:05:01 -06:00
|
|
|
m_Xs = m_fileXs;
|
2017-06-13 08:04:57 -05:00
|
|
|
}
|
2018-02-08 04:05:01 -06:00
|
|
|
}
|
|
|
|
|
|
|
|
//--------------------------------------------------------------------------------------------------
|
|
|
|
///
|
|
|
|
//--------------------------------------------------------------------------------------------------
|
2018-08-10 07:12:14 -05:00
|
|
|
std::vector<std::vector<double>> RigStimPlanFractureDefinition::generateDataLayoutFromFileDataLayout(std::vector<std::vector<double>> fileXYData) const
|
2018-02-08 04:05:01 -06:00
|
|
|
{
|
|
|
|
if (m_xMirrorMode)
|
|
|
|
{
|
|
|
|
std::vector<std::vector<double>> xyData;
|
|
|
|
size_t mirrorIndex = findMirrorXIndex(m_fileXs);
|
2017-06-13 08:04:57 -05:00
|
|
|
|
2018-02-08 04:05:01 -06:00
|
|
|
for (const auto& yData : fileXYData)
|
|
|
|
{
|
|
|
|
std::list<double> xValues;
|
|
|
|
|
|
|
|
// Mirror positive X values
|
|
|
|
xValues.push_back(yData[mirrorIndex]);
|
|
|
|
for (size_t x = mirrorIndex + 1; x < yData.size(); x++)
|
|
|
|
{
|
|
|
|
xValues.push_front(yData[x]);
|
|
|
|
xValues.push_back(yData[x]);
|
|
|
|
}
|
|
|
|
xyData.push_back(std::vector<double>(xValues.begin(), xValues.end()));
|
|
|
|
}
|
|
|
|
return xyData;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
return fileXYData;
|
|
|
|
}
|
2017-06-13 08:04:57 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
//--------------------------------------------------------------------------------------------------
|
|
|
|
///
|
|
|
|
//--------------------------------------------------------------------------------------------------
|
2018-08-10 07:12:14 -05:00
|
|
|
bool RigStimPlanFractureDefinition::numberOfParameterValuesOK(std::vector<std::vector<double>> propertyValuesAtTimestep) const
|
2017-06-13 08:04:57 -05:00
|
|
|
{
|
2018-02-08 04:05:01 -06:00
|
|
|
size_t xCount = m_Xs.size();
|
2017-06-13 08:04:57 -05:00
|
|
|
|
2018-02-08 04:05:01 -06:00
|
|
|
if ( propertyValuesAtTimestep.size() != yCount()) return false;
|
2018-08-10 07:12:14 -05:00
|
|
|
for ( const std::vector<double>& valuesAtDepthVector : propertyValuesAtTimestep )
|
2017-06-13 08:04:57 -05:00
|
|
|
{
|
2018-02-08 04:05:01 -06:00
|
|
|
if ( valuesAtDepthVector.size() != xCount ) return false;
|
2017-06-13 08:04:57 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
//--------------------------------------------------------------------------------------------------
|
|
|
|
///
|
|
|
|
//--------------------------------------------------------------------------------------------------
|
2018-02-08 04:05:01 -06:00
|
|
|
std::vector<double> RigStimPlanFractureDefinition::adjustedYCoordsAroundWellPathPosition(double wellPathIntersectionAtFractureDepth) const
|
2017-06-13 08:04:57 -05:00
|
|
|
{
|
2018-02-08 04:05:01 -06:00
|
|
|
std::vector<double> yRelativeToWellPath;
|
2017-06-13 08:04:57 -05:00
|
|
|
|
2018-02-08 04:05:01 -06:00
|
|
|
for ( const double& y : m_Ys )
|
2017-06-13 08:04:57 -05:00
|
|
|
{
|
2018-02-08 04:05:01 -06:00
|
|
|
double adjustedDepth = y + wellPathIntersectionAtFractureDepth;
|
|
|
|
yRelativeToWellPath.push_back(adjustedDepth);
|
2017-06-13 08:04:57 -05:00
|
|
|
}
|
2018-02-08 04:05:01 -06:00
|
|
|
return yRelativeToWellPath;
|
2017-06-13 08:04:57 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
//--------------------------------------------------------------------------------------------------
|
|
|
|
///
|
|
|
|
//--------------------------------------------------------------------------------------------------
|
|
|
|
std::vector<std::pair<QString, QString> > RigStimPlanFractureDefinition::getStimPlanPropertyNamesUnits() const
|
|
|
|
{
|
|
|
|
std::vector<std::pair<QString, QString> > propertyNamesUnits;
|
|
|
|
{
|
2017-06-14 02:20:15 -05:00
|
|
|
for ( const RigStimPlanResultFrames& stimPlanDataEntry : this->m_stimPlanResults )
|
2017-06-13 08:04:57 -05:00
|
|
|
{
|
|
|
|
propertyNamesUnits.push_back(std::make_pair(stimPlanDataEntry.resultName, stimPlanDataEntry.unit));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return propertyNamesUnits;
|
|
|
|
}
|
|
|
|
|
2017-06-13 07:44:04 -05:00
|
|
|
//--------------------------------------------------------------------------------------------------
|
2018-03-05 02:43:06 -06:00
|
|
|
///
|
|
|
|
//--------------------------------------------------------------------------------------------------
|
|
|
|
std::vector<std::vector<double>>
|
|
|
|
RigStimPlanFractureDefinition::conductivityValuesAtTimeStep(const QString& resultName,
|
|
|
|
int activeTimeStepIndex,
|
|
|
|
RiaEclipseUnitTools::UnitSystem requiredUnitSet) const
|
|
|
|
{
|
|
|
|
std::vector<std::vector<double>> conductivityValues;
|
|
|
|
|
|
|
|
QString conductivityUnitTextOnFile;
|
|
|
|
|
|
|
|
std::vector<std::pair<QString, QString>> propertyNamesUnitsOnFile = this->getStimPlanPropertyNamesUnits();
|
|
|
|
for (auto properyNameUnit : propertyNamesUnitsOnFile)
|
|
|
|
{
|
|
|
|
if (resultName == properyNameUnit.first)
|
|
|
|
{
|
|
|
|
conductivityUnitTextOnFile = properyNameUnit.second;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (conductivityUnitTextOnFile.isEmpty())
|
|
|
|
{
|
|
|
|
RiaLogging::error("Did not find unit for conductivity on file");
|
|
|
|
|
|
|
|
return conductivityValues;
|
|
|
|
}
|
|
|
|
|
|
|
|
conductivityValues = this->getDataAtTimeIndex(resultName, conductivityUnitTextOnFile, activeTimeStepIndex);
|
|
|
|
|
|
|
|
// Convert to the conductivity unit system used by the fracture template
|
|
|
|
// The conductivity value is used in the computations of transmissibility when exporting COMPDAT, and has unit md-m or md-ft
|
|
|
|
// This unit must match the unit used to represent coordinates of the grid used for export
|
|
|
|
|
|
|
|
for (auto& yValues : conductivityValues)
|
|
|
|
{
|
|
|
|
for (auto& xVal : yValues)
|
|
|
|
{
|
|
|
|
if (requiredUnitSet == RiaEclipseUnitTools::UNITS_FIELD)
|
|
|
|
{
|
2018-08-14 01:01:50 -05:00
|
|
|
xVal = RiaEclipseUnitTools::convertToFeet(xVal, conductivityUnitTextOnFile);
|
2018-03-05 02:43:06 -06:00
|
|
|
}
|
|
|
|
else if (requiredUnitSet == RiaEclipseUnitTools::UNITS_METRIC)
|
|
|
|
{
|
2018-08-14 01:01:50 -05:00
|
|
|
xVal = RiaEclipseUnitTools::convertToMeter(xVal, conductivityUnitTextOnFile);
|
2018-03-05 02:43:06 -06:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return conductivityValues;
|
|
|
|
}
|
|
|
|
|
|
|
|
//--------------------------------------------------------------------------------------------------
|
|
|
|
///
|
2017-06-13 07:44:04 -05:00
|
|
|
//--------------------------------------------------------------------------------------------------
|
2018-01-25 05:14:54 -06:00
|
|
|
cvf::ref<RigFractureGrid> RigStimPlanFractureDefinition::createFractureGrid(const QString& resultName,
|
2018-03-05 02:43:06 -06:00
|
|
|
int activeTimeStepIndex,
|
|
|
|
double wellPathIntersectionAtFractureDepth,
|
2018-08-10 07:12:14 -05:00
|
|
|
RiaEclipseUnitTools::UnitSystem requiredUnitSet) const
|
2017-06-13 07:44:04 -05:00
|
|
|
{
|
2018-03-05 02:43:06 -06:00
|
|
|
std::vector<std::vector<double>> conductivityValues = conductivityValuesAtTimeStep(resultName, activeTimeStepIndex, requiredUnitSet);
|
|
|
|
if (conductivityValues.empty())
|
|
|
|
{
|
|
|
|
return nullptr;
|
|
|
|
}
|
|
|
|
|
2017-06-13 07:44:04 -05:00
|
|
|
std::vector<RigFractureCell> stimPlanCells;
|
|
|
|
std::pair<size_t, size_t> wellCenterStimPlanCellIJ = std::make_pair(0, 0);
|
|
|
|
|
|
|
|
bool wellCenterStimPlanCellFound = false;
|
|
|
|
|
2018-02-08 04:05:01 -06:00
|
|
|
std::vector<double> yCoordsAtNodes = this->adjustedYCoordsAroundWellPathPosition(wellPathIntersectionAtFractureDepth);
|
|
|
|
std::vector<double> xCoordsAtNodes = this->m_Xs;
|
2017-06-13 07:44:04 -05:00
|
|
|
|
|
|
|
std::vector<double> xCoords;
|
2017-06-26 04:05:46 -05:00
|
|
|
for ( int i = 0; i < static_cast<int>(xCoordsAtNodes.size()) - 1; i++ ) xCoords.push_back((xCoordsAtNodes[i] + xCoordsAtNodes[i + 1]) / 2);
|
2017-06-13 07:44:04 -05:00
|
|
|
std::vector<double> depthCoords;
|
2018-02-08 04:05:01 -06:00
|
|
|
for ( int i = 0; i < static_cast<int>(yCoordsAtNodes.size()) - 1; i++ ) depthCoords.push_back((yCoordsAtNodes[i] + yCoordsAtNodes[i + 1]) / 2);
|
2017-06-13 07:44:04 -05:00
|
|
|
|
2017-06-26 04:05:46 -05:00
|
|
|
for ( int i = 0; i < static_cast<int>(xCoords.size()) - 1; i++ )
|
2017-06-13 07:44:04 -05:00
|
|
|
{
|
2017-06-26 04:05:46 -05:00
|
|
|
for ( int j = 0; j < static_cast<int>(depthCoords.size()) - 1; j++ )
|
2017-06-13 07:44:04 -05:00
|
|
|
{
|
|
|
|
std::vector<cvf::Vec3d> cellPolygon;
|
2017-06-26 03:28:23 -05:00
|
|
|
cellPolygon.push_back(cvf::Vec3d(xCoords[i], depthCoords[j], 0.0));
|
|
|
|
cellPolygon.push_back(cvf::Vec3d(xCoords[i + 1], depthCoords[j], 0.0));
|
|
|
|
cellPolygon.push_back(cvf::Vec3d(xCoords[i + 1], depthCoords[j + 1], 0.0));
|
|
|
|
cellPolygon.push_back(cvf::Vec3d(xCoords[i], depthCoords[j + 1], 0.0));
|
2017-06-13 07:44:04 -05:00
|
|
|
|
|
|
|
RigFractureCell stimPlanCell(cellPolygon, i, j);
|
2018-08-10 07:12:14 -05:00
|
|
|
if ( !conductivityValues.empty() ) //Assuming vector to be of correct length, or no values
|
2017-06-13 07:44:04 -05:00
|
|
|
{
|
2018-03-05 02:43:06 -06:00
|
|
|
stimPlanCell.setConductivityValue(conductivityValues[j + 1][i + 1]);
|
2017-06-13 07:44:04 -05:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
stimPlanCell.setConductivityValue(cvf::UNDEFINED_DOUBLE);
|
|
|
|
}
|
|
|
|
|
2017-11-09 07:03:52 -06:00
|
|
|
// The well path is intersecting the fracture at origo in the fracture coordinate system
|
|
|
|
// Find the Stimplan cell where the well path is intersecting
|
2017-06-13 07:44:04 -05:00
|
|
|
|
2017-11-09 07:03:52 -06:00
|
|
|
if ( cellPolygon[0].x() <= 0.0 && cellPolygon[1].x() >= 0.0 )
|
2017-06-13 07:44:04 -05:00
|
|
|
{
|
2017-11-09 07:03:52 -06:00
|
|
|
if ( cellPolygon[1].y() >= 0.0 && cellPolygon[2].y() <= 0.0 )
|
2017-06-13 07:44:04 -05:00
|
|
|
{
|
|
|
|
wellCenterStimPlanCellIJ = std::make_pair(stimPlanCell.getI(), stimPlanCell.getJ());
|
|
|
|
RiaLogging::debug(QString("Setting wellCenterStimPlanCell at cell %1, %2").
|
|
|
|
arg(QString::number(stimPlanCell.getI()), QString::number(stimPlanCell.getJ())));
|
|
|
|
|
|
|
|
wellCenterStimPlanCellFound = true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
stimPlanCells.push_back(stimPlanCell);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if ( !wellCenterStimPlanCellFound )
|
|
|
|
{
|
|
|
|
RiaLogging::error("Did not find stim plan cell at well crossing!");
|
|
|
|
}
|
|
|
|
|
2018-03-08 04:01:04 -06:00
|
|
|
cvf::ref<RigFractureGrid> fractureGrid = new RigFractureGrid;
|
|
|
|
fractureGrid->setFractureCells(stimPlanCells);
|
|
|
|
fractureGrid->setWellCenterFractureCellIJ(wellCenterStimPlanCellIJ);
|
|
|
|
fractureGrid->setICellCount(this->m_Xs.size() - 2);
|
|
|
|
fractureGrid->setJCellCount(this->adjustedYCoordsAroundWellPathPosition(wellPathIntersectionAtFractureDepth).size() - 2);
|
2017-06-13 07:44:04 -05:00
|
|
|
|
2018-03-08 04:01:04 -06:00
|
|
|
return fractureGrid;
|
2017-06-13 07:44:04 -05:00
|
|
|
}
|
|
|
|
|
2017-06-26 03:28:23 -05:00
|
|
|
//--------------------------------------------------------------------------------------------------
|
|
|
|
///
|
|
|
|
//--------------------------------------------------------------------------------------------------
|
|
|
|
std::vector<double> RigStimPlanFractureDefinition::fractureGridResults(const QString& resultName,
|
|
|
|
const QString& unitName,
|
|
|
|
size_t timeStepIndex) const
|
|
|
|
{
|
|
|
|
std::vector<double> fractureGridResults;
|
2018-02-08 04:05:01 -06:00
|
|
|
const std::vector<std::vector<double>>& resultValuesAtTimeStep = this->getDataAtTimeIndex(resultName,
|
|
|
|
unitName,
|
|
|
|
timeStepIndex);
|
2017-06-26 03:28:23 -05:00
|
|
|
|
2018-02-08 04:05:01 -06:00
|
|
|
for ( int i = 0; i < static_cast<int>(xCount()) - 2; i++ )
|
2017-06-26 03:28:23 -05:00
|
|
|
{
|
2018-02-08 04:05:01 -06:00
|
|
|
for ( int j = 0; j < static_cast<int>(yCount()) - 2; j++ )
|
2017-06-26 03:28:23 -05:00
|
|
|
{
|
2017-06-26 04:05:46 -05:00
|
|
|
if ( j+1 < static_cast<int>(resultValuesAtTimeStep.size()) && i+1 < static_cast<int>(resultValuesAtTimeStep[j + 1].size()) )
|
2017-06-26 03:28:23 -05:00
|
|
|
{
|
|
|
|
fractureGridResults.push_back(resultValuesAtTimeStep[j + 1][i + 1]);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
fractureGridResults.push_back(HUGE_VAL);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return fractureGridResults;
|
|
|
|
}
|
|
|
|
|
2017-06-13 07:44:04 -05:00
|
|
|
//--------------------------------------------------------------------------------------------------
|
|
|
|
///
|
|
|
|
//--------------------------------------------------------------------------------------------------
|
2018-02-08 04:05:01 -06:00
|
|
|
void RigStimPlanFractureDefinition::createFractureTriangleGeometry(double wellPathIntersectionAtFractureDepth,
|
2017-06-13 07:44:04 -05:00
|
|
|
const QString& fractureUserName,
|
|
|
|
std::vector<cvf::Vec3f>* vertices,
|
2018-08-10 07:12:14 -05:00
|
|
|
std::vector<cvf::uint>* triangleIndices) const
|
2017-06-13 07:44:04 -05:00
|
|
|
{
|
2018-02-08 04:05:01 -06:00
|
|
|
std::vector<double> xCoords = this->m_Xs;
|
2017-06-13 07:44:04 -05:00
|
|
|
cvf::uint lenXcoords = static_cast<cvf::uint>(xCoords.size());
|
|
|
|
|
2018-02-08 04:05:01 -06:00
|
|
|
std::vector<double> adjustedYs = this->adjustedYCoordsAroundWellPathPosition(wellPathIntersectionAtFractureDepth);
|
2017-06-13 07:44:04 -05:00
|
|
|
|
2018-02-08 04:05:01 -06:00
|
|
|
for ( cvf::uint k = 0; k < adjustedYs.size(); k++ )
|
2017-06-13 07:44:04 -05:00
|
|
|
{
|
|
|
|
for ( cvf::uint i = 0; i < lenXcoords; i++ )
|
|
|
|
{
|
2018-02-08 04:05:01 -06:00
|
|
|
cvf::Vec3f node = cvf::Vec3f(xCoords[i], adjustedYs[k], 0);
|
2017-06-13 07:44:04 -05:00
|
|
|
vertices->push_back(node);
|
|
|
|
|
2018-02-08 04:05:01 -06:00
|
|
|
if ( i < lenXcoords - 1 && k < adjustedYs.size() - 1 )
|
2017-06-13 07:44:04 -05:00
|
|
|
{
|
2018-02-08 04:05:01 -06:00
|
|
|
if ( xCoords[i] < THRESHOLD_VALUE )
|
2017-06-13 07:44:04 -05:00
|
|
|
{
|
|
|
|
//Upper triangle
|
|
|
|
triangleIndices->push_back(i + k*lenXcoords);
|
|
|
|
triangleIndices->push_back((i + 1) + k*lenXcoords);
|
|
|
|
triangleIndices->push_back((i + 1) + (k + 1)*lenXcoords);
|
|
|
|
//Lower triangle
|
|
|
|
triangleIndices->push_back(i + k*lenXcoords);
|
|
|
|
triangleIndices->push_back((i + 1) + (k + 1)*lenXcoords);
|
|
|
|
triangleIndices->push_back((i)+(k + 1)*lenXcoords);
|
|
|
|
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
//Upper triangle
|
|
|
|
triangleIndices->push_back(i + k*lenXcoords);
|
|
|
|
triangleIndices->push_back((i + 1) + k*lenXcoords);
|
|
|
|
triangleIndices->push_back((i)+(k + 1)*lenXcoords);
|
|
|
|
//Lower triangle
|
|
|
|
triangleIndices->push_back((i + 1) + k*lenXcoords);
|
|
|
|
triangleIndices->push_back((i + 1) + (k + 1)*lenXcoords);
|
|
|
|
triangleIndices->push_back((i)+ (k + 1)*lenXcoords);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
//--------------------------------------------------------------------------------------------------
|
|
|
|
///
|
|
|
|
//--------------------------------------------------------------------------------------------------
|
2018-02-14 06:34:38 -06:00
|
|
|
const std::vector<double>& RigStimPlanFractureDefinition::timeSteps() const
|
|
|
|
{
|
|
|
|
return m_timeSteps;
|
|
|
|
}
|
|
|
|
|
|
|
|
//--------------------------------------------------------------------------------------------------
|
|
|
|
///
|
|
|
|
//--------------------------------------------------------------------------------------------------
|
|
|
|
void RigStimPlanFractureDefinition::addTimeStep(double time)
|
|
|
|
{
|
|
|
|
if (!timeStepExists(time)) m_timeSteps.push_back(time);
|
|
|
|
}
|
|
|
|
|
2017-02-15 03:32:34 -06:00
|
|
|
//--------------------------------------------------------------------------------------------------
|
|
|
|
///
|
|
|
|
//--------------------------------------------------------------------------------------------------
|
2018-08-10 07:12:14 -05:00
|
|
|
bool RigStimPlanFractureDefinition::timeStepExists(double timeStepValueToCheck) const
|
2017-02-15 03:32:34 -06:00
|
|
|
{
|
2017-06-14 02:20:15 -05:00
|
|
|
for (double timeStep : m_timeSteps)
|
2017-02-15 03:32:34 -06:00
|
|
|
{
|
2018-02-08 04:05:01 -06:00
|
|
|
if (fabs(timeStepValueToCheck - timeStep) < THRESHOLD_VALUE) return true;
|
2017-02-15 03:32:34 -06:00
|
|
|
}
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
//--------------------------------------------------------------------------------------------------
|
|
|
|
///
|
|
|
|
//--------------------------------------------------------------------------------------------------
|
2018-08-10 07:12:14 -05:00
|
|
|
size_t RigStimPlanFractureDefinition::getTimeStepIndex(double timeStepValue) const
|
2017-02-15 03:32:34 -06:00
|
|
|
{
|
|
|
|
size_t index = 0;
|
2017-06-14 02:20:15 -05:00
|
|
|
while (index < m_timeSteps.size())
|
2017-02-15 03:32:34 -06:00
|
|
|
{
|
2018-01-05 02:29:53 -06:00
|
|
|
if (fabs(m_timeSteps[index] - timeStepValue) < 1e-4)
|
2017-02-15 03:32:34 -06:00
|
|
|
{
|
|
|
|
return index;
|
|
|
|
}
|
|
|
|
index++;
|
|
|
|
}
|
|
|
|
return -1; //returns -1 if not found
|
|
|
|
}
|
2017-02-17 02:29:46 -06:00
|
|
|
|
2017-02-17 05:17:57 -06:00
|
|
|
//--------------------------------------------------------------------------------------------------
|
|
|
|
///
|
|
|
|
//--------------------------------------------------------------------------------------------------
|
2018-08-10 07:12:14 -05:00
|
|
|
size_t RigStimPlanFractureDefinition::totalNumberTimeSteps() const
|
2017-03-02 04:43:28 -06:00
|
|
|
{
|
2017-06-14 02:20:15 -05:00
|
|
|
return m_timeSteps.size();
|
2017-03-02 04:43:28 -06:00
|
|
|
}
|
|
|
|
|
|
|
|
//--------------------------------------------------------------------------------------------------
|
|
|
|
///
|
|
|
|
//--------------------------------------------------------------------------------------------------
|
2017-02-20 02:31:24 -06:00
|
|
|
size_t RigStimPlanFractureDefinition::resultIndex(const QString& resultName, const QString& unit) const
|
2017-02-17 05:17:57 -06:00
|
|
|
{
|
2017-02-20 02:31:24 -06:00
|
|
|
|
2017-06-13 08:04:57 -05:00
|
|
|
for (size_t i = 0; i < m_stimPlanResults.size(); i++)
|
2017-02-17 05:17:57 -06:00
|
|
|
{
|
2017-06-13 08:04:57 -05:00
|
|
|
if (m_stimPlanResults[i].resultName == resultName && m_stimPlanResults[i].unit == unit)
|
2017-02-17 05:17:57 -06:00
|
|
|
{
|
2017-02-20 02:31:24 -06:00
|
|
|
return i;
|
2017-02-17 05:17:57 -06:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-02-20 02:31:24 -06:00
|
|
|
return cvf::UNDEFINED_SIZE_T;
|
|
|
|
}
|
|
|
|
|
|
|
|
//--------------------------------------------------------------------------------------------------
|
|
|
|
///
|
|
|
|
//--------------------------------------------------------------------------------------------------
|
2018-01-25 06:30:36 -06:00
|
|
|
void RigStimPlanFractureDefinition::setDataAtTimeValue(QString resultName, QString unit, std::vector<std::vector<double>> data, double timeStepValue)
|
2017-02-20 02:31:24 -06:00
|
|
|
{
|
|
|
|
size_t resIndex = resultIndex(resultName, unit);
|
|
|
|
|
|
|
|
if (resIndex != cvf::UNDEFINED_SIZE_T)
|
|
|
|
{
|
2017-06-13 08:04:57 -05:00
|
|
|
m_stimPlanResults[resIndex].parameterValues[getTimeStepIndex(timeStepValue)] = data;
|
2017-02-20 02:31:24 -06:00
|
|
|
}
|
|
|
|
else
|
2017-02-17 05:17:57 -06:00
|
|
|
{
|
2017-05-11 07:01:58 -05:00
|
|
|
RigStimPlanResultFrames resultData;
|
2017-02-17 05:17:57 -06:00
|
|
|
|
|
|
|
resultData.resultName = resultName;
|
|
|
|
resultData.unit = unit;
|
|
|
|
|
2017-06-14 02:20:15 -05:00
|
|
|
std::vector<std::vector<std::vector<double>>> values(m_timeSteps.size());
|
2017-02-17 05:17:57 -06:00
|
|
|
resultData.parameterValues = values;
|
|
|
|
resultData.parameterValues[getTimeStepIndex(timeStepValue)] = data;
|
|
|
|
|
2017-06-13 08:04:57 -05:00
|
|
|
m_stimPlanResults.push_back(resultData);
|
2017-02-17 05:17:57 -06:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
//--------------------------------------------------------------------------------------------------
|
|
|
|
///
|
|
|
|
//--------------------------------------------------------------------------------------------------
|
2017-06-14 02:20:15 -05:00
|
|
|
const std::vector<std::vector<double>>& RigStimPlanFractureDefinition::getDataAtTimeIndex(const QString& resultName, const QString& unit, size_t timeStepIndex) const
|
2017-02-17 05:17:57 -06:00
|
|
|
{
|
2017-02-20 02:31:24 -06:00
|
|
|
size_t resIndex = resultIndex(resultName, unit);
|
|
|
|
|
|
|
|
if (resIndex != cvf::UNDEFINED_SIZE_T)
|
2017-02-17 05:17:57 -06:00
|
|
|
{
|
2017-06-13 08:04:57 -05:00
|
|
|
if (timeStepIndex < m_stimPlanResults[resIndex].parameterValues.size())
|
2017-02-20 03:04:21 -06:00
|
|
|
{
|
2017-06-13 08:04:57 -05:00
|
|
|
return m_stimPlanResults[resIndex].parameterValues[timeStepIndex];
|
2017-02-20 03:04:21 -06:00
|
|
|
}
|
2017-02-17 05:17:57 -06:00
|
|
|
}
|
|
|
|
|
2018-01-19 02:10:30 -06:00
|
|
|
RiaLogging::error("Requested parameter does not exists in stimPlan data");
|
|
|
|
|
2017-06-14 02:20:15 -05:00
|
|
|
static std::vector<std::vector<double>> emptyVector;
|
2017-02-17 05:17:57 -06:00
|
|
|
return emptyVector;
|
|
|
|
}
|
|
|
|
|
2017-02-20 02:31:24 -06:00
|
|
|
//--------------------------------------------------------------------------------------------------
|
|
|
|
///
|
|
|
|
//--------------------------------------------------------------------------------------------------
|
2018-01-19 02:10:30 -06:00
|
|
|
void RigStimPlanFractureDefinition::appendDataToResultStatistics(const QString& resultName, const QString& unit,
|
|
|
|
MinMaxAccumulator& minMaxAccumulator,
|
|
|
|
PosNegAccumulator& posNegAccumulator) const
|
2017-02-20 02:31:24 -06:00
|
|
|
{
|
|
|
|
size_t resIndex = resultIndex(resultName, unit);
|
|
|
|
if (resIndex == cvf::UNDEFINED_SIZE_T) return;
|
|
|
|
|
2017-06-14 02:20:15 -05:00
|
|
|
for (const auto& timeValues : m_stimPlanResults[resIndex].parameterValues)
|
2017-02-20 02:31:24 -06:00
|
|
|
{
|
2017-06-14 02:20:15 -05:00
|
|
|
for (const auto& values : timeValues)
|
2017-02-20 02:31:24 -06:00
|
|
|
{
|
2018-01-19 02:10:30 -06:00
|
|
|
minMaxAccumulator.addData(values);
|
|
|
|
posNegAccumulator.addData(values);
|
2017-02-20 02:31:24 -06:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-01-25 05:14:54 -06:00
|
|
|
//--------------------------------------------------------------------------------------------------
|
|
|
|
///
|
|
|
|
//--------------------------------------------------------------------------------------------------
|
|
|
|
QStringList RigStimPlanFractureDefinition::conductivityResultNames() const
|
|
|
|
{
|
|
|
|
QStringList resultNames;
|
|
|
|
|
|
|
|
for (const auto& stimPlanResult : m_stimPlanResults)
|
|
|
|
{
|
|
|
|
if (stimPlanResult.resultName.contains("conductivity", Qt::CaseInsensitive))
|
|
|
|
{
|
|
|
|
resultNames.push_back(stimPlanResult.resultName);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return resultNames;
|
|
|
|
}
|
|
|
|
|
2018-02-08 04:05:01 -06:00
|
|
|
//--------------------------------------------------------------------------------------------------
|
|
|
|
///
|
|
|
|
//--------------------------------------------------------------------------------------------------
|
|
|
|
size_t findMirrorXIndex(std::vector<double> xs)
|
|
|
|
{
|
|
|
|
size_t mirrorIndex = cvf::UNDEFINED_SIZE_T;
|
|
|
|
|
|
|
|
for (size_t i = 0; i < xs.size(); i++)
|
|
|
|
{
|
|
|
|
if (xs[i] > -RigStimPlanFractureDefinition::THRESHOLD_VALUE)
|
|
|
|
{
|
|
|
|
mirrorIndex = i;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return mirrorIndex;
|
|
|
|
}
|