diff --git a/src/core/stac/qgsstacasset.cpp b/src/core/stac/qgsstacasset.cpp
index b2ba590dda5c..7b98b379d356 100644
--- a/src/core/stac/qgsstacasset.cpp
+++ b/src/core/stac/qgsstacasset.cpp
@@ -15,6 +15,8 @@
 
 #include "qgsstacasset.h"
 
+#include <QUrl>
+
 QgsStacAsset::QgsStacAsset( const QString &href,
                             const QString &title,
                             const QString &description,
@@ -72,3 +74,47 @@ QString QgsStacAsset::formatName() const
     return QStringLiteral( "EPT" );
   return QString();
 }
+
+QgsMimeDataUtils::Uri QgsStacAsset::uri() const
+{
+  QgsMimeDataUtils::Uri uri;
+  QUrl url( href() );
+  if ( formatName() == QLatin1String( "COG" ) )
+  {
+    uri.layerType = QStringLiteral( "raster" );
+    uri.providerKey = QStringLiteral( "gdal" );
+    if ( href().startsWith( QLatin1String( "http" ), Qt::CaseInsensitive ) ||
+         href().startsWith( QLatin1String( "ftp" ), Qt::CaseInsensitive ) )
+    {
+      uri.uri = QStringLiteral( "/vsicurl/%1" ).arg( href() );
+    }
+    else if ( href().startsWith( QLatin1String( "s3://" ), Qt::CaseInsensitive ) )
+    {
+      uri.uri = QStringLiteral( "/vsis3/%1" ).arg( href().mid( 5 ) );
+    }
+    else
+    {
+      uri.uri = href();
+    }
+  }
+  else if ( formatName() == QLatin1String( "COPC" ) )
+  {
+    uri.layerType = QStringLiteral( "pointcloud" );
+    uri.providerKey = QStringLiteral( "copc" );
+    uri.uri = href();
+  }
+  else if ( formatName() == QLatin1String( "EPT" ) )
+  {
+    uri.layerType = QStringLiteral( "pointcloud" );
+    uri.providerKey = QStringLiteral( "ept" );
+    uri.uri = href();
+  }
+  else
+  {
+    return {};
+  }
+
+  uri.name = title().isEmpty() ? url.fileName() : title();
+
+  return uri;
+}
diff --git a/src/core/stac/qgsstacasset.h b/src/core/stac/qgsstacasset.h
index c19717918fbd..5cd616b8c557 100644
--- a/src/core/stac/qgsstacasset.h
+++ b/src/core/stac/qgsstacasset.h
@@ -19,6 +19,7 @@
 #define SIP_NO_FILE
 
 #include "qgis_core.h"
+#include "qgsmimedatautils.h"
 
 #include <QString>
 #include <QStringList>
@@ -73,6 +74,12 @@ class CORE_EXPORT QgsStacAsset
      */
     QString formatName() const;
 
+    /**
+     * Returns a uri for the asset if it is a cloud optimized file like COG or COPC
+     * \since QGIS 3.42
+     */
+    QgsMimeDataUtils::Uri uri() const;
+
   private:
     QString mHref;
     QString mTitle;
