mirror of
https://github.com/OPM/ResInsight.git
synced 2025-01-24 15:26:48 -06:00
445 lines
18 KiB
C++
445 lines
18 KiB
C++
/////////////////////////////////////////////////////////////////////////////////
|
|
//
|
|
// 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 <http://www.gnu.org/licenses/gpl.html>
|
|
// for more details.
|
|
//
|
|
/////////////////////////////////////////////////////////////////////////////////
|
|
|
|
#include "RimEclipseCellColors.h"
|
|
|
|
#include "RigCaseCellResultsData.h"
|
|
#include "RigEclipseCaseData.h"
|
|
#include "RigFlowDiagResults.h"
|
|
#include "RigFormationNames.h"
|
|
|
|
#include "RimCellEdgeColors.h"
|
|
#include "RimEclipseCase.h"
|
|
#include "RimEclipseFaultColors.h"
|
|
#include "RimEclipseView.h"
|
|
#include "RimEclipseWell.h"
|
|
#include "RimEclipseWellCollection.h"
|
|
#include "RimLegendConfig.h"
|
|
#include "RimTernaryLegendConfig.h"
|
|
#include "RimViewController.h"
|
|
#include "RimViewLinker.h"
|
|
|
|
#include "RiuMainWindow.h"
|
|
|
|
#include "cafPdmUiTreeOrdering.h"
|
|
|
|
CAF_PDM_SOURCE_INIT(RimEclipseCellColors, "ResultSlot");
|
|
|
|
//--------------------------------------------------------------------------------------------------
|
|
///
|
|
//--------------------------------------------------------------------------------------------------
|
|
RimEclipseCellColors::RimEclipseCellColors()
|
|
{
|
|
CAF_PDM_InitObject("Cell Result", ":/CellResult.png", "", "");
|
|
|
|
CAF_PDM_InitFieldNoDefault(&obsoleteField_legendConfig, "LegendDefinition", "Legend Definition", "", "", "");
|
|
this->obsoleteField_legendConfig.xmlCapability()->setIOWritable(false);
|
|
|
|
CAF_PDM_InitFieldNoDefault(&m_legendConfigData, "ResultVarLegendDefinitionList", "", "", "", "");
|
|
|
|
CAF_PDM_InitFieldNoDefault(&ternaryLegendConfig, "TernaryLegendDefinition", "Ternary Legend Definition", "", "", "");
|
|
this->ternaryLegendConfig = new RimTernaryLegendConfig();
|
|
|
|
CAF_PDM_InitFieldNoDefault(&m_legendConfigPtrField, "LegendDefinitionPtrField", "Legend Definition PtrField", "", "", "");
|
|
|
|
// Make sure we have a created legend for the default/undefined result variable
|
|
changeLegendConfig(this->resultVariable());
|
|
}
|
|
|
|
//--------------------------------------------------------------------------------------------------
|
|
///
|
|
//--------------------------------------------------------------------------------------------------
|
|
RimEclipseCellColors::~RimEclipseCellColors()
|
|
{
|
|
CVF_ASSERT(obsoleteField_legendConfig() == NULL);
|
|
|
|
m_legendConfigData.deleteAllChildObjects();
|
|
|
|
delete ternaryLegendConfig();
|
|
}
|
|
|
|
//--------------------------------------------------------------------------------------------------
|
|
///
|
|
//--------------------------------------------------------------------------------------------------
|
|
void RimEclipseCellColors::fieldChangedByUi(const caf::PdmFieldHandle* changedField, const QVariant& oldValue, const QVariant& newValue)
|
|
{
|
|
RimEclipseResultDefinition::fieldChangedByUi(changedField, oldValue, newValue);
|
|
|
|
// Update of legend config must happen after RimEclipseResultDefinition::fieldChangedByUi(), as this function modifies this->resultVariable()
|
|
if (changedField == &m_resultVariableUiField)
|
|
{
|
|
if (oldValue != newValue)
|
|
{
|
|
changeLegendConfig(this->resultVariable());
|
|
}
|
|
|
|
if (newValue != RiaDefines::undefinedResultName())
|
|
{
|
|
if (m_reservoirView) m_reservoirView->hasUserRequestedAnimation = true;
|
|
}
|
|
|
|
RimEclipseFaultColors* faultColors = dynamic_cast<RimEclipseFaultColors*>(this->parentField()->ownerObject());
|
|
if (faultColors)
|
|
{
|
|
faultColors->updateConnectedEditors();
|
|
}
|
|
|
|
RimCellEdgeColors* cellEdgeColors = dynamic_cast<RimCellEdgeColors*>(this->parentField()->ownerObject());
|
|
if (cellEdgeColors)
|
|
{
|
|
cellEdgeColors->updateConnectedEditors();
|
|
}
|
|
}
|
|
|
|
if (m_reservoirView) m_reservoirView->createDisplayModelAndRedraw();
|
|
}
|
|
|
|
//--------------------------------------------------------------------------------------------------
|
|
///
|
|
//--------------------------------------------------------------------------------------------------
|
|
void RimEclipseCellColors::changeLegendConfig(QString resultVarNameOfNewLegend)
|
|
{
|
|
if (resultVarNameOfNewLegend != RiaDefines::ternarySaturationResultName())
|
|
{
|
|
QString legendResultVariable;
|
|
|
|
if (this->m_legendConfigPtrField())
|
|
{
|
|
legendResultVariable = this->m_legendConfigPtrField()->resultVariableName();
|
|
}
|
|
|
|
if (!this->m_legendConfigPtrField() || legendResultVariable != resultVarNameOfNewLegend)
|
|
{
|
|
bool found = false;
|
|
for (size_t i = 0; i < m_legendConfigData.size(); i++)
|
|
{
|
|
if (m_legendConfigData[i]->resultVariableName() == resultVarNameOfNewLegend)
|
|
{
|
|
this->m_legendConfigPtrField = m_legendConfigData[i];
|
|
found = true;
|
|
break;
|
|
}
|
|
}
|
|
|
|
// Not found ?
|
|
if (!found)
|
|
{
|
|
RimLegendConfig* newLegend = new RimLegendConfig;
|
|
newLegend->resultVariableName = resultVarNameOfNewLegend;
|
|
m_legendConfigData.push_back(newLegend);
|
|
|
|
this->m_legendConfigPtrField = newLegend;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
//--------------------------------------------------------------------------------------------------
|
|
///
|
|
//--------------------------------------------------------------------------------------------------
|
|
void RimEclipseCellColors::initAfterRead()
|
|
{
|
|
RimEclipseResultDefinition::initAfterRead();
|
|
|
|
if (this->m_legendConfigPtrField() && this->m_legendConfigPtrField()->resultVariableName == "")
|
|
{
|
|
this->m_legendConfigPtrField()->resultVariableName = this->resultVariable();
|
|
}
|
|
|
|
if (obsoleteField_legendConfig)
|
|
{
|
|
// The current legend config is NOT stored in <ResultVarLegendDefinitionList> in ResInsight up to v 1.3.7-dev
|
|
RimLegendConfig* obsoleteLegend = obsoleteField_legendConfig();
|
|
|
|
// set to NULL before pushing into container
|
|
obsoleteField_legendConfig = NULL;
|
|
|
|
m_legendConfigData.push_back(obsoleteLegend);
|
|
m_legendConfigPtrField = obsoleteLegend;
|
|
}
|
|
|
|
changeLegendConfig(this->resultVariable());
|
|
|
|
updateIconState();
|
|
}
|
|
|
|
//--------------------------------------------------------------------------------------------------
|
|
///
|
|
//--------------------------------------------------------------------------------------------------
|
|
void RimEclipseCellColors::defineUiTreeOrdering(caf::PdmUiTreeOrdering& uiTreeOrdering, QString uiConfigName /*= ""*/)
|
|
{
|
|
if (this->resultVariable() == RiaDefines::ternarySaturationResultName())
|
|
{
|
|
uiTreeOrdering.add(ternaryLegendConfig());
|
|
}
|
|
else
|
|
{
|
|
uiTreeOrdering.add(m_legendConfigPtrField());
|
|
}
|
|
|
|
uiTreeOrdering.skipRemainingChildren(true);
|
|
}
|
|
|
|
//--------------------------------------------------------------------------------------------------
|
|
///
|
|
//--------------------------------------------------------------------------------------------------
|
|
void RimEclipseCellColors::updateLegendCategorySettings()
|
|
{
|
|
changeLegendConfig(this->resultVariable());
|
|
|
|
if (this->hasCategoryResult())
|
|
{
|
|
legendConfig()->setMappingMode(RimLegendConfig::CATEGORY_INTEGER);
|
|
legendConfig()->setColorRangeMode(RimLegendConfig::CATEGORY);
|
|
}
|
|
else
|
|
{
|
|
if (legendConfig()->mappingMode() == RimLegendConfig::CATEGORY_INTEGER)
|
|
{
|
|
legendConfig()->setMappingMode(RimLegendConfig::LINEAR_CONTINUOUS);
|
|
}
|
|
|
|
if (legendConfig()->colorRangeMode() == RimLegendConfig::CATEGORY)
|
|
{
|
|
legendConfig()->setColorRangeMode(RimLegendConfig::NORMAL);
|
|
}
|
|
}
|
|
}
|
|
|
|
//--------------------------------------------------------------------------------------------------
|
|
///
|
|
//--------------------------------------------------------------------------------------------------
|
|
void RimEclipseCellColors::setReservoirView(RimEclipseView* ownerReservoirView)
|
|
{
|
|
this->setEclipseCase(ownerReservoirView->eclipseCase());
|
|
|
|
m_reservoirView = ownerReservoirView;
|
|
}
|
|
|
|
//--------------------------------------------------------------------------------------------------
|
|
///
|
|
//--------------------------------------------------------------------------------------------------
|
|
RimEclipseView* RimEclipseCellColors::reservoirView()
|
|
{
|
|
return m_reservoirView;
|
|
}
|
|
|
|
bool operator<(const cvf::Color3ub first, const cvf::Color3ub second)
|
|
{
|
|
if (first.r() != second.r()) return first.r() < second.r();
|
|
if (first.g() != second.g()) return first.g() < second.g();
|
|
if (first.b() != second.b()) return first.b() < second.b();
|
|
|
|
return false;
|
|
}
|
|
|
|
//--------------------------------------------------------------------------------------------------
|
|
///
|
|
//--------------------------------------------------------------------------------------------------
|
|
class TupleCompare
|
|
{
|
|
public :
|
|
bool operator() (const std::tuple<QString, int, cvf::Color3ub>& t1, const std::tuple<QString, int, cvf::Color3ub>& t2) const
|
|
{
|
|
using namespace std;
|
|
if (get<0>(t1) != get<0>(t2)) return get<0>(t1) < get<0>(t2);
|
|
if (get<1>(t1) != get<1>(t2)) return get<1>(t1) < get<1>(t2);
|
|
if (get<2>(t1) != get<2>(t2)) return get<2>(t1) < get<2>(t2);
|
|
|
|
return false;
|
|
}
|
|
};
|
|
|
|
|
|
//--------------------------------------------------------------------------------------------------
|
|
///
|
|
//--------------------------------------------------------------------------------------------------
|
|
void RimEclipseCellColors::updateLegendData(size_t currentTimeStep)
|
|
{
|
|
if (this->resultType() == RiaDefines::FLOW_DIAGNOSTICS)
|
|
{
|
|
double globalMin, globalMax;
|
|
double globalPosClosestToZero, globalNegClosestToZero;
|
|
RigFlowDiagResults* flowResultsData = this->flowDiagSolution()->flowDiagResults();
|
|
RigFlowDiagResultAddress resAddr = this->flowDiagResAddress();
|
|
|
|
int integerTimeStep = static_cast<int>(currentTimeStep);
|
|
|
|
flowResultsData->minMaxScalarValues(resAddr, integerTimeStep, &globalMin, &globalMax);
|
|
flowResultsData->posNegClosestToZero(resAddr, integerTimeStep, &globalPosClosestToZero, &globalNegClosestToZero);
|
|
|
|
double localMin, localMax;
|
|
double localPosClosestToZero, localNegClosestToZero;
|
|
if (this->hasDynamicResult())
|
|
{
|
|
flowResultsData->minMaxScalarValues(resAddr, integerTimeStep, &localMin, &localMax);
|
|
flowResultsData->posNegClosestToZero(resAddr, integerTimeStep, &localPosClosestToZero, &localNegClosestToZero);
|
|
}
|
|
else
|
|
{
|
|
localMin = globalMin;
|
|
localMax = globalMax;
|
|
|
|
localPosClosestToZero = globalPosClosestToZero;
|
|
localNegClosestToZero = globalNegClosestToZero;
|
|
}
|
|
|
|
CVF_ASSERT(this->legendConfig());
|
|
|
|
this->legendConfig()->disableAllTimeStepsRange(true);
|
|
this->legendConfig()->setClosestToZeroValues(globalPosClosestToZero, globalNegClosestToZero, localPosClosestToZero, localNegClosestToZero);
|
|
this->legendConfig()->setAutomaticRanges(globalMin, globalMax, localMin, localMax);
|
|
|
|
if (this->hasCategoryResult())
|
|
{
|
|
std::set<std::tuple<QString, int, cvf::Color3ub>, TupleCompare > categories;
|
|
//std::set<std::tuple<QString, int, cvf::Color3ub> > categories;
|
|
|
|
std::vector<QString> tracerNames = this->flowDiagSolution()->tracerNames();
|
|
int tracerIndex = 0;
|
|
for (const auto& tracerName : tracerNames)
|
|
{
|
|
RimEclipseWell* well = m_reservoirView->wellCollection()->findWell(RimFlowDiagSolution::removeCrossFlowEnding(tracerName));
|
|
cvf::Color3ub color(cvf::Color3::GRAY);
|
|
if (well) color = cvf::Color3ub(well->wellPipeColor());
|
|
|
|
categories.insert(std::make_tuple(tracerName, tracerIndex, color));
|
|
++tracerIndex;
|
|
}
|
|
|
|
std::vector<std::tuple<QString, int, cvf::Color3ub>> reverseCategories;
|
|
for (auto tupIt = categories.rbegin(); tupIt != categories.rend(); ++tupIt)
|
|
{
|
|
reverseCategories.push_back(*tupIt);
|
|
}
|
|
|
|
this->legendConfig()->setCategoryItems(reverseCategories);
|
|
}
|
|
}
|
|
else
|
|
{
|
|
RimEclipseCase* rimEclipseCase = nullptr;
|
|
this->firstAncestorOrThisOfType(rimEclipseCase);
|
|
CVF_ASSERT(rimEclipseCase);
|
|
if (!rimEclipseCase) return;
|
|
|
|
RigEclipseCaseData* eclipseCase = rimEclipseCase->eclipseCaseData();
|
|
CVF_ASSERT(eclipseCase);
|
|
if (!eclipseCase) return;
|
|
|
|
RifReaderInterface::PorosityModelResultType porosityModel = RigCaseCellResultsData::convertFromProjectModelPorosityModel(this->porosityModel());
|
|
RigCaseCellResultsData* cellResultsData = eclipseCase->results(porosityModel);
|
|
CVF_ASSERT(cellResultsData);
|
|
|
|
double globalMin, globalMax;
|
|
double globalPosClosestToZero, globalNegClosestToZero;
|
|
cellResultsData->minMaxCellScalarValues(this->scalarResultIndex(), globalMin, globalMax);
|
|
cellResultsData->posNegClosestToZero(this->scalarResultIndex(), globalPosClosestToZero, globalNegClosestToZero);
|
|
|
|
double localMin, localMax;
|
|
double localPosClosestToZero, localNegClosestToZero;
|
|
if (this->hasDynamicResult())
|
|
{
|
|
cellResultsData->minMaxCellScalarValues(this->scalarResultIndex(), currentTimeStep, localMin, localMax);
|
|
cellResultsData->posNegClosestToZero(this->scalarResultIndex(), currentTimeStep, localPosClosestToZero, localNegClosestToZero);
|
|
}
|
|
else
|
|
{
|
|
localMin = globalMin;
|
|
localMax = globalMax;
|
|
|
|
localPosClosestToZero = globalPosClosestToZero;
|
|
localNegClosestToZero = globalNegClosestToZero;
|
|
}
|
|
|
|
CVF_ASSERT(this->legendConfig());
|
|
|
|
this->legendConfig()->disableAllTimeStepsRange(false);
|
|
this->legendConfig()->setClosestToZeroValues(globalPosClosestToZero, globalNegClosestToZero, localPosClosestToZero, localNegClosestToZero);
|
|
this->legendConfig()->setAutomaticRanges(globalMin, globalMax, localMin, localMax);
|
|
|
|
if (this->hasCategoryResult())
|
|
{
|
|
if (this->resultType() == RiaDefines::FORMATION_NAMES)
|
|
{
|
|
const std::vector<QString>& fnVector = eclipseCase->activeFormationNames()->formationNames();
|
|
this->legendConfig()->setNamedCategoriesInverse(fnVector);
|
|
}
|
|
else if (this->resultType() == RiaDefines::DYNAMIC_NATIVE && this->resultVariable() == RiaDefines::completionTypeResultName())
|
|
{
|
|
std::vector< std::tuple<QString, int, cvf::Color3ub> > categories;
|
|
|
|
caf::AppEnum<RiaDefines::CompletionType> wellPath(RiaDefines::WELL_PATH);
|
|
caf::AppEnum<RiaDefines::CompletionType> fishbone(RiaDefines::FISHBONES);
|
|
caf::AppEnum<RiaDefines::CompletionType> perforationInterval(RiaDefines::PERFORATION_INTERVAL);
|
|
caf::AppEnum<RiaDefines::CompletionType> fracture(RiaDefines::FRACTURE);
|
|
|
|
categories.push_back(std::make_tuple(wellPath.uiText(), static_cast<int>(wellPath.index()), cvf::Color3::RED));
|
|
categories.push_back(std::make_tuple(fishbone.uiText(), static_cast<int>(fishbone.index()), cvf::Color3::DARK_GREEN));
|
|
categories.push_back(std::make_tuple(perforationInterval.uiText(), static_cast<int>(perforationInterval.index()), cvf::Color3::GREEN));
|
|
categories.push_back(std::make_tuple(fracture.uiText(), static_cast<int>(fracture.index()), cvf::Color3::YELLOW_GREEN));
|
|
|
|
legendConfig()->setCategoryItems(categories);
|
|
}
|
|
else
|
|
{
|
|
this->legendConfig()->setIntegerCategories(cellResultsData->uniqueCellScalarValues(this->scalarResultIndex()));
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
//--------------------------------------------------------------------------------------------------
|
|
///
|
|
//--------------------------------------------------------------------------------------------------
|
|
void RimEclipseCellColors::setResultVariable(const QString& val)
|
|
{
|
|
RimEclipseResultDefinition::setResultVariable(val);
|
|
|
|
this->changeLegendConfig(val);
|
|
}
|
|
|
|
//--------------------------------------------------------------------------------------------------
|
|
///
|
|
//--------------------------------------------------------------------------------------------------
|
|
RimLegendConfig* RimEclipseCellColors::legendConfig()
|
|
{
|
|
return m_legendConfigPtrField;
|
|
}
|
|
|
|
//--------------------------------------------------------------------------------------------------
|
|
///
|
|
//--------------------------------------------------------------------------------------------------
|
|
void RimEclipseCellColors::updateIconState()
|
|
{
|
|
RimViewController* viewController = m_reservoirView->viewController();
|
|
if (viewController && viewController->isResultColorControlled())
|
|
{
|
|
updateUiIconFromState(false);
|
|
}
|
|
else
|
|
{
|
|
updateUiIconFromState(true);
|
|
}
|
|
|
|
uiCapability()->updateConnectedEditors();
|
|
}
|
|
|