QGIS/src/gui/qgsidentifymenu.cpp
Nyall Dawson f35745fc70 Follow up map layer action changes
- Switch to flags instead of boolean argument
- Move logic for layer validity to canRunUsingLayer
- Add unit test

Also remove settings flag to hide duplicate features action
2018-02-24 20:15:13 +11:00

655 lines
22 KiB
C++

/***************************************************************************
qgsidentifymenu.cpp - menu to be used in identify map tool
---------------------
begin : August 2014
copyright : (C) 2014 by Denis Rouzaud
email : denis.rouzaud@gmail.com
***************************************************************************
* *
* This program 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 2 of the License, or *
* (at your option) any later version. *
* *
***************************************************************************/
#include <QMouseEvent>
#include "qgsidentifymenu.h"
#include "qgsapplication.h"
#include "qgsactionmanager.h"
#include "qgshighlight.h"
#include "qgsmapcanvas.h"
#include "qgsactionmenu.h"
#include "qgsvectorlayer.h"
#include "qgslogger.h"
#include "qgssettings.h"
#include "qgsgui.h"
QgsIdentifyMenu::QgsIdentifyMenu( QgsMapCanvas *canvas )
: QMenu( canvas )
, mCanvas( canvas )
, mAllowMultipleReturn( true )
, mExecWithSingleResult( false )
, mShowFeatureActions( false )
, mResultsIfExternalAction( false )
, mMaxLayerDisplay( 10 )
, mMaxFeatureDisplay( 10 )
, mDefaultActionName( tr( "Identify" ) )
{
}
QgsIdentifyMenu::~QgsIdentifyMenu()
{
deleteRubberBands();
}
void QgsIdentifyMenu::setMaxLayerDisplay( int maxLayerDisplay )
{
if ( maxLayerDisplay < 0 )
{
QgsDebugMsg( "invalid value for number of layers displayed." );
}
mMaxLayerDisplay = maxLayerDisplay;
}
void QgsIdentifyMenu::setMaxFeatureDisplay( int maxFeatureDisplay )
{
if ( maxFeatureDisplay < 0 )
{
QgsDebugMsg( "invalid value for number of layers displayed." );
}
mMaxFeatureDisplay = maxFeatureDisplay;
}
QList<QgsMapToolIdentify::IdentifyResult> QgsIdentifyMenu::exec( const QList<QgsMapToolIdentify::IdentifyResult> &idResults, QPoint pos )
{
clear();
mLayerIdResults.clear();
QList<QgsMapToolIdentify::IdentifyResult> returnResults = QList<QgsMapToolIdentify::IdentifyResult>();
if ( idResults.isEmpty() )
{
return returnResults;
}
if ( idResults.count() == 1 && !mExecWithSingleResult )
{
returnResults << idResults[0];
return returnResults;
}
// sort results by layer
Q_FOREACH ( const QgsMapToolIdentify::IdentifyResult &result, idResults )
{
QgsMapLayer *layer = result.mLayer;
if ( mLayerIdResults.contains( layer ) )
{
mLayerIdResults[layer].append( result );
}
else
{
mLayerIdResults.insert( layer, QList<QgsMapToolIdentify::IdentifyResult>() << result );
}
}
// add results to the menu
bool singleLayer = mLayerIdResults.count() == 1;
int count = 0;
QMapIterator< QgsMapLayer *, QList<QgsMapToolIdentify::IdentifyResult> > it( mLayerIdResults );
while ( it.hasNext() )
{
if ( mMaxLayerDisplay != 0 && count > mMaxLayerDisplay )
break;
++count;
it.next();
QgsMapLayer *layer = it.key();
if ( layer->type() == QgsMapLayer::RasterLayer )
{
addRasterLayer( layer );
}
else if ( layer->type() == QgsMapLayer::VectorLayer )
{
QgsVectorLayer *vl = qobject_cast<QgsVectorLayer *>( layer );
if ( !vl )
continue;
addVectorLayer( vl, it.value(), singleLayer );
}
}
// add an "identify all" action on the top level
if ( !singleLayer && mAllowMultipleReturn && idResults.count() > 1 )
{
addSeparator();
QAction *allAction = new QAction( QgsApplication::getThemeIcon( QStringLiteral( "/mActionIdentify.svg" ) ), tr( "%1 all (%2)" ).arg( mDefaultActionName ).arg( idResults.count() ), this );
allAction->setData( QVariant::fromValue<ActionData>( ActionData( nullptr ) ) );
connect( allAction, &QAction::hovered, this, &QgsIdentifyMenu::handleMenuHover );
addAction( allAction );
}
// exec
QAction *selectedAction = QMenu::exec( pos );
bool externalAction;
returnResults = results( selectedAction, externalAction );
// delete actions
clear();
// also remove the QgsActionMenu
qDeleteAll( findChildren<QgsActionMenu *>() );
if ( externalAction && !mResultsIfExternalAction )
{
return QList<QgsMapToolIdentify::IdentifyResult>();
}
else
{
return returnResults;
}
}
void QgsIdentifyMenu::closeEvent( QCloseEvent *e )
{
deleteRubberBands();
QMenu::closeEvent( e );
}
void QgsIdentifyMenu::addRasterLayer( QgsMapLayer *layer )
{
QAction *layerAction = nullptr;
QMenu *layerMenu = nullptr;
QList<QgsMapLayerAction *> separators = QList<QgsMapLayerAction *>();
QList<QgsMapLayerAction *> layerActions = mCustomActionRegistry.mapLayerActions( layer, QgsMapLayerAction::Layer );
int nCustomActions = layerActions.count();
if ( nCustomActions )
{
separators.append( layerActions[0] );
}
if ( mShowFeatureActions )
{
layerActions.append( QgsGui::mapLayerActionRegistry()->mapLayerActions( layer, QgsMapLayerAction::Layer ) );
if ( layerActions.count() > nCustomActions )
{
separators.append( layerActions[nCustomActions] );
}
}
// use a menu only if actions will be listed
if ( layerActions.isEmpty() )
{
layerAction = new QAction( layer->name(), this );
}
else
{
layerMenu = new QMenu( layer->name(), this );
layerAction = layerMenu->menuAction();
}
// add layer action to the top menu
layerAction->setIcon( QgsApplication::getThemeIcon( QStringLiteral( "/mIconRasterLayer.svg" ) ) );
layerAction->setData( QVariant::fromValue<ActionData>( ActionData( layer ) ) );
connect( layerAction, &QAction::hovered, this, &QgsIdentifyMenu::handleMenuHover );
addAction( layerAction );
// no need to go further if there is no menu
if ( !layerMenu )
return;
// add default identify action
QAction *identifyFeatureAction = new QAction( mDefaultActionName, layerMenu );
connect( identifyFeatureAction, &QAction::hovered, this, &QgsIdentifyMenu::handleMenuHover );
identifyFeatureAction->setData( QVariant::fromValue<ActionData>( ActionData( layer ) ) );
layerMenu->addAction( identifyFeatureAction );
// add custom/layer actions
Q_FOREACH ( QgsMapLayerAction *mapLayerAction, layerActions )
{
QAction *action = new QAction( mapLayerAction->icon(), mapLayerAction->text(), layerMenu );
action->setData( QVariant::fromValue<ActionData>( ActionData( layer, true ) ) );
connect( action, &QAction::hovered, this, &QgsIdentifyMenu::handleMenuHover );
connect( action, &QAction::triggered, this, &QgsIdentifyMenu::triggerMapLayerAction );
layerMenu->addAction( action );
if ( separators.contains( mapLayerAction ) )
{
layerMenu->insertSeparator( action );
}
}
}
void QgsIdentifyMenu::addVectorLayer( QgsVectorLayer *layer, const QList<QgsMapToolIdentify::IdentifyResult> &results, bool singleLayer )
{
QAction *layerAction = nullptr;
QMenu *layerMenu = nullptr;
// do not add actions with MultipleFeatures as target if only 1 feature is found for this layer
// targets defines which actions will be shown
QgsMapLayerAction::Targets targets = results.count() > 1 ? QgsMapLayerAction::Layer | QgsMapLayerAction::MultipleFeatures : QgsMapLayerAction::Layer;
QList<QgsMapLayerAction *> separators = QList<QgsMapLayerAction *>();
QList<QgsMapLayerAction *> layerActions = mCustomActionRegistry.mapLayerActions( layer, targets );
int nCustomActions = layerActions.count();
if ( nCustomActions )
{
separators << layerActions[0];
}
if ( mShowFeatureActions )
{
layerActions << QgsGui::mapLayerActionRegistry()->mapLayerActions( layer, targets );
if ( layerActions.count() > nCustomActions )
{
separators << layerActions[nCustomActions];
}
}
// determines if a menu should be created or not. Following cases:
// 1. only one result and no feature action to be shown => just create an action
// 2. several features (2a) or display feature actions (2b) => create a menu
// 3. case 2 but only one layer (singeLayer) => do not create a menu, but give the top menu instead
bool createMenu = results.count() > 1 || !layerActions.isEmpty();
// case 2b: still create a menu for layer, if there is a sub-level for features
// i.e custom actions or map layer actions at feature level
if ( !createMenu )
{
createMenu = !mCustomActionRegistry.mapLayerActions( layer, QgsMapLayerAction::SingleFeature ).isEmpty();
if ( !createMenu && mShowFeatureActions )
{
QgsActionMenu *featureActionMenu = new QgsActionMenu( layer, results[0].mFeature, QStringLiteral( "Feature" ), this );
featureActionMenu->setMode( QgsAttributeForm::IdentifyMode );
createMenu = !featureActionMenu->actions().isEmpty();
delete featureActionMenu;
}
}
// use a menu only if actions will be listed
if ( !createMenu )
{
// case 1
QString featureTitle = results[0].mFeature.attribute( layer->displayField() ).toString();
if ( featureTitle.isEmpty() )
featureTitle = QStringLiteral( "%1" ).arg( results[0].mFeature.id() );
layerAction = new QAction( QStringLiteral( "%1 (%2)" ).arg( layer->name(), featureTitle ), this );
}
else
{
if ( singleLayer )
{
// case 3
layerMenu = this;
}
else
{
// case 2a
if ( results.count() > 1 )
{
layerMenu = new QMenu( layer->name(), this );
}
// case 2b
else
{
QString featureTitle = results[0].mFeature.attribute( layer->displayField() ).toString();
if ( featureTitle.isEmpty() )
featureTitle = QStringLiteral( "%1" ).arg( results[0].mFeature.id() );
layerMenu = new QMenu( QStringLiteral( "%1 (%2)" ).arg( layer->name(), featureTitle ), this );
}
layerAction = layerMenu->menuAction();
}
}
// case 1 or 2
if ( layerAction )
{
// icons
switch ( layer->geometryType() )
{
case QgsWkbTypes::PointGeometry:
layerAction->setIcon( QgsApplication::getThemeIcon( QStringLiteral( "/mIconPointLayer.svg" ) ) );
break;
case QgsWkbTypes::LineGeometry:
layerAction->setIcon( QgsApplication::getThemeIcon( QStringLiteral( "/mIconLineLayer.svg" ) ) );
break;
case QgsWkbTypes::PolygonGeometry:
layerAction->setIcon( QgsApplication::getThemeIcon( QStringLiteral( "/mIconPolygonLayer.svg" ) ) );
break;
default:
break;
}
// add layer action to the top menu
layerAction->setData( QVariant::fromValue<ActionData>( ActionData( layer ) ) );
connect( layerAction, &QAction::hovered, this, &QgsIdentifyMenu::handleMenuHover );
addAction( layerAction );
}
// case 1. no need to go further
if ( !layerMenu )
return;
// add results to the menu
int count = 0;
Q_FOREACH ( const QgsMapToolIdentify::IdentifyResult &result, results )
{
if ( mMaxFeatureDisplay != 0 && count > mMaxFeatureDisplay )
break;
++count;
QAction *featureAction = nullptr;
QMenu *featureMenu = nullptr;
QgsActionMenu *featureActionMenu = nullptr;
QList<QgsMapLayerAction *> customFeatureActions = mCustomActionRegistry.mapLayerActions( layer, QgsMapLayerAction::SingleFeature );
if ( mShowFeatureActions )
{
featureActionMenu = new QgsActionMenu( layer, result.mFeature, QStringLiteral( "Feature" ), layerMenu );
featureActionMenu->setMode( QgsAttributeForm::IdentifyMode );
featureActionMenu->setExpressionContextScope( mExpressionContextScope );
}
// feature title
QString featureTitle = result.mFeature.attribute( layer->displayField() ).toString();
if ( featureTitle.isEmpty() )
featureTitle = QStringLiteral( "%1" ).arg( result.mFeature.id() );
if ( customFeatureActions.isEmpty() && ( !featureActionMenu || featureActionMenu->actions().isEmpty() ) )
{
featureAction = new QAction( featureTitle, layerMenu );
// add the feature action (or menu) to the layer menu
featureAction->setData( QVariant::fromValue<ActionData>( ActionData( layer, result.mFeature.id() ) ) );
connect( featureAction, &QAction::hovered, this, &QgsIdentifyMenu::handleMenuHover );
layerMenu->addAction( featureAction );
}
else if ( results.count() == 1 )
{
// if we are here with only one results, this means there is a sub-feature level (for actions)
// => skip the feature level since there would be only a single entry
// => give the layer menu as pointer instead of a new feature menu
featureMenu = layerMenu;
}
else
{
featureMenu = new QMenu( featureTitle, layerMenu );
// get the action from the menu
featureAction = featureMenu->menuAction();
// add the feature action (or menu) to the layer menu
featureAction->setData( QVariant::fromValue<ActionData>( ActionData( layer, result.mFeature.id() ) ) );
connect( featureAction, &QAction::hovered, this, &QgsIdentifyMenu::handleMenuHover );
layerMenu->addAction( featureAction );
}
// if no feature menu, no need to go further
if ( !featureMenu )
continue;
// add default identify action
QAction *identifyFeatureAction = new QAction( QgsApplication::getThemeIcon( QStringLiteral( "/mActionIdentify.svg" ) ), mDefaultActionName, featureMenu );
connect( identifyFeatureAction, &QAction::hovered, this, &QgsIdentifyMenu::handleMenuHover );
identifyFeatureAction->setData( QVariant::fromValue<ActionData>( ActionData( layer, result.mFeature.id() ) ) );
featureMenu->addAction( identifyFeatureAction );
featureMenu->addSeparator();
// custom action at feature level
Q_FOREACH ( QgsMapLayerAction *mapLayerAction, customFeatureActions )
{
QAction *action = new QAction( mapLayerAction->icon(), mapLayerAction->text(), featureMenu );
action->setData( QVariant::fromValue<ActionData>( ActionData( layer, result.mFeature.id(), mapLayerAction ) ) );
connect( action, &QAction::hovered, this, &QgsIdentifyMenu::handleMenuHover );
connect( action, &QAction::triggered, this, &QgsIdentifyMenu::triggerMapLayerAction );
featureMenu->addAction( action );
}
// use QgsActionMenu for feature actions
if ( featureActionMenu )
{
Q_FOREACH ( QAction *action, featureActionMenu->actions() )
{
connect( action, &QAction::hovered, this, &QgsIdentifyMenu::handleMenuHover );
featureMenu->addAction( action );
}
}
}
// back to layer level
// identify all action
if ( mAllowMultipleReturn && results.count() > 1 )
{
layerMenu->addSeparator();
QAction *allAction = new QAction( QgsApplication::getThemeIcon( QStringLiteral( "/mActionIdentify.svg" ) ), tr( "%1 all (%2)" ).arg( mDefaultActionName ).arg( results.count() ), layerMenu );
allAction->setData( QVariant::fromValue<ActionData>( ActionData( layer ) ) );
connect( allAction, &QAction::hovered, this, &QgsIdentifyMenu::handleMenuHover );
layerMenu->addAction( allAction );
}
// add custom/layer actions
Q_FOREACH ( QgsMapLayerAction *mapLayerAction, layerActions )
{
QString title = mapLayerAction->text();
if ( mapLayerAction->targets().testFlag( QgsMapLayerAction::MultipleFeatures ) )
title.append( QStringLiteral( " (%1)" ).arg( results.count() ) );
QAction *action = new QAction( mapLayerAction->icon(), title, layerMenu );
action->setData( QVariant::fromValue<ActionData>( ActionData( layer, mapLayerAction ) ) );
connect( action, &QAction::hovered, this, &QgsIdentifyMenu::handleMenuHover );
connect( action, &QAction::triggered, this, &QgsIdentifyMenu::triggerMapLayerAction );
layerMenu->addAction( action );
if ( separators.contains( mapLayerAction ) )
{
layerMenu->insertSeparator( action );
}
}
}
void QgsIdentifyMenu::triggerMapLayerAction()
{
QAction *action = qobject_cast<QAction *>( sender() );
if ( !action )
return;
QVariant varData = action->data();
if ( !varData.isValid() || !varData.canConvert<ActionData>() )
return;
ActionData actData = action->data().value<ActionData>();
if ( actData.mIsValid && actData.mMapLayerAction )
{
// layer
if ( actData.mMapLayerAction->targets().testFlag( QgsMapLayerAction::Layer ) )
{
actData.mMapLayerAction->triggerForLayer( actData.mLayer );
}
// multiples features
if ( actData.mMapLayerAction->targets().testFlag( QgsMapLayerAction::MultipleFeatures ) )
{
QList<QgsFeature> featureList;
Q_FOREACH ( const QgsMapToolIdentify::IdentifyResult &result, mLayerIdResults[actData.mLayer] )
{
featureList << result.mFeature;
}
actData.mMapLayerAction->triggerForFeatures( actData.mLayer, featureList );
}
// single feature
if ( actData.mMapLayerAction->targets().testFlag( QgsMapLayerAction::SingleFeature ) )
{
Q_FOREACH ( const QgsMapToolIdentify::IdentifyResult &result, mLayerIdResults[actData.mLayer] )
{
if ( result.mFeature.id() == actData.mFeatureId )
{
actData.mMapLayerAction->triggerForFeature( actData.mLayer, new QgsFeature( result.mFeature ) );
return;
}
}
QgsDebugMsg( QString( "Identify menu: could not retrieve feature for action %1" ).arg( action->text() ) );
}
}
}
QList<QgsMapToolIdentify::IdentifyResult> QgsIdentifyMenu::results( QAction *action, bool &externalAction )
{
QList<QgsMapToolIdentify::IdentifyResult> idResults = QList<QgsMapToolIdentify::IdentifyResult>();
externalAction = false;
ActionData actData;
bool hasData = false;
if ( !action )
return idResults;
QVariant varData = action->data();
if ( !varData.isValid() )
{
QgsDebugMsg( "Identify menu: could not retrieve results from menu entry (invalid data)" );
return idResults;
}
if ( varData.canConvert<ActionData>() )
{
actData = action->data().value<ActionData>();
if ( actData.mIsValid )
{
externalAction = actData.mIsExternalAction;
hasData = true;
}
}
if ( !hasData && varData.canConvert<QgsActionMenu::ActionData>() )
{
QgsActionMenu::ActionData dataSrc = action->data().value<QgsActionMenu::ActionData>();
if ( dataSrc.actionType != QgsActionMenu::Invalid )
{
externalAction = true;
actData = ActionData( dataSrc.mapLayer, dataSrc.featureId );
hasData = true;
}
}
if ( !hasData )
{
QgsDebugMsg( "Identify menu: could not retrieve results from menu entry (no data found)" );
return idResults;
}
// return all results
if ( actData.mAllResults )
{
// this means "All" action was triggered
QMapIterator< QgsMapLayer *, QList<QgsMapToolIdentify::IdentifyResult> > it( mLayerIdResults );
while ( it.hasNext() )
{
it.next();
idResults << it.value();
}
return idResults;
}
if ( !mLayerIdResults.contains( actData.mLayer ) )
{
QgsDebugMsg( "Identify menu: could not retrieve results from menu entry (layer not found)" );
return idResults;
}
if ( actData.mLevel == LayerLevel )
{
return mLayerIdResults[actData.mLayer];
}
if ( actData.mLevel == FeatureLevel )
{
Q_FOREACH ( const QgsMapToolIdentify::IdentifyResult &res, mLayerIdResults[actData.mLayer] )
{
if ( res.mFeature.id() == actData.mFeatureId )
{
idResults << res;
return idResults;
}
}
}
QgsDebugMsg( "Identify menu: could not retrieve results from menu entry (don't know what happened')" );
return idResults;
}
void QgsIdentifyMenu::handleMenuHover()
{
if ( !mCanvas )
return;
deleteRubberBands();
QAction *senderAction = qobject_cast<QAction *>( sender() );
if ( !senderAction )
return;
bool externalAction;
QList<QgsMapToolIdentify::IdentifyResult> idResults = results( senderAction, externalAction );
Q_FOREACH ( const QgsMapToolIdentify::IdentifyResult &result, idResults )
{
QgsVectorLayer *vl = qobject_cast<QgsVectorLayer *>( result.mLayer );
if ( !vl )
continue;
QgsHighlight *hl = new QgsHighlight( mCanvas, result.mFeature.geometry(), vl );
QgsSettings settings;
QColor color = QColor( settings.value( QStringLiteral( "Map/highlight/color" ), Qgis::DEFAULT_HIGHLIGHT_COLOR.name() ).toString() );
int alpha = settings.value( QStringLiteral( "Map/highlight/colorAlpha" ), Qgis::DEFAULT_HIGHLIGHT_COLOR.alpha() ).toInt();
double buffer = settings.value( QStringLiteral( "Map/highlight/buffer" ), Qgis::DEFAULT_HIGHLIGHT_BUFFER_MM ).toDouble();
double minWidth = settings.value( QStringLiteral( "Map/highlight/minWidth" ), Qgis::DEFAULT_HIGHLIGHT_MIN_WIDTH_MM ).toDouble();
hl->setColor( color ); // sets also fill with default alpha
color.setAlpha( alpha );
hl->setFillColor( color ); // sets fill with alpha
hl->setBuffer( buffer );
hl->setMinWidth( minWidth );
mRubberBands.append( hl );
connect( vl, &QObject::destroyed, this, &QgsIdentifyMenu::layerDestroyed );
}
}
void QgsIdentifyMenu::deleteRubberBands()
{
QList<QgsHighlight *>::const_iterator it = mRubberBands.constBegin();
for ( ; it != mRubberBands.constEnd(); ++it )
delete *it;
mRubberBands.clear();
}
void QgsIdentifyMenu::layerDestroyed()
{
QList<QgsHighlight *>::iterator it = mRubberBands.begin();
while ( it != mRubberBands.end() )
{
if ( ( *it )->layer() == sender() )
{
delete *it;
it = mRubberBands.erase( it );
}
else
{
++it;
}
}
}
void QgsIdentifyMenu::removeCustomActions()
{
mCustomActionRegistry.clear();
}
void QgsIdentifyMenu::setExpressionContextScope( const QgsExpressionContextScope &scope )
{
mExpressionContextScope = scope;
}
QgsExpressionContextScope QgsIdentifyMenu::expressionContextScope() const
{
return mExpressionContextScope;
}