diff --git a/src/core/stac/qgsstacitem.cpp b/src/core/stac/qgsstacitem.cpp
index 010d19910587..aece5886dddb 100644
--- a/src/core/stac/qgsstacitem.cpp
+++ b/src/core/stac/qgsstacitem.cpp
@@ -191,50 +191,14 @@ QString QgsStacItem::description() const
 QgsMimeDataUtils::UriList QgsStacItem::uris() const
 {
   QgsMimeDataUtils::UriList uris;
-  for ( auto it = mAssets.constBegin(); it != mAssets.constEnd(); ++it )
+  for ( const QgsStacAsset &asset : std::as_const( mAssets ) )
   {
-    QgsMimeDataUtils::Uri uri;
-    QUrl url( it->href() );
-    if ( url.isLocalFile() )
-    {
-      uri.uri = it->href();
-    }
-    else if ( it->formatName() == QLatin1String( "COG" ) )
-    {
-      uri.layerType = QStringLiteral( "raster" );
-      uri.providerKey = QStringLiteral( "gdal" );
-      if ( it->href().startsWith( QLatin1String( "http" ), Qt::CaseInsensitive ) ||
-           it->href().startsWith( QLatin1String( "ftp" ), Qt::CaseInsensitive ) )
-      {
-        uri.uri = QStringLiteral( "/vsicurl/%1" ).arg( it->href() );
-      }
-      else if ( it->href().startsWith( QLatin1String( "s3://" ), Qt::CaseInsensitive ) )
-      {
-        uri.uri = QStringLiteral( "/vsis3/%1" ).arg( it->href().mid( 5 ) );
-      }
-      else
-      {
-        uri.uri = it->href();
-      }
-    }
-    else if ( it->formatName() == QLatin1String( "COPC" ) )
-    {
-      uri.layerType = QStringLiteral( "pointcloud" );
-      uri.providerKey = QStringLiteral( "copc" );
-      uri.uri = it->href();
-    }
-    else if ( it->formatName() == QLatin1String( "EPT" ) )
-    {
-      uri.layerType = QStringLiteral( "pointcloud" );
-      uri.providerKey = QStringLiteral( "ept" );
-      uri.uri = it->href();
-    }
+    QgsMimeDataUtils::Uri uri = asset.uri();
 
     // skip assets with incompatible formats
     if ( uri.uri.isEmpty() )
       continue;
 
-    uri.name = it->title().isEmpty() ? url.fileName() : it->title();
     uris.append( uri );
   }
   return uris;
diff --git a/src/gui/stac/qgsstacdataitemguiprovider.cpp b/src/gui/stac/qgsstacdataitemguiprovider.cpp
index e093f501544e..8dc1648d711f 100644
--- a/src/gui/stac/qgsstacdataitemguiprovider.cpp
+++ b/src/gui/stac/qgsstacdataitemguiprovider.cpp
@@ -15,7 +15,6 @@
 
 #include "qgsstacdataitemguiprovider.h"
 #include "moc_qgsstacdataitemguiprovider.cpp"
-#include "qgsnetworkcontentfetcherregistry.h"
 #include "qgsstaccontroller.h"
 #include "qgsstacdataitems.h"
 #include "qgsstacconnection.h"
@@ -25,7 +24,6 @@
 #include "qgsstacitem.h"
 #include "qgsstacdownloadassetsdialog.h"
 #include "qgsstacobjectdetailsdialog.h"
-#include "qgsapplication.h"
 
 
 ///@cond PRIVATE
@@ -195,83 +193,9 @@ void QgsStacDataItemGuiProvider::downloadAssets( QgsDataItem *item, QgsDataItemG
 
   QgsStacDownloadAssetsDialog dialog;
   dialog.setStacItem( itemItem->stacItem() );
-
-  if ( dialog.exec() == QDialog::Accepted )
-  {
-    const QString folder = dialog.selectedFolder();
-    const QStringList urls = dialog.selectedUrls();
-    for ( const QString &url : urls )
-    {
-      QgsNetworkContentFetcherTask *fetcher = new QgsNetworkContentFetcherTask( url,
-          itemItem->stacController()->authCfg(),
-          QgsTask::CanCancel,
-          tr( "Downloading STAC asset" ) );
-
-      connect( fetcher, &QgsNetworkContentFetcherTask::errorOccurred, item, [context]( QNetworkReply::NetworkError, const QString & errorMsg )
-      {
-        notify( tr( "Error downloading STAC asset" ),
-                errorMsg,
-                context,
-                Qgis::MessageLevel::Critical );
-      } );
-
-      connect( fetcher, &QgsNetworkContentFetcherTask::fetched, item, [fetcher, folder, context]
-      {
-        QNetworkReply *reply = fetcher->reply();
-        if ( !reply || reply->error() != QNetworkReply::NoError )
-        {
-          // canceled or failed
-          return;
-        }
-        else
-        {
-          const QString fileName = fetcher->contentDispositionFilename().isEmpty() ? reply->url().fileName() : fetcher->contentDispositionFilename();
-          QFileInfo fi( fileName );
-          QFile file( QStringLiteral( "%1/%2" ).arg( folder, fileName ) );
-          int i = 1;
-          while ( file.exists() )
-          {
-            QString uniqueName = QStringLiteral( "%1/%2(%3)" ).arg( folder, fi.baseName() ).arg( i++ );
-            if ( !fi.completeSuffix().isEmpty() )
-              uniqueName.append( QStringLiteral( ".%1" ).arg( fi.completeSuffix() ) );
-            file.setFileName( uniqueName );
-          }
-
-          bool failed = false;
-          if ( file.open( QIODevice::WriteOnly ) )
-          {
-            const QByteArray data = reply->readAll();
-            if ( file.write( data ) < 0 )
-              failed = true;
-
-            file.close();
-          }
-          else
-          {
-            failed = true;
-          }
-
-          if ( failed )
-          {
-            notify( tr( "Error downloading STAC asset" ),
-                    tr( "Could not write to file %1" ).arg( file.fileName() ),
-                    context,
-                    Qgis::MessageLevel::Critical );
-          }
-          else
-          {
-            notify( tr( "STAC asset downloaded" ),
-                    file.fileName(),
-                    context,
-                    Qgis::MessageLevel::Success );
-          }
-        }
-      } );
-
-      QgsApplication::taskManager()->addTask( fetcher );
-    }
-  }
-
+  dialog.setMessageBar( context.messageBar() );
+  dialog.setAuthCfg( itemItem->stacController()->authCfg() );
+  dialog.exec();
 }
 
 ///@endcond
diff --git a/src/gui/stac/qgsstacdownloadassetsdialog.cpp b/src/gui/stac/qgsstacdownloadassetsdialog.cpp
index 6aeca1a88877..30b61d63cdd4 100644
--- a/src/gui/stac/qgsstacdownloadassetsdialog.cpp
+++ b/src/gui/stac/qgsstacdownloadassetsdialog.cpp
@@ -16,8 +16,11 @@
 #include "qgsstacdownloadassetsdialog.h"
 #include "moc_qgsstacdownloadassetsdialog.cpp"
 #include "qgsgui.h"
+#include "qgsnetworkcontentfetchertask.h"
 #include "qgssettings.h"
 #include "qgsproject.h"
+#include "qgsmessagebar.h"
+#include "qgsapplication.h"
 
 #include <QTreeWidget>
 #include <QPushButton>
@@ -51,6 +54,97 @@ QgsStacDownloadAssetsDialog::QgsStacDownloadAssetsDialog( QWidget *parent ) :
            this, &QgsStacDownloadAssetsDialog::showContextMenu );
 }
 
