///////////////////////////////////////////////////////////////////////////////// // // 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 "RimSummaryCaseCollection.h" #include "RiaFieldHandleTools.h" #include "RimDerivedEnsembleCaseCollection.h" #include "RimEnsembleCurveSet.h" #include "RimGridSummaryCase.h" #include "RimProject.h" #include "RimSummaryCase.h" #include "RifSummaryReaderInterface.h" #include #include #include #include CAF_PDM_SOURCE_INIT(RimSummaryCaseCollection, "SummaryCaseSubCollection"); //-------------------------------------------------------------------------------------------------- /// //-------------------------------------------------------------------------------------------------- RimSummaryCaseCollection::RimSummaryCaseCollection() { CAF_PDM_InitObject("Summary Case Group", ":/SummaryGroup16x16.png", "", ""); CAF_PDM_InitFieldNoDefault(&m_cases, "SummaryCases", "", "", "", ""); m_cases.uiCapability()->setUiHidden(true); CAF_PDM_InitField(&m_name, "SummaryCollectionName", QString("Group"), "Name", "", "", ""); CAF_PDM_InitFieldNoDefault(&m_nameAndItemCount, "NameCount", "Name", "", "", ""); m_nameAndItemCount.registerGetMethod(this, &RimSummaryCaseCollection::nameAndItemCount); RiaFieldhandleTools::disableWriteAndSetFieldHidden(&m_nameAndItemCount); CAF_PDM_InitField(&m_isEnsemble, "IsEnsemble", false, "Is Ensemble", "", "", ""); m_isEnsemble.uiCapability()->setUiHidden(true); m_commonAddressCount = 0; } //-------------------------------------------------------------------------------------------------- /// //-------------------------------------------------------------------------------------------------- RimSummaryCaseCollection::~RimSummaryCaseCollection() { m_cases.deleteAllChildObjectsAsync(); updateReferringCurveSets(); } //-------------------------------------------------------------------------------------------------- /// //-------------------------------------------------------------------------------------------------- void RimSummaryCaseCollection::removeCase(RimSummaryCase* summaryCase) { size_t caseCountBeforeRemove = m_cases.size(); m_cases.removeChildObject(summaryCase); updateReferringCurveSets(); if (m_isEnsemble && m_cases.size() != caseCountBeforeRemove) { if(dynamic_cast(summaryCase) == nullptr) calculateEnsembleParametersIntersectionHash(); } } //-------------------------------------------------------------------------------------------------- /// //-------------------------------------------------------------------------------------------------- void RimSummaryCaseCollection::addCase(RimSummaryCase* summaryCase, bool updateCurveSets) { m_cases.push_back(summaryCase); // Update derived ensemble cases (if any) std::vector referringObjects; objectsWithReferringPtrFieldsOfType(referringObjects); for (auto derEnsemble : referringObjects) { if (!derEnsemble) continue; derEnsemble->updateDerivedEnsembleCases(); if (updateCurveSets) derEnsemble->updateReferringCurveSets(); } if (m_isEnsemble) { validateEnsembleCases({ summaryCase }); calculateEnsembleParametersIntersectionHash(); } if(updateCurveSets) updateReferringCurveSets(); } //-------------------------------------------------------------------------------------------------- /// //-------------------------------------------------------------------------------------------------- std::vector RimSummaryCaseCollection::allSummaryCases() const { return m_cases.childObjects(); } //-------------------------------------------------------------------------------------------------- /// //-------------------------------------------------------------------------------------------------- void RimSummaryCaseCollection::setName(const QString& name) { m_name = name; } //-------------------------------------------------------------------------------------------------- /// //-------------------------------------------------------------------------------------------------- QString RimSummaryCaseCollection::name() const { return m_name; } //-------------------------------------------------------------------------------------------------- /// //-------------------------------------------------------------------------------------------------- bool RimSummaryCaseCollection::isEnsemble() const { return m_isEnsemble(); } //-------------------------------------------------------------------------------------------------- /// //-------------------------------------------------------------------------------------------------- void RimSummaryCaseCollection::setAsEnsemble(bool isEnsemble) { if (isEnsemble != m_isEnsemble) { m_isEnsemble = isEnsemble; updateIcon(); if (m_isEnsemble && dynamic_cast(this) == nullptr) { validateEnsembleCases(allSummaryCases()); calculateEnsembleParametersIntersectionHash(); } } } //-------------------------------------------------------------------------------------------------- /// //-------------------------------------------------------------------------------------------------- std::set RimSummaryCaseCollection::ensembleSummaryAddresses() const { std::set addresses; size_t maxAddrCount = 0; int maxAddrIndex = -1; for (int i = 0; i < (int)m_cases.size(); i++) { RimSummaryCase* currCase = m_cases[i]; if (!currCase) continue; RifSummaryReaderInterface* reader = currCase->summaryReader(); if (!reader) continue; size_t addrCount = reader->allResultAddresses().size(); if (addrCount > maxAddrCount) { maxAddrCount = addrCount; maxAddrIndex = (int)i; } } if (maxAddrIndex >= 0) { const std::set& addrs = m_cases[maxAddrIndex]->summaryReader()->allResultAddresses(); addresses.insert(addrs.begin(), addrs.end()); } return addresses; } //-------------------------------------------------------------------------------------------------- /// //-------------------------------------------------------------------------------------------------- EnsembleParameter RimSummaryCaseCollection::ensembleParameter(const QString& paramName) const { if (!isEnsemble() || paramName.isEmpty()) return EnsembleParameter(); EnsembleParameter eParam; eParam.name = paramName; bool numericValuesCount = 0; bool textValuesCount = 0; // Prepare case realization params, and check types for (const auto& rimCase : allSummaryCases()) { auto crp = rimCase->caseRealizationParameters(); if (!crp) continue; auto value = crp->parameterValue(paramName); if (!value.isValid()) continue; if (value.isNumeric()) { double numVal = value.numericValue(); eParam.values.push_back(QVariant(numVal)); if (numVal < eParam.minValue) eParam.minValue = numVal; if (numVal > eParam.maxValue) eParam.maxValue = numVal; numericValuesCount++; } else if (value.isText()) { eParam.values.push_back(QVariant(value.textValue())); textValuesCount++; } } if (numericValuesCount && !textValuesCount) { eParam.type = EnsembleParameter::TYPE_NUMERIC; } else if (textValuesCount && !numericValuesCount) { eParam.type = EnsembleParameter::TYPE_TEXT; } if (numericValuesCount && textValuesCount) { // A mix of types have been added to parameter values if (numericValuesCount > textValuesCount) { // Use numeric type for (auto& val : eParam.values) { if (val.type() == QVariant::String) { val.setValue(std::numeric_limits::infinity()); } } eParam.type = EnsembleParameter::TYPE_NUMERIC; } else { // Use text type for (auto& val : eParam.values) { if (val.type() == QVariant::Double) { val.setValue(QString::number(val.value())); } } eParam.type = EnsembleParameter::TYPE_TEXT; eParam.minValue = std::numeric_limits::infinity(); eParam.maxValue = -std::numeric_limits::infinity(); } } if (eParam.isText()) { // Remove duplicate texts std::set valueSet; for (const auto& val : eParam.values) { valueSet.insert(val.toString()); } eParam.values.clear(); for (const auto& val : valueSet) { eParam.values.push_back(QVariant(val)); } } return eParam; } //-------------------------------------------------------------------------------------------------- /// //-------------------------------------------------------------------------------------------------- void RimSummaryCaseCollection::calculateEnsembleParametersIntersectionHash() { clearEnsembleParametersHashes(); // Find ensemble parameters intersection std::set paramNames; auto sumCases = allSummaryCases(); for (size_t i = 0; i < sumCases.size(); i++) { auto crp = sumCases[i]->caseRealizationParameters(); if (!crp) continue; auto caseParamNames = crp->parameterNames(); if (i == 0) paramNames = caseParamNames; else { std::set newIntersection; std::set_intersection(paramNames.begin(), paramNames.end(), caseParamNames.begin(), caseParamNames.end(), std::inserter(newIntersection, newIntersection.end())); if(paramNames.size() != newIntersection.size()) paramNames = newIntersection; } } for (auto sumCase : sumCases) { auto crp = sumCase->caseRealizationParameters(); if(crp) crp->calculateParametersHash(paramNames); } // Find common addess count for (const auto sumCase : sumCases) { const auto reader = sumCase->summaryReader(); if(!reader) continue; auto currAddrCount = reader->allResultAddresses().size(); if (m_commonAddressCount == 0) { m_commonAddressCount = currAddrCount; } else { if (currAddrCount != m_commonAddressCount) { m_commonAddressCount = 0; break; } } } } //-------------------------------------------------------------------------------------------------- /// //-------------------------------------------------------------------------------------------------- void RimSummaryCaseCollection::clearEnsembleParametersHashes() { for (auto sumCase : allSummaryCases()) { auto crp = sumCase->caseRealizationParameters(); if (crp) crp->clearParametersHash(); } } //-------------------------------------------------------------------------------------------------- /// //-------------------------------------------------------------------------------------------------- void RimSummaryCaseCollection::loadDataAndUpdate() { onLoadDataAndUpdate(); } //-------------------------------------------------------------------------------------------------- /// //-------------------------------------------------------------------------------------------------- bool RimSummaryCaseCollection::validateEnsembleCases(const std::vector cases) { // Validate ensemble parameters try { QString errors; std::hash paramsHasher; size_t paramsHash = 0; for (RimSummaryCase* rimCase : cases) { if (rimCase->caseRealizationParameters() == nullptr || rimCase->caseRealizationParameters()->parameters().empty()) { errors.append(QString("The case %1 has no ensemble parameters\n").arg(QFileInfo(rimCase->summaryHeaderFilename()).fileName())); } else { QString paramNames; for (std::pair paramPair : rimCase->caseRealizationParameters()->parameters()) { paramNames.append(paramPair.first); } size_t currHash = paramsHasher(paramNames.toStdString()); if (paramsHash == 0) { paramsHash = currHash; } else if (paramsHash != currHash) { throw QString("Ensemble parameters differ between cases"); } } } if (!errors.isEmpty()) { errors.prepend("Missing ensemble parameters\n\n"); errors.append("\n"); errors.append("No parameters file (parameters.txt or runspecification.xml) was found in \n"); errors.append("the searched folders. ResInsight searches the home folder of the summary \n"); errors.append("case file and the three folder levels above that.\n"); throw errors; } return true; } catch (QString errorMessage) { QMessageBox mbox; mbox.setIcon(QMessageBox::Icon::Warning); mbox.setText(errorMessage); mbox.exec(); return false; } } //-------------------------------------------------------------------------------------------------- /// Sorting operator for sets and maps. Sorts by name. //-------------------------------------------------------------------------------------------------- bool RimSummaryCaseCollection::operator<(const RimSummaryCaseCollection& rhs) const { return name() < rhs.name(); } //-------------------------------------------------------------------------------------------------- /// //-------------------------------------------------------------------------------------------------- caf::PdmFieldHandle* RimSummaryCaseCollection::userDescriptionField() { return &m_name; } //-------------------------------------------------------------------------------------------------- /// //-------------------------------------------------------------------------------------------------- void RimSummaryCaseCollection::onLoadDataAndUpdate() { if (m_isEnsemble) calculateEnsembleParametersIntersectionHash(); } //-------------------------------------------------------------------------------------------------- /// //-------------------------------------------------------------------------------------------------- void RimSummaryCaseCollection::updateReferringCurveSets() { // Update curve set referring to this group std::vector referringObjects; objectsWithReferringPtrFieldsOfType(referringObjects); for (auto curveSet : referringObjects) { if (curveSet) curveSet->updateAllCurves(); } } //-------------------------------------------------------------------------------------------------- /// //-------------------------------------------------------------------------------------------------- QString RimSummaryCaseCollection::nameAndItemCount() const { size_t itemCount = m_cases.size(); if (itemCount > 20) { return QString("%1 (%2)").arg(m_name()).arg(itemCount); } return m_name(); } //-------------------------------------------------------------------------------------------------- /// //-------------------------------------------------------------------------------------------------- void RimSummaryCaseCollection::updateIcon() { if (m_isEnsemble) setUiIcon(QIcon(":/SummaryEnsemble16x16.png")); else setUiIcon(QIcon(":/SummaryGroup16x16.png")); } //-------------------------------------------------------------------------------------------------- /// //-------------------------------------------------------------------------------------------------- void RimSummaryCaseCollection::initAfterRead() { updateIcon(); } //-------------------------------------------------------------------------------------------------- /// //-------------------------------------------------------------------------------------------------- void RimSummaryCaseCollection::fieldChangedByUi(const caf::PdmFieldHandle* changedField, const QVariant& oldValue, const QVariant& newValue) { if (changedField == &m_isEnsemble) { updateIcon(); } } //-------------------------------------------------------------------------------------------------- /// //-------------------------------------------------------------------------------------------------- void RimSummaryCaseCollection::defineUiOrdering(QString uiConfigName, caf::PdmUiOrdering& uiOrdering) { uiOrdering.add(&m_name); uiOrdering.skipRemainingFields(true); } //-------------------------------------------------------------------------------------------------- /// //-------------------------------------------------------------------------------------------------- void RimSummaryCaseCollection::setNameAsReadOnly() { m_name.uiCapability()->setUiReadOnly(true); }