///////////////////////////////////////////////////////////////////////////////// // // 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 "RiuDragDrop.h" #include "RiaGuiApplication.h" #include "OperationsUsingObjReferences/RicPasteEclipseCasesFeature.h" #include "RicCloseCaseFeature.h" #include "WellLogCommands/RicNewWellLogFileCurveFeature.h" #include "WellLogCommands/RicWellLogPlotTrackFeatureImpl.h" #include "RimCaseCollection.h" #include "RimEclipseCase.h" #include "RimEclipseResultCase.h" #include "RimIdenticalGridCaseGroup.h" #include "RimMimeData.h" #include "RimMultiPlot.h" #include "RimPlot.h" #include "RimSummaryCase.h" #include "RimSummaryCaseCollection.h" #include "RimSummaryCaseMainCollection.h" #include "RimSummaryCurve.h" #include "RimSummaryCurveCollection.h" #include "RimSummaryPlot.h" #include "RimSurface.h" #include "RimSurfaceCollection.h" #include "RimWellAllocationPlot.h" #include "RimWellLogCurve.h" #include "RimWellLogFileChannel.h" #include "RimWellLogPlot.h" #include "RimWellLogTrack.h" #include "RiuMainWindow.h" #include "RicWellLogTools.h" #include "cafPdmUiItem.h" #include "cafPdmUiTreeView.h" #include "cafSelectionManager.h" #include #include //-------------------------------------------------------------------------------------------------- /// Utility class to enforce updates of the tree view after drop is complete. /// Does the update on destruction of the object. //-------------------------------------------------------------------------------------------------- class RiuDragAndDropTreeViewUpdater { public: RiuDragAndDropTreeViewUpdater( caf::PdmUiTreeView* treeView, const QModelIndex& indexToUpdate, const std::vector draggedUiItems ) : m_treeView( treeView ) , m_indexToUpdate( indexToUpdate ) , m_draggedUiItems( draggedUiItems ) { } ~RiuDragAndDropTreeViewUpdater() { m_treeView->updateSubTree( m_indexToUpdate ); m_treeView->selectItems( m_draggedUiItems ); } private: caf::PdmUiTreeView* m_treeView; QModelIndex m_indexToUpdate; std::vector m_draggedUiItems; }; //-------------------------------------------------------------------------------------------------- /// //-------------------------------------------------------------------------------------------------- template class RiuTypedPdmObjects { public: explicit RiuTypedPdmObjects( const caf::PdmObjectGroup& objectGroup ) { objectGroup.objectsByType( &m_objects ); } explicit RiuTypedPdmObjects( const std::vector>& objectHandles ) { for ( size_t i = 0; i < objectHandles.size(); i++ ) { caf::PdmObject* obj = dynamic_cast( objectHandles[i].p() ); if ( obj ) m_objects.push_back( obj ); } } static std::vector typedObjectsFromGroup( const caf::PdmObjectGroup& objectGroup ) { RiuTypedPdmObjects typedObjectsGetter( objectGroup ); return typedObjectsGetter.typedObjects(); } static bool containsTypedObjects( const caf::PdmObjectGroup& objectGroup ) { RiuTypedPdmObjects typedObjectsGetter( objectGroup ); return typedObjectsGetter.typedObjects().size() > 0; } static bool containsTypedObjects( const std::vector>& objectHandles ) { RiuTypedPdmObjects typedObjectsGetter( objectHandles ); return typedObjectsGetter.typedObjects().size() > 0; } static std::vector typedAncestorsFromGroup( const caf::PdmObjectGroup& objectGroup ) { RiuTypedPdmObjects typedObjectsGetter( objectGroup ); return typedObjectsGetter.typedAncestors(); } static std::vector typedAncestorsFromGroup( const std::vector>& objectHandles ) { RiuTypedPdmObjects typedObjectsGetter( objectHandles ); return typedObjectsGetter.typedAncestors(); } private: std::vector typedObjects() { std::vector typedObjectsVec; for ( size_t i = 0; i < m_objects.size(); i++ ) { T* typedObject = dynamic_cast( m_objects[i].p() ); if ( typedObject ) { typedObjectsVec.push_back( typedObject ); } } return typedObjectsVec; } std::vector typedAncestors() { std::vector typedAncestorsVec; for ( size_t i = 0; i < m_objects.size(); i++ ) { T* typedAncestor = nullptr; m_objects[i]->firstAncestorOfType( typedAncestor ); if ( typedAncestor ) { typedAncestorsVec.push_back( typedAncestor ); } } return typedAncestorsVec; } private: std::vector> m_objects; }; //-------------------------------------------------------------------------------------------------- /// //-------------------------------------------------------------------------------------------------- RiuDragDrop::RiuDragDrop() { m_proposedDropAction = Qt::MoveAction; } //-------------------------------------------------------------------------------------------------- /// //-------------------------------------------------------------------------------------------------- RiuDragDrop::~RiuDragDrop() { } //-------------------------------------------------------------------------------------------------- /// //-------------------------------------------------------------------------------------------------- Qt::DropActions RiuDragDrop::supportedDropActions() const { // Keep drag items so that we can determine allowed actions while dragging m_dragItems = objectHandlesFromSelection(); if ( RiuTypedPdmObjects::containsTypedObjects( m_dragItems ) ) { return Qt::CopyAction | Qt::MoveAction; } else if ( RiuTypedPdmObjects::containsTypedObjects( m_dragItems ) ) { return Qt::CopyAction; } else if ( RiuTypedPdmObjects::containsTypedObjects( m_dragItems ) ) { // return Qt::CopyAction; } return Qt::MoveAction; } //-------------------------------------------------------------------------------------------------- /// //-------------------------------------------------------------------------------------------------- Qt::ItemFlags RiuDragDrop::flags( const QModelIndex& index ) const { Qt::ItemFlags itemflags = nullptr; if ( index.isValid() && RiaGuiApplication::activeMainWindow() ) { caf::PdmUiTreeView* uiTreeView = RiaGuiApplication::activeMainWindow()->projectTreeView(); caf::PdmUiItem* uiItem = uiTreeView->uiItemFromModelIndex( index ); caf::PdmObject* pdmObj = dynamic_cast( uiItem ); if ( pdmObj ) { RimWellAllocationPlot* wellAllocationPlot = nullptr; pdmObj->firstAncestorOrThisOfType( wellAllocationPlot ); if ( wellAllocationPlot ) return itemflags; } if ( dynamic_cast( uiItem ) || dynamic_cast( uiItem ) || dynamic_cast( uiItem ) || dynamic_cast( uiItem ) || dynamic_cast( uiItem ) || dynamic_cast( uiItem ) || dynamic_cast( uiItem ) ) { // TODO: Remember to handle reservoir holding the main grid itemflags |= Qt::ItemIsDragEnabled; } if ( m_dragItems.empty() ) return itemflags; if ( dynamic_cast( uiItem ) ) { if ( RiuTypedPdmObjects::containsTypedObjects( m_dragItems ) ) { itemflags |= Qt::ItemIsDropEnabled; } } else if ( m_proposedDropAction == Qt::MoveAction ) { if ( dynamic_cast( uiItem ) ) { if ( RiuTypedPdmObjects::containsTypedObjects( m_dragItems ) ) { auto plotParents = RiuTypedPdmObjects::typedAncestorsFromGroup( m_dragItems ); bool draggedOntoSameParent = index.row() == -1 && plotParents.size() == 1u && plotParents.front() == uiItem; if ( !draggedOntoSameParent ) { itemflags |= Qt::ItemIsDropEnabled; } } } else if ( dynamic_cast( uiItem ) ) { if ( RiuTypedPdmObjects::containsTypedObjects( m_dragItems ) && !RiuTypedPdmObjects::containsTypedObjects( m_dragItems ) ) { itemflags |= Qt::ItemIsDropEnabled; } } else if ( dynamic_cast( uiItem ) ) { if ( RiuTypedPdmObjects::containsTypedObjects( m_dragItems ) ) { auto trackParents = RiuTypedPdmObjects::typedAncestorsFromGroup( m_dragItems ); bool draggedOntoSameParent = index.row() == -1 && trackParents.size() == 1u && trackParents.front() == uiItem; if ( !draggedOntoSameParent ) { itemflags |= Qt::ItemIsDropEnabled; } } } else if ( dynamic_cast( uiItem ) || dynamic_cast( uiItem ) ) { if ( RiuTypedPdmObjects::containsTypedObjects( m_dragItems ) ) { itemflags |= Qt::ItemIsDropEnabled; } } else if ( dynamic_cast( uiItem ) ) { if ( RiuTypedPdmObjects::containsTypedObjects( m_dragItems ) ) { itemflags |= Qt::ItemIsDropEnabled; } } else if ( dynamic_cast( uiItem ) ) { if ( RiuTypedPdmObjects::containsTypedObjects( m_dragItems ) ) { itemflags |= Qt::ItemIsDropEnabled; } } else if ( dynamic_cast( uiItem ) ) { if ( RiuTypedPdmObjects::containsTypedObjects( m_dragItems ) ) { itemflags |= Qt::ItemIsDropEnabled; } } } else if ( m_proposedDropAction == Qt::CopyAction ) { if ( dynamic_cast( uiItem ) ) { if ( RiuTypedPdmObjects::containsTypedObjects( m_dragItems ) ) { itemflags |= Qt::ItemIsDropEnabled; } } else if ( dynamic_cast( uiItem ) ) { if ( RiuTypedPdmObjects::containsTypedObjects( m_dragItems ) ) { itemflags |= Qt::ItemIsDropEnabled; } } } } return itemflags; } //-------------------------------------------------------------------------------------------------- /// //-------------------------------------------------------------------------------------------------- bool RiuDragDrop::dropMimeData( const QMimeData* data, Qt::DropAction action, int row, int column, const QModelIndex& dropTargetIndex ) { CVF_ASSERT( RiaGuiApplication::activeMainWindow() ); caf::PdmUiTreeView* uiTreeView = RiaGuiApplication::activeMainWindow()->projectTreeView(); caf::PdmUiItem* dropTargetUiItem = uiTreeView->uiItemFromModelIndex( dropTargetIndex ); caf::PdmObjectHandle* dropTarget = dynamic_cast( dropTargetUiItem ); if ( dropTarget ) { caf::PdmObjectGroup draggedObjects; const MimeDataWithIndexes* myMimeData = qobject_cast( data ); if ( myMimeData && dropTargetIndex.isValid() ) { QModelIndexList indices = myMimeData->indexes(); objectGroupFromModelIndexes( &draggedObjects, indices ); } else { return false; } RiuDragAndDropTreeViewUpdater updater( uiTreeView, dropTargetIndex.parent(), RiuTypedPdmObjects::typedObjectsFromGroup( draggedObjects ) ); RimIdenticalGridCaseGroup* gridCaseGroup; dropTarget->firstAncestorOrThisOfType( gridCaseGroup ); if ( gridCaseGroup ) { return handleGridCaseGroupDrop( action, draggedObjects, gridCaseGroup ); } RimWellLogTrack* wellLogPlotTrack; dropTarget->firstAncestorOrThisOfType( wellLogPlotTrack ); if ( wellLogPlotTrack ) { return handleWellLogPlotTrackDrop( action, draggedObjects, wellLogPlotTrack, row ); } RimWellLogPlot* wellLogPlot; dropTarget->firstAncestorOrThisOfType( wellLogPlot ); if ( wellLogPlot ) { return handleWellLogPlotDrop( action, draggedObjects, wellLogPlot, row ); } RimSummaryPlot* summaryPlot; dropTarget->firstAncestorOrThisOfType( summaryPlot ); if ( summaryPlot ) { return handleSummaryPlotDrop( action, draggedObjects, summaryPlot, row ); } RimMultiPlot* multiPlot; dropTarget->firstAncestorOrThisOfType( multiPlot ); if ( multiPlot ) { return handleMultiPlotDrop( action, draggedObjects, multiPlot, row ); } RimSummaryCaseCollection* summaryCaseCollection; dropTarget->firstAncestorOrThisOfType( summaryCaseCollection ); if ( summaryCaseCollection ) { return handleSummaryCaseCollectionDrop( action, draggedObjects, summaryCaseCollection ); } RimSummaryCaseMainCollection* summaryCaseMainCollection; dropTarget->firstAncestorOrThisOfType( summaryCaseMainCollection ); if ( summaryCaseMainCollection ) { return handleSummaryCaseMainCollectionDrop( action, draggedObjects, summaryCaseMainCollection ); } RimSurfaceCollection* surfaceCollection; dropTarget->firstAncestorOrThisOfType( surfaceCollection ); if ( surfaceCollection ) { return handleSurfaceCollectionDrop( action, row, draggedObjects, surfaceCollection ); } } return false; } //-------------------------------------------------------------------------------------------------- /// //-------------------------------------------------------------------------------------------------- QMimeData* RiuDragDrop::mimeData( const QModelIndexList& indexes ) const { MimeDataWithIndexes* myObj = new MimeDataWithIndexes(); myObj->setIndexes( indexes ); return myObj; } //-------------------------------------------------------------------------------------------------- /// //-------------------------------------------------------------------------------------------------- QStringList RiuDragDrop::mimeTypes() const { QStringList types; types << MimeDataWithIndexes::formatName(); return types; } //-------------------------------------------------------------------------------------------------- /// //-------------------------------------------------------------------------------------------------- void RiuDragDrop::onDragCanceled() { m_dragItems.clear(); } //-------------------------------------------------------------------------------------------------- /// //-------------------------------------------------------------------------------------------------- void RiuDragDrop::moveCasesToGridGroup( caf::PdmObjectGroup& objectGroup, RimIdenticalGridCaseGroup* gridCaseGroup ) { std::vector casesToBeDeleted = RiuTypedPdmObjects::typedObjectsFromGroup( objectGroup ); if ( RicCloseCaseFeature::userConfirmedGridCaseGroupChange( casesToBeDeleted ) ) { RicPasteEclipseCasesFeature::addCasesToGridCaseGroup( objectGroup, gridCaseGroup ); for ( size_t i = 0; i < casesToBeDeleted.size(); i++ ) { RicCloseCaseFeature::deleteEclipseCase( casesToBeDeleted[i] ); } } } //-------------------------------------------------------------------------------------------------- /// //-------------------------------------------------------------------------------------------------- bool RiuDragDrop::handleGridCaseGroupDrop( Qt::DropAction action, caf::PdmObjectGroup& objectGroup, RimIdenticalGridCaseGroup* gridCaseGroup ) { if ( action == Qt::CopyAction ) { RicPasteEclipseCasesFeature::addCasesToGridCaseGroup( objectGroup, gridCaseGroup ); } else if ( action == Qt::MoveAction ) { moveCasesToGridGroup( objectGroup, gridCaseGroup ); } return true; } //-------------------------------------------------------------------------------------------------- /// //-------------------------------------------------------------------------------------------------- bool RiuDragDrop::handleMultiPlotDrop( Qt::DropAction action, caf::PdmObjectGroup& draggedObjects, RimMultiPlot* multiPlot, int insertAtPosition ) { std::vector plots = RiuTypedPdmObjects::typedObjectsFromGroup( draggedObjects ); if ( plots.size() > 0 ) { if ( action == Qt::MoveAction ) { multiPlot->movePlotsToThis( plots, insertAtPosition ); return true; } } return false; } //-------------------------------------------------------------------------------------------------- /// //-------------------------------------------------------------------------------------------------- bool RiuDragDrop::handleWellLogPlotTrackDrop( Qt::DropAction action, caf::PdmObjectGroup& draggedObjects, RimWellLogTrack* trackTarget, int insertAtPosition ) { std::vector wellLogFileChannels = RiuTypedPdmObjects::typedObjectsFromGroup( draggedObjects ); if ( wellLogFileChannels.size() > 0 ) { if ( action == Qt::CopyAction ) { RicWellLogTools::addWellLogChannelsToPlotTrack( trackTarget, wellLogFileChannels ); return true; } } std::vector wellLogPlotCurves = RiuTypedPdmObjects::typedObjectsFromGroup( draggedObjects ); if ( wellLogPlotCurves.size() > 0 ) { if ( action == Qt::MoveAction ) { RicWellLogPlotTrackFeatureImpl::moveCurvesToWellLogPlotTrack( trackTarget, wellLogPlotCurves, insertAtPosition ); return true; } } return false; } //-------------------------------------------------------------------------------------------------- /// //-------------------------------------------------------------------------------------------------- bool RiuDragDrop::handleWellLogPlotDrop( Qt::DropAction action, caf::PdmObjectGroup& draggedObjects, RimWellLogPlot* wellLogPlotTarget, int insertAtPosition ) { std::vector wellLogPlotTracks = RiuTypedPdmObjects::typedObjectsFromGroup( draggedObjects ); if ( wellLogPlotTracks.size() > 0 ) { if ( action == Qt::MoveAction ) { RicWellLogPlotTrackFeatureImpl::moveTracksToWellLogPlot( wellLogPlotTarget, wellLogPlotTracks, insertAtPosition ); return true; } } return false; } //-------------------------------------------------------------------------------------------------- /// //-------------------------------------------------------------------------------------------------- bool RiuDragDrop::handleSummaryPlotDrop( Qt::DropAction action, caf::PdmObjectGroup& objectGroup, RimSummaryPlot* summaryPlot, int insertAtPosition ) { std::vector summaryCurves = RiuTypedPdmObjects::typedObjectsFromGroup( objectGroup ); if ( summaryCurves.size() > 0 ) { if ( action == Qt::MoveAction ) { RimSummaryPlot::moveCurvesToPlot( summaryPlot, summaryCurves, insertAtPosition ); return true; } } return false; } //-------------------------------------------------------------------------------------------------- /// //-------------------------------------------------------------------------------------------------- bool RiuDragDrop::handleSummaryCaseCollectionDrop( Qt::DropAction action, caf::PdmObjectGroup& draggedObjects, RimSummaryCaseCollection* summaryCaseDropTarget ) { std::vector summaryCases = RiuTypedPdmObjects::typedObjectsFromGroup( draggedObjects ); if ( action != Qt::MoveAction || summaryCases.size() == 0 ) return false; for ( RimSummaryCase* summaryCase : summaryCases ) { RimSummaryCaseCollection* summaryCaseCollection; summaryCase->firstAncestorOrThisOfType( summaryCaseCollection ); if ( summaryCaseCollection ) { summaryCaseCollection->removeCase( summaryCase ); summaryCaseDropTarget->addCase( summaryCase ); summaryCaseCollection->updateConnectedEditors(); continue; } RimSummaryCaseMainCollection* summaryCaseMainCollection; summaryCase->firstAncestorOrThisOfType( summaryCaseMainCollection ); if ( summaryCaseMainCollection ) { summaryCaseMainCollection->removeCase( summaryCase ); summaryCaseDropTarget->addCase( summaryCase ); summaryCaseMainCollection->updateConnectedEditors(); } } summaryCaseDropTarget->updateConnectedEditors(); return true; } //-------------------------------------------------------------------------------------------------- /// //-------------------------------------------------------------------------------------------------- bool RiuDragDrop::handleSummaryCaseMainCollectionDrop( Qt::DropAction action, caf::PdmObjectGroup& draggedObjects, RimSummaryCaseMainCollection* summaryCaseDropTarget ) { std::vector summaryCases = RiuTypedPdmObjects::typedObjectsFromGroup( draggedObjects ); if ( action != Qt::MoveAction || summaryCases.size() == 0 ) return false; for ( RimSummaryCase* summaryCase : summaryCases ) { RimSummaryCaseCollection* summaryCaseCollection; summaryCase->firstAncestorOrThisOfType( summaryCaseCollection ); if ( summaryCaseCollection ) { summaryCaseCollection->removeCase( summaryCase ); summaryCaseDropTarget->addCase( summaryCase ); summaryCaseCollection->updateConnectedEditors(); } } summaryCaseDropTarget->updateConnectedEditors(); return true; } //-------------------------------------------------------------------------------------------------- /// //-------------------------------------------------------------------------------------------------- void RiuDragDrop::objectGroupFromModelIndexes( caf::PdmObjectGroup* objectGroup, const QModelIndexList& indexes ) { CVF_ASSERT( objectGroup ); objectGroup->objects.clear(); caf::PdmUiTreeView* uiTreeView = RiuMainWindow::instance()->projectTreeView(); for ( int i = 0; i < indexes.size(); i++ ) { caf::PdmUiItem* uiItem = uiTreeView->uiItemFromModelIndex( indexes[i] ); caf::PdmObjectHandle* objHandle = dynamic_cast( uiItem ); if ( objHandle ) { objectGroup->objects.push_back( objHandle ); } } } //-------------------------------------------------------------------------------------------------- /// //-------------------------------------------------------------------------------------------------- std::vector> RiuDragDrop::objectHandlesFromSelection() { std::vector selection; caf::SelectionManager::instance()->objectsByType( &selection ); std::vector> objectHandles; for ( size_t sIdx = 0; sIdx < selection.size(); sIdx++ ) { objectHandles.push_back( selection[sIdx] ); } return objectHandles; } //-------------------------------------------------------------------------------------------------- /// //-------------------------------------------------------------------------------------------------- void RiuDragDrop::onProposedDropActionUpdated( Qt::DropAction action ) { m_proposedDropAction = action; } //-------------------------------------------------------------------------------------------------- /// //-------------------------------------------------------------------------------------------------- bool RiuDragDrop::handleSurfaceCollectionDrop( Qt::DropAction action, int row, caf::PdmObjectGroup& objectGroup, RimSurfaceCollection* targetCollection ) { std::vector surfaces = RiuTypedPdmObjects::typedObjectsFromGroup( objectGroup ); if ( action != Qt::MoveAction || surfaces.size() == 0 ) return false; for ( RimSurface* surface : surfaces ) { RimSurfaceCollection* sourceCollection; surface->firstAncestorOrThisOfType( sourceCollection ); if ( sourceCollection ) { sourceCollection->removeSurface( surface ); sourceCollection->updateConnectedEditors(); } } targetCollection->addSurfacesAtIndex( row, surfaces ); targetCollection->updateConnectedEditors(); return true; }