+void QgsStacDownloadAssetsDialog::accept()
+{
+  const QString folder = selectedFolder();
+  const QStringList urls = selectedUrls();
+  for ( const QString &url : urls )
+  {
+    QgsNetworkContentFetcherTask *fetcher = new QgsNetworkContentFetcherTask( url,
+        mAuthCfg,
+        QgsTask::CanCancel,
+        tr( "Downloading STAC asset" ) );
+
+    connect( fetcher, &QgsNetworkContentFetcherTask::errorOccurred, fetcher, [bar = mMessageBar]( QNetworkReply::NetworkError, const QString & errorMsg )
+    {
+      if ( bar )
+        bar->pushMessage(
+          tr( "Error downloading STAC asset" ),
+          errorMsg,
+          Qgis::MessageLevel::Critical );
+    } );
+
+    connect( fetcher, &QgsNetworkContentFetcherTask::fetched, fetcher, [fetcher, folder, bar = mMessageBar]
+    {
+      QNetworkReply *reply = fetcher->reply();
+      if ( !reply || reply->error() != QNetworkReply::NoError )
+      {
+        // canceled or failed
+        return;
+      }
+      else
+      {
+        const QString fileName = fetcher->contentDispositionFilename().isEmpty() ? reply->url().fileName() : fetcher->contentDispositionFilename();
+        QFileInfo fi( fileName );
+        QFile file( QStringLiteral( "%1/%2" ).arg( folder, fileName ) );
+        int i = 1;
+        while ( file.exists() )
+        {
+          QString uniqueName = QStringLiteral( "%1/%2(%3)" ).arg( folder, fi.baseName() ).arg( i++ );
+          if ( !fi.completeSuffix().isEmpty() )
+            uniqueName.append( QStringLiteral( ".%1" ).arg( fi.completeSuffix() ) );
+          file.setFileName( uniqueName );
+        }
+
+        bool failed = false;
+        if ( file.open( QIODevice::WriteOnly ) )
+        {
+          const QByteArray data = reply->readAll();
+          if ( file.write( data ) < 0 )
+            failed = true;
+
+          file.close();
+        }
+        else
+        {
+          failed = true;
+        }
+
+        if ( failed )
+        {
+          if ( bar )
+            bar->pushMessage(
+              tr( "Error downloading STAC asset" ),
+              tr( "Could not write to file %1" ).arg( file.fileName() ),
+              Qgis::MessageLevel::Critical );
+        }
+        else
+        {
+          if ( bar )
+            bar->pushMessage(
+              tr( "STAC asset downloaded" ),
+              file.fileName(),
+              Qgis::MessageLevel::Success );
+        }
+      }
+    } );
+
+    QgsApplication::taskManager()->addTask( fetcher );
+  }
+
+  QDialog::accept();
+}
+
+void QgsStacDownloadAssetsDialog::setAuthCfg( const QString &authCfg )
+{
+  mAuthCfg = authCfg;
+}
+
+void QgsStacDownloadAssetsDialog::setMessageBar( QgsMessageBar *bar )
+{
+  mMessageBar = bar;
+}
+
 void QgsStacDownloadAssetsDialog::setStacItem( QgsStacItem *stacItem )
 {
   if ( ! stacItem )
diff --git a/src/gui/stac/qgsstacdownloadassetsdialog.h b/src/gui/stac/qgsstacdownloadassetsdialog.h
index 5324ec46105c..bca43d4d5a25 100644
--- a/src/gui/stac/qgsstacdownloadassetsdialog.h
+++ b/src/gui/stac/qgsstacdownloadassetsdialog.h
@@ -24,6 +24,7 @@
 
 #include <QDialog>
 
+class QgsMessageBar;
 
 class QgsStacDownloadAssetsDialog : public QDialog, private Ui::QgsStacDownloadAssetsDialog
 {
@@ -32,6 +33,10 @@ class QgsStacDownloadAssetsDialog : public QDialog, private Ui::QgsStacDownloadA
   public:
     explicit QgsStacDownloadAssetsDialog( QWidget *parent = nullptr );
 
+    void accept() override;
+
+    void setAuthCfg( const QString &authCfg );
+    void setMessageBar( QgsMessageBar *bar );
     void setStacItem( QgsStacItem *stacItem );
     QString selectedFolder();
     QStringList selectedUrls();
@@ -44,6 +49,8 @@ class QgsStacDownloadAssetsDialog : public QDialog, private Ui::QgsStacDownloadA
     void deselectAll();
 
     QMenu *mContextMenu = nullptr;
+    QString mAuthCfg;
+    QgsMessageBar *mMessageBar = nullptr;
 };
 
 ///@endcond
diff --git a/src/gui/stac/qgsstacitemlistmodel.cpp b/src/gui/stac/qgsstacitemlistmodel.cpp
index 1c149a9e6f04..17a81a49f973 100644
--- a/src/gui/stac/qgsstacitemlistmodel.cpp
+++ b/src/gui/stac/qgsstacitemlistmodel.cpp
@@ -105,6 +105,10 @@ QVariant QgsStacItemListModel::data( const QModelIndex &index, int role ) const
       }
       return QStringList( formats.cbegin(), formats.cend() );
     }
