///////////////////////////////////////////////////////////////////////////////// // // 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 // for more details. // ///////////////////////////////////////////////////////////////////////////////// #include "RimSimWellInView.h" #include "RigActiveCellInfo.h" #include "RigCell.h" #include "RigEclipseCaseData.h" #include "RigMainGrid.h" #include "RigSimWellData.h" #include "RigSimulationWellCenterLineCalculator.h" #include "RimCellRangeFilterCollection.h" #include "RimEclipseCase.h" #include "RimEclipseView.h" #include "RimIntersectionCollection.h" #include "RimPropertyFilterCollection.h" #include "RimSimWellFracture.h" #include "RimSimWellFractureCollection.h" #include "RimSimWellInViewCollection.h" #include "RimIntersection.h" #include "Rim2dIntersectionView.h" #include "RiuMainWindow.h" #include "RivReservoirViewPartMgr.h" #include "cafPdmUiTreeOrdering.h" #include "cvfMath.h" //-------------------------------------------------------------------------------------------------- /// Internal functions //-------------------------------------------------------------------------------------------------- Rim2dIntersectionView* corresponding2dIntersectionView(RimSimWellInView *simWellInView); CAF_PDM_SOURCE_INIT(RimSimWellInView, "Well"); //-------------------------------------------------------------------------------------------------- /// //-------------------------------------------------------------------------------------------------- RimSimWellInView::RimSimWellInView() { CAF_PDM_InitObject("Well", ":/Well.png", "", ""); CAF_PDM_InitFieldNoDefault(&name, "WellName", "Name", "", "", ""); CAF_PDM_InitField(&showWell, "ShowWell", true, "Show well ", "", "", ""); CAF_PDM_InitField(&showWellLabel, "ShowWellLabel", true, "Label", "", "", ""); CAF_PDM_InitField(&showWellHead, "ShowWellHead", true, "Well Head", "", "", ""); CAF_PDM_InitField(&showWellPipe, "ShowWellPipe", true, "Pipe", "", "", ""); CAF_PDM_InitField(&showWellSpheres, "ShowWellSpheres", false, "Spheres", "", "", ""); CAF_PDM_InitField(&wellHeadScaleFactor, "WellHeadScaleFactor", 1.0, "Well Head Scale", "", "", ""); CAF_PDM_InitField(&pipeScaleFactor, "WellPipeRadiusScale", 1.0, "Pipe Radius Scale", "", "", ""); CAF_PDM_InitField(&wellPipeColor, "WellPipeColor", cvf::Color3f(0.588f, 0.588f, 0.804f), "Pipe Color", "", "", ""); CAF_PDM_InitField(&showWellCells, "ShowWellCells", false, "Well Cells", "", "", ""); CAF_PDM_InitField(&showWellCellFence, "ShowWellCellFence", false, "Well Cell Fence", "", "", ""); CAF_PDM_InitFieldNoDefault(&simwellFractureCollection, "FractureCollection", "Fractures", "", "", ""); name.uiCapability()->setUiHidden(true); name.uiCapability()->setUiReadOnly(true); m_resultWellIndex = cvf::UNDEFINED_SIZE_T; simwellFractureCollection= new RimSimWellFractureCollection(); } //-------------------------------------------------------------------------------------------------- /// //-------------------------------------------------------------------------------------------------- RimSimWellInView::~RimSimWellInView() { if (simwellFractureCollection()) delete simwellFractureCollection(); } //-------------------------------------------------------------------------------------------------- /// //-------------------------------------------------------------------------------------------------- caf::PdmFieldHandle* RimSimWellInView::userDescriptionField() { return &name; } //-------------------------------------------------------------------------------------------------- /// //-------------------------------------------------------------------------------------------------- void RimSimWellInView::fieldChangedByUi(const caf::PdmFieldHandle* changedField, const QVariant& oldValue, const QVariant& newValue) { RimEclipseView* reservoirView = nullptr; this->firstAncestorOrThisOfType(reservoirView); if (reservoirView) { if (&showWellLabel == changedField || &showWellHead == changedField || &showWellPipe == changedField || &showWellSpheres == changedField || &wellPipeColor == changedField) { reservoirView->scheduleCreateDisplayModelAndRedraw(); schedule2dIntersectionViewUpdate(); } else if (&showWell == changedField || &showWellCells == changedField || &showWellCellFence == changedField) { reservoirView->scheduleGeometryRegen(VISIBLE_WELL_CELLS); reservoirView->scheduleCreateDisplayModelAndRedraw(); schedule2dIntersectionViewUpdate(); } else if ( &pipeScaleFactor == changedField || &wellHeadScaleFactor == changedField) { reservoirView->scheduleSimWellGeometryRegen(); reservoirView->scheduleCreateDisplayModelAndRedraw(); schedule2dIntersectionViewUpdate(); } } RimSimWellInViewCollection* wellColl = nullptr; this->firstAncestorOrThisOfType(wellColl); if (wellColl) { wellColl->updateStateForVisibilityCheckboxes(); RiuMainWindow::instance()->refreshDrawStyleActions(); } if (changedField == &wellPipeColor) { RimSimWellInViewCollection::updateWellAllocationPlots(); } } //-------------------------------------------------------------------------------------------------- /// //-------------------------------------------------------------------------------------------------- caf::PdmFieldHandle* RimSimWellInView::objectToggleField() { return &showWell; } //-------------------------------------------------------------------------------------------------- /// //-------------------------------------------------------------------------------------------------- std::vector RimSimWellInView::wellPipeBranches() const { RimSimWellInViewCollection* simWellCollection = nullptr; this->firstAncestorOrThisOfTypeAsserted(simWellCollection); RimEclipseCase* eclipseCase = nullptr; this->firstAncestorOrThisOfTypeAsserted(eclipseCase); RigEclipseCaseData* caseData = eclipseCase->eclipseCaseData(); CVF_ASSERT(caseData); if (caseData) { bool includeCellCenters = this->isUsingCellCenterForPipe(); bool detectBrances = simWellCollection->isAutoDetectingBranches; return caseData->simulationWellBranches(this->name(), includeCellCenters, detectBrances); } return std::vector(); } //-------------------------------------------------------------------------------------------------- /// //-------------------------------------------------------------------------------------------------- void RimSimWellInView::calculateWellPipeStaticCenterLine(std::vector>& pipeBranchesCLCoords, std::vector>& pipeBranchesCellIds) { RigSimulationWellCenterLineCalculator::calculateWellPipeStaticCenterline(this, pipeBranchesCLCoords, pipeBranchesCellIds); } //-------------------------------------------------------------------------------------------------- /// frameIndex = -1 will use the static well frame //-------------------------------------------------------------------------------------------------- void RimSimWellInView::wellHeadTopBottomPosition(int frameIndex, cvf::Vec3d* top, cvf::Vec3d* bottom) { RimEclipseView* m_rimReservoirView; firstAncestorOrThisOfTypeAsserted(m_rimReservoirView); RigEclipseCaseData* rigReservoir = m_rimReservoirView->eclipseCase()->eclipseCaseData(); const RigWellResultFrame* wellResultFramePtr = nullptr; const RigCell* whCellPtr = nullptr; if (frameIndex >= 0) { if ( !this->simWellData()->hasAnyValidCells(frameIndex) ) return; wellResultFramePtr = &(this->simWellData()->wellResultFrame(frameIndex)); whCellPtr = &(rigReservoir->cellFromWellResultCell(wellResultFramePtr->wellHeadOrStartCell())); } else { wellResultFramePtr = &(this->simWellData()->staticWellCells()); whCellPtr = &(rigReservoir->cellFromWellResultCell(wellResultFramePtr->wellHeadOrStartCell())); } const RigCell& whCell = *whCellPtr; // Match this position with pipe start position in RivWellPipesPartMgr::calculateWellPipeCenterline() (*bottom) = whCell.faceCenter(cvf::StructGridInterface::NEG_K); // Compute well head based on the z position of the top of the K column the well head is part of (*top) = (*bottom); if ( m_rimReservoirView->wellCollection()->wellHeadPosition() == RimSimWellInViewCollection::WELLHEAD_POS_TOP_COLUMN ) { // Position well head at top active cell of IJ-column size_t i, j, k; rigReservoir->mainGrid()->ijkFromCellIndex(whCell.mainGridCellIndex(), &i, &j, &k); size_t kIndexWellHeadCell = k; k = 0; size_t topActiveCellIndex = rigReservoir->mainGrid()->cellIndexFromIJK(i, j, k); while ( k < kIndexWellHeadCell && !m_rimReservoirView->currentActiveCellInfo()->isActive(topActiveCellIndex) ) { k++; topActiveCellIndex = rigReservoir->mainGrid()->cellIndexFromIJK(i, j, k); } const RigCell& topActiveCell = rigReservoir->mainGrid()->cell(topActiveCellIndex); cvf::Vec3d topCellPos = topActiveCell.faceCenter(cvf::StructGridInterface::NEG_K); // Modify position if top active cell is closer to sea than well head if ( kIndexWellHeadCell > k ) { top->z() = topCellPos.z(); } } else { // Position well head at top of active cells bounding box cvf::Vec3d activeCellsBoundingBoxMax = m_rimReservoirView->currentActiveCellInfo()->geometryBoundingBox().max(); top->z() = activeCellsBoundingBoxMax.z(); } } //-------------------------------------------------------------------------------------------------- /// //-------------------------------------------------------------------------------------------------- double RimSimWellInView::pipeRadius() { RimEclipseView* reservoirView; firstAncestorOrThisOfTypeAsserted(reservoirView); RigEclipseCaseData* rigReservoir = reservoirView->eclipseCase()->eclipseCaseData(); double characteristicCellSize = rigReservoir->mainGrid()->characteristicIJCellSize(); double pipeRadius = reservoirView->wellCollection()->pipeScaleFactor() * this->pipeScaleFactor() * characteristicCellSize; return pipeRadius; } //-------------------------------------------------------------------------------------------------- /// //-------------------------------------------------------------------------------------------------- int RimSimWellInView::pipeCrossSectionVertexCount() { RimSimWellInViewCollection* simWellCollection = nullptr; this->firstAncestorOrThisOfTypeAsserted(simWellCollection); return simWellCollection->pipeCrossSectionVertexCount(); } //-------------------------------------------------------------------------------------------------- /// //-------------------------------------------------------------------------------------------------- bool RimSimWellInView::intersectsDynamicWellCellsFilteredCells(size_t frameIndex) const { if (this->simWellData() == nullptr) return false; if (!simWellData()->hasWellResult(frameIndex)) return false; const RigWellResultFrame& wrsf = this->simWellData()->wellResultFrame(frameIndex); return intersectsWellCellsFilteredCells(wrsf, frameIndex); } //-------------------------------------------------------------------------------------------------- /// //-------------------------------------------------------------------------------------------------- bool RimSimWellInView::intersectsWellCellsFilteredCells(const RigWellResultFrame &wrsf, size_t frameIndex) const { RimEclipseView* reservoirView = nullptr; this->firstAncestorOrThisOfType(reservoirView); if (!reservoirView) return false; const std::vector& visGridParts = reservoirView->visibleGridParts(); RivReservoirViewPartMgr* rvMan = reservoirView->reservoirGridPartManager(); for (const RivCellSetEnum& visGridPart : visGridParts) { if (visGridPart == ALL_WELL_CELLS || visGridPart == VISIBLE_WELL_CELLS || visGridPart == VISIBLE_WELL_FENCE_CELLS || visGridPart == VISIBLE_WELL_CELLS_OUTSIDE_RANGE_FILTER || visGridPart == VISIBLE_WELL_FENCE_CELLS_OUTSIDE_RANGE_FILTER ) { // Exclude all cells related to well cells continue; } // First check the wellhead: size_t gridIndex = wrsf.m_wellHead.m_gridIndex; size_t gridCellIndex = wrsf.m_wellHead.m_gridCellIndex; if (gridIndex != cvf::UNDEFINED_SIZE_T && gridCellIndex != cvf::UNDEFINED_SIZE_T) { const cvf::UByteArray* cellVisibility = rvMan->cellVisibility(visGridPart, gridIndex, frameIndex); if (gridCellIndex < cellVisibility->size() && (*cellVisibility)[gridCellIndex]) { return true; } } // Then check the rest of the well, with all the branches const std::vector& wellResSegments = wrsf.m_wellResultBranches; for (const RigWellResultBranch& branchSegment : wellResSegments) { const std::vector& wsResCells = branchSegment.m_branchResultPoints; for (const RigWellResultPoint& wellResultPoint : wsResCells) { if (wellResultPoint.isCell()) { gridIndex = wellResultPoint.m_gridIndex; gridCellIndex = wellResultPoint.m_gridCellIndex; const cvf::UByteArray* cellVisibility = rvMan->cellVisibility(visGridPart, gridIndex, frameIndex); if (gridCellIndex < cellVisibility->size() && (*cellVisibility)[gridCellIndex]) { return true; } } } } } return false; } //-------------------------------------------------------------------------------------------------- /// //-------------------------------------------------------------------------------------------------- void RimSimWellInView::schedule2dIntersectionViewUpdate() { Rim2dIntersectionView* intersectionView = corresponding2dIntersectionView(this); if (intersectionView) { intersectionView->scheduleCreateDisplayModelAndRedraw(); } } //-------------------------------------------------------------------------------------------------- /// //-------------------------------------------------------------------------------------------------- bool RimSimWellInView::intersectsStaticWellCellsFilteredCells() const { if (this->simWellData() == nullptr) return false; // NOTE: Read out static well cells, union of well cells across all time steps const RigWellResultFrame& wrsf = this->simWellData()->staticWellCells(); // NOTE: Use first time step for visibility evaluation size_t frameIndex = 0; return intersectsWellCellsFilteredCells(wrsf, frameIndex); } //-------------------------------------------------------------------------------------------------- /// //-------------------------------------------------------------------------------------------------- void RimSimWellInView::defineUiOrdering(QString uiConfigName, caf::PdmUiOrdering& uiOrdering) { caf::PdmUiGroup* appearanceGroup = uiOrdering.addNewGroup("Visibility"); appearanceGroup->add(&showWellLabel); appearanceGroup->add(&showWellHead); appearanceGroup->add(&showWellPipe); appearanceGroup->add(&showWellSpheres); caf::PdmUiGroup* filterGroup = uiOrdering.addNewGroup("Well Cells and Fence"); filterGroup->add(&showWellCells); filterGroup->add(&showWellCellFence); showWellCellFence.uiCapability()->setUiReadOnly(!showWellCells()); caf::PdmUiGroup* sizeScalingGroup = uiOrdering.addNewGroup("Size Scaling"); sizeScalingGroup->add(&wellHeadScaleFactor); sizeScalingGroup->add(&pipeScaleFactor); caf::PdmUiGroup* colorGroup = uiOrdering.addNewGroup("Colors"); colorGroup->add(&wellPipeColor); uiOrdering.skipRemainingFields(true); } //-------------------------------------------------------------------------------------------------- /// //-------------------------------------------------------------------------------------------------- void RimSimWellInView::defineUiTreeOrdering(caf::PdmUiTreeOrdering& uiTreeOrdering, QString uiConfigName /*= ""*/) { for (RimSimWellFracture* fracture : simwellFractureCollection()->simwellFractures()) { uiTreeOrdering.add(fracture); } uiTreeOrdering.skipRemainingChildren(true); const RimEclipseView* reservoirView = nullptr; this->firstAncestorOrThisOfType(reservoirView); if (!reservoirView) return; if (reservoirView->rangeFilterCollection() && !reservoirView->rangeFilterCollection()->hasActiveFilters()) { this->uiCapability()->setUiReadOnly(false); return; } const RimSimWellInViewCollection* wellColl = nullptr; this->firstAncestorOrThisOfType(wellColl); if (!wellColl) return; if (wellColl->showWellsIntersectingVisibleCells() && !this->intersectsDynamicWellCellsFilteredCells(static_cast(reservoirView->currentTimeStep()))) { // Mark well as read only if well is not intersecting visible cells this->uiCapability()->setUiReadOnly(true); } else { this->uiCapability()->setUiReadOnly(false); } } //-------------------------------------------------------------------------------------------------- /// //-------------------------------------------------------------------------------------------------- bool RimSimWellInView::isWellCellsVisible() const { const RimEclipseView* reservoirView = nullptr; this->firstAncestorOrThisOfType(reservoirView); if (reservoirView == nullptr) return false; if (this->simWellData() == nullptr) return false; if (!reservoirView->wellCollection()->isActive()) return false; if (!this->showWell()) return false; if (!this->showWellCells()) return false; if (reservoirView->crossSectionCollection()->hasActiveIntersectionForSimulationWell(this)) return true; if (reservoirView->wellCollection()->showWellsIntersectingVisibleCells() && reservoirView->rangeFilterCollection()->hasActiveFilters()) { return intersectsStaticWellCellsFilteredCells(); } else { return true; } } //-------------------------------------------------------------------------------------------------- /// //-------------------------------------------------------------------------------------------------- bool RimSimWellInView::isWellPipeVisible(size_t frameIndex) const { const RimEclipseView* reservoirView = nullptr; this->firstAncestorOrThisOfType(reservoirView); if (reservoirView == nullptr) return false; if (this->simWellData() == nullptr) return false; if (frameIndex >= this->simWellData()->m_resultTimeStepIndexToWellTimeStepIndex.size()) { return false; } size_t wellTimeStepIndex = this->simWellData()->m_resultTimeStepIndexToWellTimeStepIndex[frameIndex]; if (wellTimeStepIndex == cvf::UNDEFINED_SIZE_T) { return false; } if (!reservoirView->wellCollection()->isActive()) return false; if (!this->showWell()) return false; if (!this->showWellPipe()) return false; if (reservoirView->crossSectionCollection()->hasActiveIntersectionForSimulationWell(this)) return true; if ( reservoirView->wellCollection()->showWellsIntersectingVisibleCells() && ( reservoirView->rangeFilterCollection()->hasActiveFilters() || reservoirView->propertyFilterCollection()->hasActiveFilters()) ) { return intersectsDynamicWellCellsFilteredCells(frameIndex); } else { return true; } } //-------------------------------------------------------------------------------------------------- /// //-------------------------------------------------------------------------------------------------- bool RimSimWellInView::isWellSpheresVisible(size_t frameIndex) const { const RimEclipseView* reservoirView = nullptr; this->firstAncestorOrThisOfType(reservoirView); if (reservoirView == nullptr) return false; if (this->simWellData() == nullptr) return false; if (frameIndex >= this->simWellData()->m_resultTimeStepIndexToWellTimeStepIndex.size()) { return false; } size_t wellTimeStepIndex = this->simWellData()->m_resultTimeStepIndexToWellTimeStepIndex[frameIndex]; if (wellTimeStepIndex == cvf::UNDEFINED_SIZE_T) { return false; } if (!reservoirView->wellCollection()->isActive()) return false; if (!this->showWell()) return false; if (!this->showWellSpheres()) return false; if (reservoirView->crossSectionCollection()->hasActiveIntersectionForSimulationWell(this)) return true; if (reservoirView->wellCollection()->showWellsIntersectingVisibleCells() && reservoirView->rangeFilterCollection()->hasActiveFilters()) { return intersectsDynamicWellCellsFilteredCells(frameIndex); } else { return true; } } //-------------------------------------------------------------------------------------------------- /// //-------------------------------------------------------------------------------------------------- bool RimSimWellInView::isUsingCellCenterForPipe() const { const RimSimWellInViewCollection* wellColl = nullptr; this->firstAncestorOrThisOfType(wellColl); return (wellColl && wellColl->wellPipeCoordType() == RimSimWellInViewCollection::WELLPIPE_CELLCENTER); } //-------------------------------------------------------------------------------------------------- /// //-------------------------------------------------------------------------------------------------- void RimSimWellInView::setSimWellData(RigSimWellData* simWellData, size_t resultWellIndex) { m_simWellData = simWellData; m_resultWellIndex = resultWellIndex; } //-------------------------------------------------------------------------------------------------- /// //-------------------------------------------------------------------------------------------------- RigSimWellData* RimSimWellInView::simWellData() { return m_simWellData.p(); } //-------------------------------------------------------------------------------------------------- /// //-------------------------------------------------------------------------------------------------- const RigSimWellData* RimSimWellInView::simWellData() const { return m_simWellData.p(); } //-------------------------------------------------------------------------------------------------- /// //-------------------------------------------------------------------------------------------------- size_t RimSimWellInView::resultWellIndex() const { return m_resultWellIndex; } //-------------------------------------------------------------------------------------------------- /// Internal functions //-------------------------------------------------------------------------------------------------- Rim2dIntersectionView* corresponding2dIntersectionView(RimSimWellInView *simWellInView) { Rim3dView* tdView; simWellInView->firstAncestorOrThisOfType(tdView); std::vector intersectionColls; if (tdView) { tdView->descendantsIncludingThisOfType(intersectionColls); if (intersectionColls.size() == 1) { for (const auto intersection : intersectionColls[0]->intersections()) { if (intersection->simulationWell() == simWellInView) { return intersection->correspondingIntersectionView(); } } } } return nullptr; }