///////////////////////////////////////////////////////////////////////////////// // // Copyright (C) 2015- Statoil ASA // Copyright (C) 2015- Ceetron Solutions 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 "RimViewLinker.h" #include "RiaApplication.h" #include "RigCaseData.h" #include "RimCase.h" #include "RimEclipseCellColors.h" #include "RimEclipseInputCase.h" #include "RimEclipseResultCase.h" #include "RimEclipseResultDefinition.h" #include "RimEclipseView.h" #include "RimGeoMechCase.h" #include "RimGeoMechCellColors.h" #include "RimGeoMechResultDefinition.h" #include "RimGeoMechView.h" #include "RimViewLink.h" #include "RimProject.h" #include "RimView.h" #include "RimViewLinkerCollection.h" #include "RiuViewer.h" #include "cvfCamera.h" #include "cvfScene.h" #include "cafFrameAnimationControl.h" #include "cvfMatrix4.h" #include "cafPdmUiTreeOrdering.h" CAF_PDM_SOURCE_INIT(RimViewLinker, "RimViewLinker"); //-------------------------------------------------------------------------------------------------- /// //-------------------------------------------------------------------------------------------------- RimViewLinker::RimViewLinker(void) { CAF_PDM_InitObject("Linked Views", "", "", ""); CAF_PDM_InitField(&m_name, "Name", QString("View Group Name"), "View Group Name", "", "", ""); m_name.uiCapability()->setUiHidden(true); CAF_PDM_InitFieldNoDefault(&m_mainView, "MainView", "Main View", "", "", ""); m_mainView.uiCapability()->setUiChildrenHidden(true); m_mainView.uiCapability()->setUiHidden(true); CAF_PDM_InitFieldNoDefault(&viewLinks, "ManagedViews", "Managed Views", "", "", ""); viewLinks.uiCapability()->setUiHidden(true); viewLinks.uiCapability()->setUiChildrenHidden(true); } //-------------------------------------------------------------------------------------------------- /// //-------------------------------------------------------------------------------------------------- RimViewLinker::~RimViewLinker(void) { removeOverrides(); viewLinks.deleteAllChildObjects(); } //-------------------------------------------------------------------------------------------------- /// //-------------------------------------------------------------------------------------------------- void RimViewLinker::updateTimeStep(RimView* sourceView, int timeStep) { CVF_ASSERT(sourceView); if (!isActive()) return; if (mainView() != sourceView) { RimViewLink* sourceViewLink = viewLinkForView(sourceView); CVF_ASSERT(sourceViewLink); if (!sourceViewLink->isActive() || !sourceViewLink->syncTimeStep()) { return; } } if (m_mainView && m_mainView->viewer() && sourceView != m_mainView) { m_mainView->viewer()->setCurrentFrame(timeStep); m_mainView->viewer()->animationControl()->setCurrentFrameOnly(timeStep); } for (size_t i = 0; i < viewLinks.size(); i++) { RimViewLink* viewLink = viewLinks[i]; if (!viewLink->isActive) continue; if (viewLink->managedView() && viewLink->managedView() != sourceView) { if (viewLink->syncTimeStep() && viewLink->managedView()->viewer()) { viewLink->managedView()->viewer()->setCurrentFrame(timeStep); viewLink->managedView()->viewer()->animationControl()->setCurrentFrameOnly(timeStep); } } } } //-------------------------------------------------------------------------------------------------- /// //-------------------------------------------------------------------------------------------------- void RimViewLinker::updateCellResult() { if (!isActive()) return; RimView* rimView = m_mainView; RimEclipseView* masterEclipseView = dynamic_cast(rimView); if (masterEclipseView && masterEclipseView->cellResult()) { RimEclipseResultDefinition* eclipseCellResultDefinition = masterEclipseView->cellResult(); for (size_t i = 0; i < viewLinks.size(); i++) { RimViewLink* viewLink = viewLinks[i]; if (!viewLink->isActive) continue; if (viewLink->managedView()) { if (viewLink->syncCellResult()) { RimView* rimView = viewLink->managedView(); RimEclipseView* eclipeView = dynamic_cast(rimView); if (eclipeView) { eclipeView->cellResult()->setPorosityModel(eclipseCellResultDefinition->porosityModel()); eclipeView->cellResult()->setResultType(eclipseCellResultDefinition->resultType()); eclipeView->cellResult()->setResultVariable(eclipseCellResultDefinition->resultVariable()); eclipeView->scheduleCreateDisplayModelAndRedraw(); } } } } } RimGeoMechView* masterGeoView = dynamic_cast(rimView); if (masterGeoView && masterGeoView->cellResult()) { RimGeoMechResultDefinition* geoMechResultDefinition = masterGeoView->cellResult(); for (size_t i = 0; i < viewLinks.size(); i++) { RimViewLink* viewLink = viewLinks[i]; if (!viewLink->isActive) continue; if (viewLink->managedView()) { if (viewLink->syncCellResult()) { RimView* rimView = viewLink->managedView(); RimGeoMechView* geoView = dynamic_cast(rimView); if (geoView) { geoView->cellResult()->setResultAddress(geoMechResultDefinition->resultAddress()); geoView->scheduleCreateDisplayModelAndRedraw(); } } } } } } //-------------------------------------------------------------------------------------------------- /// //-------------------------------------------------------------------------------------------------- void RimViewLinker::updateRangeFilters() { if (!isActive()) return; this->scheduleGeometryRegenForDepViews(RANGE_FILTERED); this->scheduleGeometryRegenForDepViews(RANGE_FILTERED_INACTIVE); this->scheduleCreateDisplayModelAndRedrawForDependentViews(); } //-------------------------------------------------------------------------------------------------- /// //-------------------------------------------------------------------------------------------------- void RimViewLinker::updatePropertyFilters() { if (!isActive()) return; this->scheduleGeometryRegenForDepViews(PROPERTY_FILTERED); this->scheduleCreateDisplayModelAndRedrawForDependentViews(); } //-------------------------------------------------------------------------------------------------- /// //-------------------------------------------------------------------------------------------------- void RimViewLinker::configureOverrides() { for (size_t i = 0; i < viewLinks.size(); i++) { RimViewLink* viewLink = viewLinks[i]; if (viewLink->isActive) { viewLink->configureOverrides(); } else { viewLink->removeOverrides(); } } } //-------------------------------------------------------------------------------------------------- /// //-------------------------------------------------------------------------------------------------- void RimViewLinker::allViewsForCameraSync(RimView* source, std::vector& views) { if (!isActive()) return; if (source != m_mainView()) { views.push_back(m_mainView()); } for (size_t i = 0; i < viewLinks.size(); i++) { if (viewLinks[i]->isActive() && viewLinks[i]->syncCamera && viewLinks[i]->managedView() && source != viewLinks[i]->managedView()) { views.push_back(viewLinks[i]->managedView()); } } } //-------------------------------------------------------------------------------------------------- /// //-------------------------------------------------------------------------------------------------- void RimViewLinker::applyAllOperations() { configureOverrides(); updateCellResult(); updateTimeStep(m_mainView, m_mainView->currentTimeStep()); updateRangeFilters(); updatePropertyFilters(); updateScaleZ(m_mainView, m_mainView->scaleZ()); updateCamera(m_mainView); } //-------------------------------------------------------------------------------------------------- /// //-------------------------------------------------------------------------------------------------- QString RimViewLinker::displayNameForView(RimView* view) { QString displayName = "None"; if (view) { RimCase* rimCase = NULL; view->firstAnchestorOrThisOfType(rimCase); displayName = rimCase->caseUserDescription() + ": " + view->name; } return displayName; } //-------------------------------------------------------------------------------------------------- /// //-------------------------------------------------------------------------------------------------- void RimViewLinker::setMainView(RimView* view) { m_mainView = view; setNameAndIcon(); updateUiIcon(); } //-------------------------------------------------------------------------------------------------- /// //-------------------------------------------------------------------------------------------------- RimView* RimViewLinker::mainView() { return m_mainView; } //-------------------------------------------------------------------------------------------------- /// //-------------------------------------------------------------------------------------------------- void RimViewLinker::allViews(std::vector& views) { views.push_back(m_mainView()); for (size_t i = 0; i < viewLinks.size(); i++) { if (viewLinks[i]->managedView()) { views.push_back(viewLinks[i]->managedView()); } } } //-------------------------------------------------------------------------------------------------- /// //-------------------------------------------------------------------------------------------------- void RimViewLinker::initAfterRead() { setNameAndIcon(); updateUiIcon(); } //-------------------------------------------------------------------------------------------------- /// //-------------------------------------------------------------------------------------------------- void RimViewLinker::updateScaleZ(RimView* sourceView, double scaleZ) { if (!isActive()) return; if (mainView() != sourceView) { RimViewLink* sourceViewLink = viewLinkForView(sourceView); CVF_ASSERT(sourceViewLink); if (!sourceViewLink->isActive() || !sourceViewLink->syncCamera()) { return; } } std::vector views; allViewsForCameraSync(sourceView, views); // Make sure scale factors are identical for (size_t i = 0; i < views.size(); i++) { views[i]->setScaleZAndUpdate(scaleZ); } } //-------------------------------------------------------------------------------------------------- /// //-------------------------------------------------------------------------------------------------- bool RimViewLinker::isActive() { RimViewLinkerCollection* viewLinkerCollection = NULL; this->firstAnchestorOrThisOfType(viewLinkerCollection); return viewLinkerCollection->isActive(); } //-------------------------------------------------------------------------------------------------- /// Hande icon update locally as PdmUiItem::updateUiIconFromState works only for static icons //-------------------------------------------------------------------------------------------------- void RimViewLinker::applyIconEnabledState(caf::PdmObject* obj, const QIcon& icon, bool disable) { QPixmap icPixmap; icPixmap = icon.pixmap(16, 16, QIcon::Normal); if (disable) { QIcon temp(icPixmap); icPixmap = temp.pixmap(16, 16, QIcon::Disabled); } QIcon newIcon(icPixmap); obj->setUiIcon(newIcon); } //-------------------------------------------------------------------------------------------------- /// //-------------------------------------------------------------------------------------------------- void RimViewLinker::updateUiIcon() { RimViewLinker::applyIconEnabledState(this, m_originalIcon, false); } //-------------------------------------------------------------------------------------------------- /// //-------------------------------------------------------------------------------------------------- void RimViewLinker::setNameAndIcon() { RimViewLinker::findNameAndIconFromView(&m_name.v(), &m_originalIcon, m_mainView); } //-------------------------------------------------------------------------------------------------- /// //-------------------------------------------------------------------------------------------------- void RimViewLinker::scheduleGeometryRegenForDepViews(RivCellSetEnum geometryType) { for (size_t i = 0; i < viewLinks.size(); i++) { if (!viewLinks[i]->isActive) continue; if ( viewLinks[i]->syncVisibleCells() || viewLinks[i]->syncPropertyFilters() || viewLinks[i]->syncRangeFilters() ) { if (viewLinks[i]->managedView()) { if (viewLinks[i]->syncVisibleCells()) { viewLinks[i]->managedView()->scheduleGeometryRegen(OVERRIDDEN_CELL_VISIBILITY); }else{ viewLinks[i]->managedView()->scheduleGeometryRegen(geometryType); } } } } } //-------------------------------------------------------------------------------------------------- /// //-------------------------------------------------------------------------------------------------- void RimViewLinker::scheduleCreateDisplayModelAndRedrawForDependentViews() { for (size_t i = 0; i < viewLinks.size(); i++) { if (!viewLinks[i]->isActive) continue; if (viewLinks[i]->syncVisibleCells() || viewLinks[i]->syncPropertyFilters() || viewLinks[i]->syncRangeFilters() ) { if (viewLinks[i]->managedView()) { viewLinks[i]->managedView()->scheduleCreateDisplayModelAndRedraw(); } } } } //-------------------------------------------------------------------------------------------------- /// //-------------------------------------------------------------------------------------------------- void RimViewLinker::removeOverrides() { for (size_t i = 0; i < viewLinks.size(); i++) { if (viewLinks[i]->managedView()) { viewLinks[i]->removeOverrides(); } } } //-------------------------------------------------------------------------------------------------- /// //-------------------------------------------------------------------------------------------------- void RimViewLinker::findNameAndIconFromView(QString* name, QIcon* icon, RimView* view) { CVF_ASSERT(name && icon); *name = displayNameForView(view); if (view) { RimCase* rimCase = NULL; view->firstAnchestorOrThisOfType(rimCase); if (dynamic_cast(rimCase)) { *icon = QIcon(":/GeoMechCase48x48.png"); } else if (dynamic_cast(rimCase)) { *icon = QIcon(":/Case48x48.png"); } else if (dynamic_cast(rimCase)) { *icon = QIcon(":/EclipseInput48x48.png"); } } else { *icon = QIcon(); } } //-------------------------------------------------------------------------------------------------- /// //-------------------------------------------------------------------------------------------------- RimViewLink* RimViewLinker::viewLinkForView(const RimView* view) { RimViewLink* viewLink = NULL; std::vector reffingObjs; view->objectsWithReferringPtrFields(reffingObjs); for (size_t i = 0; i < reffingObjs.size(); ++i) { viewLink = dynamic_cast(reffingObjs[i]); if (viewLink) break; } return viewLink; } //-------------------------------------------------------------------------------------------------- /// Return view linker if view is the main view (controlling) view //-------------------------------------------------------------------------------------------------- RimViewLinker* RimViewLinker::viewLinkerIfMainView(RimView* view) { RimViewLinker* viewLinker = NULL; std::vector reffingObjs; view->objectsWithReferringPtrFields(reffingObjs); for (size_t i = 0; i < reffingObjs.size(); ++i) { viewLinker = dynamic_cast(reffingObjs[i]); if (viewLinker) break; } return viewLinker; } //-------------------------------------------------------------------------------------------------- /// //-------------------------------------------------------------------------------------------------- RimViewLinker* RimViewLinker::viewLinkerForMainOrControlledView(RimView* view) { RimViewLinker* viewLinker = RimViewLinker::viewLinkerIfMainView(view); if (!viewLinker) { RimViewLink* viewLink = RimViewLinker::viewLinkForView(view); if (viewLink) { viewLinker = viewLink->ownerViewLinker(); } } return viewLinker; } //-------------------------------------------------------------------------------------------------- /// //-------------------------------------------------------------------------------------------------- void RimViewLinker::updateCamera(RimView* sourceView) { if (!sourceView->viewer()) return; if (!isActive()) return; RimViewLink* viewLink = RimViewLinker::viewLinkForView(sourceView); if (viewLink) { if ((!viewLink->isActive() || !viewLink->syncCamera())) { return; } } std::vector viewsToUpdate; allViewsForCameraSync(sourceView, viewsToUpdate); cvf::Vec3d sourceCamUp; cvf::Vec3d sourceCamEye; cvf::Vec3d sourceCamViewRefPoint; sourceView->viewer()->mainCamera()->toLookAt(&sourceCamEye, &sourceCamViewRefPoint, &sourceCamUp); cvf::Vec3d sourceCamGlobalEye = sourceCamEye; cvf::Vec3d sourceCamGlobalViewRefPoint = sourceCamViewRefPoint; // Source bounding box in global coordinates including scaleZ cvf::BoundingBox sourceSceneBB = sourceView->viewer()->currentScene()->boundingBox(); RimEclipseView* eclipseView = dynamic_cast(sourceView); if (eclipseView && eclipseView->eclipseCase() && eclipseView->eclipseCase()->reservoirData() && eclipseView->eclipseCase()->reservoirData()->mainGrid()) { cvf::Vec3d offset = eclipseView->eclipseCase()->reservoirData()->mainGrid()->displayModelOffset(); offset.z() *= eclipseView->scaleZ(); sourceCamGlobalEye += offset; sourceCamGlobalViewRefPoint += offset; cvf::Mat4d trans; trans.setTranslation(offset); sourceSceneBB.transform(trans); } // Propagate view matrix to all relevant views const cvf::Mat4d mat = sourceView->viewer()->mainCamera()->viewMatrix(); for (size_t i = 0; i < viewsToUpdate.size(); i++) { if (viewsToUpdate[i] && viewsToUpdate[i]->viewer()) { RiuViewer* destinationViewer = viewsToUpdate[i]->viewer(); // Destination bounding box in global coordinates including scaleZ cvf::BoundingBox destSceneBB = destinationViewer->currentScene()->boundingBox(); RimEclipseView* destEclipseView = dynamic_cast(viewsToUpdate[i]); if (destEclipseView && destEclipseView->eclipseCase() && destEclipseView->eclipseCase()->reservoirData() && destEclipseView->eclipseCase()->reservoirData()->mainGrid()) { cvf::Vec3d destOffset = destEclipseView->eclipseCase()->reservoirData()->mainGrid()->displayModelOffset(); destOffset.z() *= destEclipseView->scaleZ(); cvf::Vec3d destinationCamEye = sourceCamGlobalEye - destOffset; cvf::Vec3d destinationCamViewRefPoint = sourceCamGlobalViewRefPoint - destOffset; cvf::Mat4d trans; trans.setTranslation(destOffset); destSceneBB.transform(trans); if (isBoundingBoxesOverlappingOrClose(sourceSceneBB, destSceneBB)) { destinationViewer->mainCamera()->setFromLookAt(destinationCamEye, destinationCamViewRefPoint, sourceCamUp); } else { // Fallback using values from source camera destinationViewer->mainCamera()->setFromLookAt(sourceCamEye, sourceCamViewRefPoint, sourceCamUp); } } else { if (isBoundingBoxesOverlappingOrClose(sourceSceneBB, destSceneBB)) { destinationViewer->mainCamera()->setFromLookAt(sourceCamGlobalEye, sourceCamGlobalViewRefPoint, sourceCamUp); } else { // Fallback using values from source camera destinationViewer->mainCamera()->setFromLookAt(sourceCamEye, sourceCamViewRefPoint, sourceCamUp); } } destinationViewer->update(); } } } //-------------------------------------------------------------------------------------------------- /// //-------------------------------------------------------------------------------------------------- bool RimViewLinker::isBoundingBoxesOverlappingOrClose(const cvf::BoundingBox& sourceBB, const cvf::BoundingBox& destBB) { if (!sourceBB.isValid() || !destBB.isValid()) return false; if (sourceBB.intersects(destBB)) return true; double largestExtent = sourceBB.extent().length(); if (destBB.extent().length() > largestExtent) { largestExtent = destBB.extent().length(); } double centerDist = (sourceBB.center() - destBB.center()).length(); if (centerDist < largestExtent * 5) { return true; } return false; }