+    case Role::Geometry:
+    {
+      return QVariant::fromValue( mItems.at( index.row() )->geometry() );
+    }
   }
 
   return QVariant();
@@ -176,6 +180,11 @@ void QgsStacItemListModel::addItems( const QVector<QgsStacItem *> &items )
   }
 }
 
+QVector<QgsStacItem *> QgsStacItemListModel::items() const
+{
+  return mItems;
+}
+
 
 
 QgsStacItemDelegate::QgsStacItemDelegate( QObject *parent )
diff --git a/src/gui/stac/qgsstacitemlistmodel.h b/src/gui/stac/qgsstacitemlistmodel.h
index 414821136f09..4a04737f8a0c 100644
--- a/src/gui/stac/qgsstacitemlistmodel.h
+++ b/src/gui/stac/qgsstacitemlistmodel.h
@@ -56,6 +56,8 @@ class QgsStacItemListModel : public QAbstractListModel
     void setCollections( const QVector< QgsStacCollection * > &collections );
     //! Add items to the model. Takes ownership
     void addItems( const QVector< QgsStacItem * > &items );
+    //! Returns all items in the model. Does not transfer ownership
+    QVector< QgsStacItem * > items() const;
 
   private:
     QVector< QgsStacItem * > mItems;
diff --git a/src/gui/stac/qgsstacsourceselect.cpp b/src/gui/stac/qgsstacsourceselect.cpp
index 4c64de89a4d2..eda959f02212 100644
--- a/src/gui/stac/qgsstacsourceselect.cpp
+++ b/src/gui/stac/qgsstacsourceselect.cpp
@@ -15,6 +15,7 @@
 
 #include "qgsstacsourceselect.h"
 #include "moc_qgsstacsourceselect.cpp"
+#include "qgsdatasourcemanagerdialog.h"
 #include "qgsgui.h"
 #include "qgsmapcanvas.h"
 #include "qgsstaccontroller.h"
@@ -78,7 +79,7 @@ QgsStacSourceSelect::QgsStacSourceSelect( QWidget *parent, Qt::WindowFlags fl, Q
   connect( mItemsView->selectionModel(), &QItemSelectionModel::currentChanged, this, &QgsStacSourceSelect::onCurrentItemChanged );
   connect( mItemsView->verticalScrollBar(), &QScrollBar::valueChanged, this, &QgsStacSourceSelect::onItemsViewScroll );
 
-
+  connect( mFootprintsCheckBox, &QCheckBox::clicked, this, &QgsStacSourceSelect::showFootprints );
 
   mParametersDialog = new QgsStacSearchParametersDialog( mapCanvas(), this );
   mFiltersLabel->clear();
@@ -89,8 +90,27 @@ QgsStacSourceSelect::~QgsStacSourceSelect()
   delete mStac;
 }
 
+void QgsStacSourceSelect::hideEvent( QHideEvent *e )
+{
+  if ( !e->spontaneous() )
+  {
+    showFootprints( false );
+  }
+  QgsAbstractDataSourceWidget::hideEvent( e );
+}
+
+void QgsStacSourceSelect::showEvent( QShowEvent *e )
+{
+  if ( !e->spontaneous() && mFootprintsCheckBox->isChecked() )
+  {
+    showFootprints( true );
+  }
+  QgsAbstractDataSourceWidget::showEvent( e );
+}
+
 void QgsStacSourceSelect::addButtonClicked()
 {
+  QgsTemporaryCursorOverride cursorOverride( Qt::WaitCursor );
   const QItemSelection selection = mItemsView->selectionModel()->selection();
   const QModelIndexList selectedIndices = selection.indexes();
 
@@ -103,20 +123,7 @@ void QgsStacSourceSelect::addButtonClicked()
 
   for ( auto &uri : std::as_const( allUris ) )
   {
-    if ( uri.layerType == QLatin1String( "raster" ) )
-    {
-      Q_NOWARN_DEPRECATED_PUSH
-      emit addRasterLayer( uri.uri, uri.name, uri.providerKey );
-      Q_NOWARN_DEPRECATED_POP
-      emit addLayer( Qgis::LayerType::Raster, uri.uri, uri.name, uri.providerKey );
-    }
-    else if ( uri.layerType == QLatin1String( "pointcloud" ) )
-    {
-      Q_NOWARN_DEPRECATED_PUSH
-      emit addPointCloudLayer( uri.uri, uri.name, uri.providerKey );
-      Q_NOWARN_DEPRECATED_POP
-      emit addLayer( Qgis::LayerType::PointCloud, uri.uri, uri.name, uri.providerKey );
-    }
+    loadUri( uri );
   }
 }
 
@@ -140,6 +147,9 @@ void QgsStacSourceSelect::onCurrentItemChanged( const QModelIndex &current, cons
 {
   Q_UNUSED( previous )
 
+  if ( mFootprintsCheckBox->isChecked() )
+    highlightFootprint( current );
+
   const QVariant mediaTypes = current.data( QgsStacItemListModel::Role::MediaTypes );
   emit enableButtons( !mediaTypes.toStringList().isEmpty() );
 }
@@ -156,6 +166,8 @@ void QgsStacSourceSelect::btnConnect_clicked()
   mSearchUrl.clear();
   mNextPageUrl.clear();
   mItemsModel->clear();
+  qDeleteAll( mRubberBands );
+  mRubberBands.clear();
   mStatusLabel->setText( tr( "Connecting…" ) );
   mStac->cancelPendingAsyncRequests();
   mStac->fetchStacObjectAsync( connection.url );
@@ -336,6 +348,13 @@ void QgsStacSourceSelect::onItemCollectionRequestFinished( int requestId, QStrin
   const QVector< QgsStacItem *> items = col->takeItems();
   mItemsModel->addItems( items );
 
+  for ( QgsStacItem *i : items )
+  {
+    QgsRubberBand *band = new QgsRubberBand( mapCanvas(), Qgis::GeometryType::Polygon );
+    band->setToGeometry( i->geometry(), QgsCoordinateReferenceSystem::fromEpsgId( 4326 ) );
+    mRubberBands.append( band );
+  }
+
   const int count = mItemsModel->rowCount();
   if ( mNextPageUrl.isEmpty() )
   {
@@ -436,6 +455,8 @@ void QgsStacSourceSelect::openSearchParametersDialog()
     return;
 
   mItemsModel->clear();
+  qDeleteAll( mRubberBands );
+  mRubberBands.clear();
   mItemsModel->setCollections( mParametersDialog->collections() );
   mNextPageUrl.clear();
   mStatusLabel->setText( tr( "Searching…" ) );
@@ -473,6 +494,69 @@ void QgsStacSourceSelect::showItemsContextMenu( QPoint point )
 
   QMenu *menu = new QMenu( this );
 
+  QgsMessageBar *bar = nullptr;
+  QgsDataSourceManagerDialog *dsm = qobject_cast<QgsDataSourceManagerDialog *>( window() );
+  if ( dsm )
+    bar = dsm->messageBar();
+
+  const QgsStacItem *item = dynamic_cast<QgsStacItem *>( index.data( QgsStacItemListModel::Role::StacObject ).value<QgsStacObject *>() );
+  QMenu *assetsMenu = menu->addMenu( tr( "Add Layer" ) );
+  const QMap<QString, QgsStacAsset> assets = item->assets();
+  for ( const QgsStacAsset &asset : assets )
+  {
+    if ( asset.isCloudOptimized() )
+    {
+      QAction *loadAssetAction = new QAction( asset.title(), assetsMenu );
+      connect( loadAssetAction, &QAction::triggered, this, [this, &asset]
+      {
+        QgsTemporaryCursorOverride cursorOverride( Qt::WaitCursor );
+        loadUri( asset.uri() );
+      } );
+      assetsMenu->addAction( loadAssetAction );
+    }
+  }
+
+  QAction *zoomToAction = new QAction( tr( "Zoom to Item" ), menu );
+  connect( zoomToAction, &QAction::triggered, this, [index, this]
+  {
+    QgsGeometry geom = index.data( QgsStacItemListModel::Role::Geometry ).value<QgsGeometry>();
+    if ( QgsMapCanvas *map = mapCanvas() )
+    {
+      const QgsRectangle bbox = geom.boundingBox();
+      const QgsCoordinateTransform ct( QgsCoordinateReferenceSystem::fromEpsgId( 4324 ),
+                                       map->mapSettings().destinationCrs(),
+                                       QgsProject::instance() );
+      QgsRectangle extent = ct.transformBoundingBox( bbox );
+      map->zoomToFeatureExtent( extent );
+    }
+  } );
+
+  QAction *panToAction = new QAction( tr( "Pan to Item" ), menu );
+  connect( panToAction, &QAction::triggered, this, [index, this]
+  {
+    QgsGeometry geom = index.data( QgsStacItemListModel::Role::Geometry ).value<QgsGeometry>();
+    if ( QgsMapCanvas *map = mapCanvas() )
+    {
+      const QgsRectangle bbox = geom.boundingBox();
+      const QgsCoordinateTransform ct( QgsCoordinateReferenceSystem::fromEpsgId( 4324 ),
+                                       map->mapSettings().destinationCrs(),
+                                       QgsProject::instance() );
+      const QgsRectangle extent = ct.transformBoundingBox( bbox );
+      map->setCenter( extent.center() );
+    }
+  } );
+
+  QAction *downloadAction = new QAction( tr( "Download Assets…" ), menu );
+  connect( downloadAction, &QAction::triggered, this, [index, bar, authCfg = mStac->authCfg()]
+  {
+    QgsStacDownloadAssetsDialog dialog;
+    QgsStacItem *item = dynamic_cast<QgsStacItem *>( index.data( QgsStacItemListModel::Role::StacObject ).value<QgsStacObject *>() );
+    dialog.setStacItem( item );
+    dialog.setMessageBar( bar );
+    dialog.setAuthCfg( authCfg );
+    dialog.exec();
+  } );
+
   QAction *detailsAction = new QAction( tr( "Details…" ), menu );
   connect( detailsAction, &QAction::triggered, this, [this, index]
   {
@@ -480,10 +564,70 @@ void QgsStacSourceSelect::showItemsContextMenu( QPoint point )
     details.setStacObject( index.data( QgsStacItemListModel::Role::StacObject ).value<QgsStacObject *>() );
     details.exec();
   } );
+
+
+  menu->addAction( zoomToAction );
+  menu->addAction( panToAction );
+  if ( !assetsMenu->isEmpty() )
+    menu->addMenu( assetsMenu );
+  menu->addAction( downloadAction );
   menu->addAction( detailsAction );
 
   menu->popup( mItemsView->mapToGlobal( point ) );
   connect( menu, &QMenu::aboutToHide, menu, &QMenu::deleteLater );
 }
 
+void QgsStacSourceSelect::highlightFootprint( const QModelIndex &index )
+{
+  QgsGeometry geom = index.data( QgsStacItemListModel::Role::Geometry ).value<QgsGeometry>();
+  if ( QgsMapCanvas *map = mapCanvas() )
+  {
+    mCurrentItemBand.reset( new QgsRubberBand( map, Qgis::GeometryType::Polygon ) );
+    mCurrentItemBand->setFillColor( QColor::fromRgb( 255, 0, 0, 128 ) );
+    mCurrentItemBand->setToGeometry( geom, QgsCoordinateReferenceSystem::fromEpsgId( 4326 ) );
+  }
+}
+
+void QgsStacSourceSelect::showFootprints( bool enable )
+{
+  if ( enable )
+  {
+    const QVector<QgsStacItem *> items = mItemsModel->items();
+    for ( QgsStacItem *i : items )
+    {
+      QgsRubberBand *band = new QgsRubberBand( mapCanvas(), Qgis::GeometryType::Polygon );
+      band->setToGeometry( i->geometry(), QgsCoordinateReferenceSystem::fromEpsgId( 4326 ) );
+      mRubberBands.append( band );
+    }
+    const QModelIndex index = mItemsView->selectionModel()->currentIndex();
+    if ( index.isValid() )
+    {
+      highlightFootprint( index );
+    }
+  }
+  else
+  {
+    qDeleteAll( mRubberBands );
+    mRubberBands.clear();
+    mCurrentItemBand.reset();
+  }
+}
+
+void QgsStacSourceSelect::loadUri( const QgsMimeDataUtils::Uri &uri )
+{
+  if ( uri.layerType == QLatin1String( "raster" ) )
+  {
+    Q_NOWARN_DEPRECATED_PUSH
+    emit addRasterLayer( uri.uri, uri.name, uri.providerKey );
+    Q_NOWARN_DEPRECATED_POP
+    emit addLayer( Qgis::LayerType::Raster, uri.uri, uri.name, uri.providerKey );
+  }
+  else if ( uri.layerType == QLatin1String( "pointcloud" ) )
+  {
+    Q_NOWARN_DEPRECATED_PUSH
+    emit addPointCloudLayer( uri.uri, uri.name, uri.providerKey );
+    Q_NOWARN_DEPRECATED_POP
+    emit addLayer( Qgis::LayerType::PointCloud, uri.uri, uri.name, uri.providerKey );
+  }
+}
 ///@endcond
diff --git a/src/gui/stac/qgsstacsourceselect.h b/src/gui/stac/qgsstacsourceselect.h
index 99d3f7736922..3bee8fa74f14 100644
--- a/src/gui/stac/qgsstacsourceselect.h
+++ b/src/gui/stac/qgsstacsourceselect.h
@@ -19,6 +19,8 @@
 #include "ui_qgsstacsourceselectbase.h"
 #include "qgsabstractdatasourcewidget.h"
 #include "qgis_gui.h"
+#include "qgsmimedatautils.h"
+#include "qobjectuniqueptr.h"
 
 #include <QStandardItemModel>
 #include <QStyledItemDelegate>
@@ -30,7 +32,7 @@
 class QgsStacSearchParametersDialog;
 class QgsStacItemListModel;
 class QgsStacController;
-
+class QgsRubberBand;
 
 class GUI_EXPORT QgsStacSourceSelect : public QgsAbstractDataSourceWidget, private Ui::QgsStacSourceSelectBase
 {
@@ -42,6 +44,9 @@ class GUI_EXPORT QgsStacSourceSelect : public QgsAbstractDataSourceWidget, priva
     //! Destructor
     ~QgsStacSourceSelect() override;
 
+    void hideEvent( QHideEvent *e ) override;
+    void showEvent( QShowEvent *e ) override;
+
     void addButtonClicked() override;
 
   private slots:
@@ -79,7 +84,7 @@ class GUI_EXPORT QgsStacSourceSelect : public QgsAbstractDataSourceWidget, priva
     //! Called when double clicking a result item
     void onItemDoubleClicked( const QModelIndex &index );
 
-    //! Enables Add Layers button based on current item
+    //! Enables Add Layers button based on current item, updates rubber bands
     void onCurrentItemChanged( const QModelIndex &current, const QModelIndex &previous );
 
   private:
@@ -93,6 +98,10 @@ class GUI_EXPORT QgsStacSourceSelect : public QgsAbstractDataSourceWidget, priva
 
     void showItemsContextMenu( QPoint point );
 
+    void highlightFootprint( const QModelIndex &index );
+    void showFootprints( bool enable );
+    void loadUri( const QgsMimeDataUtils::Uri &uri );
+
     QString mCollectionsUrl;
     QString mSearchUrl;
     QUrl mNextPageUrl;
@@ -100,6 +109,8 @@ class GUI_EXPORT QgsStacSourceSelect : public QgsAbstractDataSourceWidget, priva
     QgsStacController *mStac = nullptr;
     QgsStacItemListModel *mItemsModel = nullptr;
     QgsStacSearchParametersDialog *mParametersDialog = nullptr;
+    QObjectUniquePtr<QgsRubberBand> mCurrentItemBand;
+    QVector<QgsRubberBand *> mRubberBands;
 };
 
 ///@endcond
diff --git a/src/ui/qgsstacsourceselectbase.ui b/src/ui/qgsstacsourceselectbase.ui
index 382fe34a4e88..5e1e755f50e8 100644
--- a/src/ui/qgsstacsourceselectbase.ui
+++ b/src/ui/qgsstacsourceselectbase.ui
@@ -154,11 +154,38 @@
     </widget>
    </item>
    <item>
-    <widget class="QLabel" name="mStatusLabel">
-     <property name="text">
-      <string/>
-     </property>
-    </widget>
+    <layout class="QHBoxLayout" name="horizontalLayout_3">
+     <item>
+      <widget class="QLabel" name="mStatusLabel">
+       <property name="text">
+        <string/>
+       </property>
+      </widget>
+     </item>
+     <item>
+      <spacer name="horizontalSpacer_2">
+       <property name="orientation">
+        <enum>Qt::Horizontal</enum>
+       </property>
+       <property name="sizeHint" stdset="0">
+        <size>
+         <width>40</width>
+         <height>20</height>
+        </size>
+       </property>
+      </spacer>
+     </item>
+     <item>
+      <widget class="QCheckBox" name="mFootprintsCheckBox">
+       <property name="text">
+        <string>Show footprints</string>
+       </property>
+       <property name="checked">
+        <bool>true</bool>
+       </property>
+      </widget>
+     </item>
+    </layout>
    </item>
    <item>
     <widget class="QDialogButtonBox" name="buttonBox">
diff --git a/tests/src/core/testqgsstac.cpp b/tests/src/core/testqgsstac.cpp
index 02d0d1a15b35..a4cb32205dcf 100644
--- a/tests/src/core/testqgsstac.cpp
+++ b/tests/src/core/testqgsstac.cpp
@@ -200,14 +200,27 @@ void TestQgsStac::testParseLocalItem()
   QVERIFY( asset.isCloudOptimized() );
   QCOMPARE( asset.formatName(), QStringLiteral( "COG" ) );
 
+  QgsMimeDataUtils::Uri uri = asset.uri();
+  QCOMPARE( uri.uri, basePath + QStringLiteral( "20201211_223832_CS2_analytic.tif" ) );
+  QCOMPARE( uri.name, QStringLiteral( "4-Band Analytic" ) );
+  QCOMPARE( uri.layerType, QStringLiteral( "raster" ) );
+
   asset = item->assets().value( QStringLiteral( "thumbnail" ), QgsStacAsset( {}, {}, {}, {}, {} ) );
   QCOMPARE( asset.href(), QStringLiteral( "https://storage.googleapis.com/open-cogs/stac-examples/20201211_223832_CS2.jpg" ) );
   QVERIFY( !asset.isCloudOptimized() );
+  uri = asset.uri();
+  QVERIFY( !uri.isValid() );
+  QVERIFY( uri.uri.isEmpty() );
+  QVERIFY( uri.name.isEmpty() );
 
   // normal geotiff is not cloud optimized
   asset = item->assets().value( QStringLiteral( "udm" ), QgsStacAsset( {}, {}, {}, {}, {} ) );
   QVERIFY( !asset.isCloudOptimized() );
   QCOMPARE( asset.formatName(), QString() );
+  uri = asset.uri();
+  QVERIFY( !uri.isValid() );
+  QVERIFY( uri.uri.isEmpty() );
+  QVERIFY( uri.name.isEmpty() );
 
   delete item;
 }