mirror of
https://github.com/qgis/QGIS.git
synced 2025-04-09 00:05:52 -04:00
1378 lines
44 KiB
C++
1378 lines
44 KiB
C++
/***************************************************************************
|
|
qgspluginmanager.cpp - description
|
|
-------------------
|
|
begin : Someday 2003
|
|
copyright : (C) 2003 by Gary E.Sherman
|
|
email : sherman at mrcc.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 <math.h>
|
|
|
|
#include <QApplication>
|
|
#include <QFileDialog>
|
|
#include <QLineEdit>
|
|
#include <QMessageBox>
|
|
#include <QLibrary>
|
|
#include <QSettings>
|
|
#include <QStandardItem>
|
|
#include <QPushButton>
|
|
#include <QRegExp>
|
|
#include <QSortFilterProxyModel>
|
|
#include <QActionGroup>
|
|
#include <QTextStream>
|
|
|
|
#include "qgis.h"
|
|
#include "qgsapplication.h"
|
|
#include "qgsconfig.h"
|
|
#include "qgsproviderregistry.h"
|
|
#include "qgspluginregistry.h"
|
|
#include "qgspythonrunner.h"
|
|
#include "qgspluginmanager.h"
|
|
#include "qgisplugin.h"
|
|
#include "qgslogger.h"
|
|
|
|
// Do we need this?
|
|
// #define TESTLIB
|
|
#ifdef TESTLIB
|
|
// This doesn't work on WIN32 and causes problems with plugins
|
|
// on OS X (the code doesn't cause a problem but including dlfcn.h
|
|
// renders plugins unloadable)
|
|
#if !defined(WIN32) && !defined(Q_OS_MACX)
|
|
#include <dlfcn.h>
|
|
#endif
|
|
#endif
|
|
|
|
|
|
QgsPluginManager::QgsPluginManager( QWidget * parent, bool pluginsAreEnabled, Qt::WFlags fl )
|
|
: QgsOptionsDialogBase( "PluginManager", parent, fl )
|
|
{
|
|
// initialize pointer
|
|
mPythonUtils = NULL;
|
|
|
|
setupUi( this );
|
|
|
|
// QgsOptionsDialogBase handles saving/restoring of geometry, splitter and current tab states,
|
|
// switching vertical tabs between icon/text to icon-only modes (splitter collapsed to left),
|
|
// and connecting QDialogButtonBox's accepted/rejected signals to dialog's accept/reject slots
|
|
initOptionsBase( true );
|
|
|
|
// Don't let QgsOptionsDialogBase to narrow the vertical tab list widget
|
|
mOptListWidget->setMaximumWidth( 16777215 );
|
|
|
|
// Restiore UI state for widgets not handled by QgsOptionsDialogBase
|
|
QSettings settings;
|
|
mPluginsDetailsSplitter->restoreState( settings.value( QString( "/Windows/PluginManager/secondSplitterState" ) ).toByteArray() );
|
|
|
|
// load translated description strings from qgspluginmanager_texts
|
|
initTabDescriptions();
|
|
|
|
// set internal variable
|
|
mPluginsAreEnabled = pluginsAreEnabled;
|
|
|
|
// Init models
|
|
mModelPlugins = new QStandardItemModel( 0, 1 );
|
|
mModelProxy = new QgsPluginSortFilterProxyModel( this );
|
|
mModelProxy->setSourceModel( mModelPlugins );
|
|
mModelProxy->setSortCaseSensitivity( Qt::CaseInsensitive );
|
|
mModelProxy->setSortRole( Qt::DisplayRole );
|
|
mModelProxy->setDynamicSortFilter( true );
|
|
mModelProxy->sort( 0, Qt::AscendingOrder );
|
|
vwPlugins->setModel( mModelProxy );
|
|
vwPlugins->setFocus();
|
|
|
|
// Preset widgets
|
|
leFilter->setFocus( Qt::MouseFocusReason );
|
|
rbFilterNames->setChecked( true );
|
|
|
|
// Don't restore the last used tab from QSettings
|
|
mOptionsListWidget->setCurrentRow( 0 );
|
|
|
|
// Connect other signals
|
|
connect( mOptionsListWidget, SIGNAL( currentRowChanged( int ) ), this, SLOT( setCurrentTab( int ) ) );
|
|
connect( vwPlugins->selectionModel(), SIGNAL( currentChanged( const QModelIndex &, const QModelIndex & ) ), this, SLOT( currentPluginChanged( const QModelIndex & ) ) );
|
|
connect( mModelPlugins, SIGNAL( itemChanged( QStandardItem * ) ), this, SLOT( pluginItemChanged( QStandardItem * ) ) );
|
|
|
|
// Force setting the status filter (if the active tab was 0, the setCurrentRow( 0 ) above doesn't take any action)
|
|
setCurrentTab( 0 );
|
|
|
|
// Hide widgets only suitable with Python support enabled (they will be uncovered back in setPythonUtils)
|
|
rbFilterTags->hide();
|
|
rbFilterAuthors->hide();
|
|
buttonUpgradeAll->hide();
|
|
buttonInstall->hide();
|
|
buttonUninstall->hide();
|
|
frameSettings->setHidden( true );
|
|
}
|
|
|
|
|
|
|
|
QgsPluginManager::~QgsPluginManager()
|
|
{
|
|
delete mModelProxy;
|
|
delete mModelPlugins;
|
|
|
|
QSettings settings;
|
|
settings.setValue( QString( "/Windows/PluginManager/secondSplitterState" ), mPluginsDetailsSplitter->saveState() );
|
|
}
|
|
|
|
|
|
|
|
void QgsPluginManager::setPythonUtils( QgsPythonUtils* pythonUtils )
|
|
{
|
|
mPythonUtils = pythonUtils;
|
|
|
|
// Now enable Python support:
|
|
// Show and preset widgets only suitable when Python support active
|
|
rbFilterTags->show();
|
|
rbFilterAuthors->show();
|
|
buttonUpgradeAll->show();
|
|
buttonInstall->show();
|
|
buttonUninstall->show();
|
|
frameSettings->setHidden( false );
|
|
labelNoPython->setHidden( true );
|
|
buttonRefreshRepos->setEnabled( false );
|
|
buttonEditRep->setEnabled( false );
|
|
buttonDeleteRep->setEnabled( false );
|
|
|
|
// Add context menu to the plugins list view
|
|
QAction* actionSortByName = new QAction( tr( "sort by name" ), vwPlugins );
|
|
QAction* actionSortByDownloads = new QAction( tr( "sort by downloads" ), vwPlugins );
|
|
QAction* actionSortByVote = new QAction( tr( "sort by vote" ), vwPlugins );
|
|
QAction* actionSortByStatus = new QAction( tr( "sort by status" ), vwPlugins );
|
|
actionSortByName->setCheckable( true );
|
|
actionSortByDownloads->setCheckable( true );
|
|
actionSortByVote->setCheckable( true );
|
|
actionSortByStatus->setCheckable( true );
|
|
QActionGroup * group = new QActionGroup( vwPlugins );
|
|
actionSortByName->setActionGroup( group );
|
|
actionSortByDownloads->setActionGroup( group );
|
|
actionSortByVote->setActionGroup( group );
|
|
actionSortByStatus->setActionGroup( group );
|
|
actionSortByName->setChecked( true );
|
|
vwPlugins->addAction( actionSortByName );
|
|
vwPlugins->addAction( actionSortByDownloads );
|
|
vwPlugins->addAction( actionSortByVote );
|
|
vwPlugins->addAction( actionSortByStatus );
|
|
vwPlugins->setContextMenuPolicy( Qt::ActionsContextMenu );
|
|
connect( actionSortByName, SIGNAL( triggered( ) ), mModelProxy, SLOT( sortPluginsByName( ) ) );
|
|
connect( actionSortByDownloads, SIGNAL( triggered( ) ), mModelProxy, SLOT( sortPluginsByDownloads( ) ) );
|
|
connect( actionSortByVote, SIGNAL( triggered( ) ), mModelProxy, SLOT( sortPluginsByVote( ) ) );
|
|
connect( actionSortByStatus, SIGNAL( triggered( ) ), mModelProxy, SLOT( sortPluginsByStatus( ) ) );
|
|
|
|
// get the QSettings group from the installer
|
|
QString settingsGroup;
|
|
QgsPythonRunner::eval( "pyplugin_installer.instance().exportSettingsGroup()", settingsGroup );
|
|
|
|
// Initialize list of allowed checking intervals
|
|
mCheckingOnStartIntervals << 0 << 1 << 3 << 7 << 14 << 30 ;
|
|
|
|
// Initialize the "Settings" tab widgets
|
|
QSettings settings;
|
|
if ( settings.value( settingsGroup + "/checkOnStart", false ).toBool() )
|
|
{
|
|
ckbCheckUpdates->setChecked( true );
|
|
}
|
|
|
|
if ( settings.value( settingsGroup + "/allowExperimental", false ).toBool() )
|
|
{
|
|
ckbExperimental->setChecked( true );
|
|
}
|
|
|
|
int interval = settings.value( settingsGroup + "/checkOnStartInterval", "" ).toInt( );
|
|
int indx = mCheckingOnStartIntervals.indexOf( interval ); // if none found, just use -1 index.
|
|
comboInterval->setCurrentIndex( indx );
|
|
}
|
|
|
|
|
|
|
|
void QgsPluginManager::loadPlugin( QString id )
|
|
{
|
|
const QMap<QString, QString>* plugin = pluginMetadata( id );
|
|
|
|
if ( ! plugin )
|
|
{
|
|
return;
|
|
}
|
|
|
|
QApplication::setOverrideCursor( Qt::WaitCursor );
|
|
|
|
QgsPluginRegistry *pRegistry = QgsPluginRegistry::instance();
|
|
QString library = plugin->value( "library" );
|
|
if ( plugin->value( "pythonic" ) == "true" )
|
|
{
|
|
library = plugin->value( "id" );
|
|
QgsDebugMsg( "Loading Python plugin: " + library );
|
|
pRegistry->loadPythonPlugin( library );
|
|
}
|
|
else // C++ plugin
|
|
{
|
|
QgsDebugMsg( "Loading C++ plugin: " + library );
|
|
pRegistry->loadCppPlugin( library );
|
|
}
|
|
|
|
QgsDebugMsg( "Plugin loaded: " + library );
|
|
QApplication::restoreOverrideCursor();
|
|
}
|
|
|
|
|
|
|
|
void QgsPluginManager::unloadPlugin( QString id )
|
|
{
|
|
const QMap<QString, QString>* plugin = pluginMetadata( id );
|
|
|
|
if ( ! plugin )
|
|
{
|
|
return;
|
|
}
|
|
|
|
QgsPluginRegistry *pRegistry = QgsPluginRegistry::instance();
|
|
QString library = plugin->value( "library" );
|
|
|
|
if ( plugin->value( "pythonic" ) == "true" )
|
|
{
|
|
library = plugin->value( "id" );
|
|
QgsDebugMsg( "Unloading Python plugin: " + library );
|
|
pRegistry->unloadPythonPlugin( library );
|
|
}
|
|
else // C++ plugin
|
|
{
|
|
QgsDebugMsg( "Unloading C++ plugin: " + library );
|
|
pRegistry->unloadCppPlugin( library );
|
|
}
|
|
}
|
|
|
|
|
|
|
|
void QgsPluginManager::savePluginState( QString id, bool state )
|
|
{
|
|
const QMap<QString, QString>* plugin = pluginMetadata( id );
|
|
if ( ! plugin )
|
|
{
|
|
return;
|
|
}
|
|
|
|
QSettings settings;
|
|
if ( plugin->value( "pythonic" ) == "true" )
|
|
{
|
|
// Python plugin
|
|
settings.setValue( "/PythonPlugins/" + id, state );
|
|
}
|
|
else
|
|
{
|
|
// C++ plugin
|
|
// Trim "cpp:" prefix from cpp plugin id
|
|
id = id.mid( 4 );
|
|
settings.setValue( "/Plugins/" + id, state );
|
|
}
|
|
}
|
|
|
|
|
|
|
|
void QgsPluginManager::getCppPluginsMetadata()
|
|
{
|
|
QString sharedLibExtension;
|
|
#if defined(WIN32) || defined(__CYGWIN__)
|
|
sharedLibExtension = "*.dll";
|
|
#else
|
|
sharedLibExtension = "*.so*";
|
|
#endif
|
|
|
|
// check all libs in the current ans user plugins directories, and get name and descriptions
|
|
// First, the qgis install directory/lib (this info is available from the provider registry so we use it here)
|
|
QgsProviderRegistry *pr = QgsProviderRegistry::instance();
|
|
QStringList myPathList( pr->libraryDirectory().path() );
|
|
|
|
QSettings settings;
|
|
QString myPaths = settings.value( "plugins/searchPathsForPlugins", "" ).toString();
|
|
if ( !myPaths.isEmpty() )
|
|
{
|
|
myPathList.append( myPaths.split( "|" ) );
|
|
}
|
|
|
|
for ( int j = 0; j < myPathList.size(); ++j )
|
|
{
|
|
QString myPluginDir = myPathList.at( j );
|
|
QDir pluginDir( myPluginDir, sharedLibExtension, QDir::Name | QDir::IgnoreCase, QDir::Files | QDir::NoSymLinks );
|
|
|
|
if ( pluginDir.count() == 0 )
|
|
{
|
|
QMessageBox::information( this, tr( "No Plugins" ), tr( "No QGIS plugins found in %1" ).arg( myPluginDir ) );
|
|
return;
|
|
}
|
|
|
|
for ( uint i = 0; i < pluginDir.count(); i++ )
|
|
{
|
|
QString lib = QString( "%1/%2" ).arg( myPluginDir ).arg( pluginDir[i] );
|
|
|
|
#ifdef TESTLIB
|
|
// This doesn't work on WIN32 and causes problems with plugins
|
|
// on OS X (the code doesn't cause a problem but including dlfcn.h
|
|
// renders plugins unloadable)
|
|
#if !defined(WIN32) && !defined(Q_OS_MACX)
|
|
// test code to help debug loading problems
|
|
// This doesn't work on WIN32 and causes problems with plugins
|
|
// on OS X (the code doesn't cause a problem but including dlfcn.h
|
|
// renders plugins unloadable)
|
|
|
|
//void *handle = dlopen( (const char *) lib, RTLD_LAZY);
|
|
void *handle = dlopen( lib.toLocal8Bit().data(), RTLD_LAZY | RTLD_GLOBAL );
|
|
if ( !handle )
|
|
{
|
|
QgsDebugMsg( "Error in dlopen: " );
|
|
QgsDebugMsg( dlerror() );
|
|
}
|
|
else
|
|
{
|
|
QgsDebugMsg( "dlopen suceeded for " + lib );
|
|
dlclose( handle );
|
|
}
|
|
#endif //#ifndef WIN32 && Q_OS_MACX
|
|
#endif //#ifdef TESTLIB
|
|
|
|
QgsDebugMsg( "Examining: " + lib );
|
|
QLibrary *myLib = new QLibrary( lib );
|
|
bool loaded = myLib->load();
|
|
if ( !loaded )
|
|
{
|
|
QgsDebugMsg( QString( "Failed to load: %1 (%2)" ).arg( myLib->fileName() ).arg( myLib->errorString() ) );
|
|
delete myLib;
|
|
continue;
|
|
}
|
|
|
|
QgsDebugMsg( "Loaded library: " + myLib->fileName() );
|
|
|
|
// Don't bother with libraries that are providers
|
|
//if(!myLib->resolve( "isProvider" ) )
|
|
|
|
//MH: Replaced to allow for plugins that are linked to providers
|
|
//type is only used in non-provider plugins
|
|
if ( !myLib->resolve( "type" ) )
|
|
{
|
|
delete myLib;
|
|
continue;
|
|
}
|
|
|
|
// resolve the metadata from plugin
|
|
name_t *pName = ( name_t * ) cast_to_fptr( myLib->resolve( "name" ) );
|
|
description_t *pDesc = ( description_t * ) cast_to_fptr( myLib->resolve( "description" ) );
|
|
category_t *pCat = ( category_t * ) cast_to_fptr( myLib->resolve( "category" ) );
|
|
version_t *pVersion = ( version_t * ) cast_to_fptr( myLib->resolve( "version" ) );
|
|
icon_t* pIcon = ( icon_t * ) cast_to_fptr( myLib->resolve( "icon" ) );
|
|
experimental_t *pExperimental = ( experimental_t * ) cast_to_fptr( myLib->resolve( "experimental" ) );
|
|
|
|
// show the values (or lack of) for each function
|
|
if ( pName )
|
|
{
|
|
QgsDebugMsg( "Plugin name: " + pName() );
|
|
}
|
|
else
|
|
{
|
|
QgsDebugMsg( "Plugin name not returned when queried" );
|
|
}
|
|
if ( pDesc )
|
|
{
|
|
QgsDebugMsg( "Plugin description: " + pDesc() );
|
|
}
|
|
else
|
|
{
|
|
QgsDebugMsg( "Plugin description not returned when queried" );
|
|
}
|
|
if ( pCat )
|
|
{
|
|
QgsDebugMsg( "Plugin category: " + pCat() );
|
|
}
|
|
else
|
|
{
|
|
QgsDebugMsg( "Plugin category not returned when queried" );
|
|
}
|
|
if ( pVersion )
|
|
{
|
|
QgsDebugMsg( "Plugin version: " + pVersion() );
|
|
}
|
|
else
|
|
{
|
|
QgsDebugMsg( "Plugin version not returned when queried" );
|
|
}
|
|
if ( pIcon )
|
|
{
|
|
QgsDebugMsg( "Plugin icon: " + pIcon() );
|
|
}
|
|
|
|
if ( !pName || !pDesc || !pVersion )
|
|
{
|
|
QgsDebugMsg( "Failed to get name, description, or type for " + myLib->fileName() );
|
|
delete myLib;
|
|
continue;
|
|
}
|
|
|
|
// Add "cpp:" prefix in case of two: Python and C++ plugins with the same name
|
|
QString baseName = "cpp:" + QFileInfo( lib ).baseName();
|
|
|
|
QMap<QString, QString> metadata;
|
|
metadata["id"] = baseName;
|
|
metadata["name"] = pName();
|
|
metadata["description"] = pDesc();
|
|
metadata["category"] = ( pCat ? pCat() : tr( "Plugins" ) );
|
|
metadata["version_installed"] = pVersion();
|
|
metadata["icon"] = ( pIcon ? pIcon() : QString() );
|
|
metadata["library"] = myLib->fileName();
|
|
metadata["pythonic"] = "false";
|
|
metadata["installed"] = "true";
|
|
metadata["readonly"] = "true";
|
|
metadata["status"] = "orphan";
|
|
metadata["experimental"] = ( pExperimental ? pExperimental() : QString() );
|
|
mPlugins.insert( baseName, metadata );
|
|
|
|
delete myLib;
|
|
}
|
|
}
|
|
}
|
|
|
|
|
|
|
|
QStandardItem * QgsPluginManager::createSpacerItem( QString text, QString value )
|
|
{
|
|
QStandardItem * mySpacerltem = new QStandardItem( text );
|
|
mySpacerltem->setData( value, PLUGIN_STATUS_ROLE );
|
|
mySpacerltem->setData( "status", SPACER_ROLE );
|
|
mySpacerltem->setEnabled( false );
|
|
mySpacerltem->setEditable( false );
|
|
QFont font = mySpacerltem->font();
|
|
font.setBold( true );
|
|
mySpacerltem->setFont( font );
|
|
mySpacerltem->setTextAlignment( Qt::AlignHCenter );
|
|
return mySpacerltem;
|
|
}
|
|
|
|
|
|
|
|
void QgsPluginManager::reloadModelData()
|
|
{
|
|
mModelPlugins->clear();
|
|
|
|
for ( QMap<QString, QMap<QString, QString> >::iterator it = mPlugins.begin();
|
|
it != mPlugins.end();
|
|
it++ )
|
|
{
|
|
if ( ! it->value( "id" ).isEmpty() )
|
|
{
|
|
QString baseName = it->value( "id" );
|
|
QString pluginName = it->value( "name" );
|
|
QString description = it->value( "description" );
|
|
QString author = it->value( "author_name" );
|
|
QString iconPath = it->value( "icon" );
|
|
QString status = it->value( "status" );
|
|
QString error = it->value( "error" );
|
|
|
|
QStandardItem * mypDetailItem = new QStandardItem( pluginName.left( 32 ) );
|
|
|
|
mypDetailItem->setData( baseName, PLUGIN_BASE_NAME_ROLE );
|
|
mypDetailItem->setData( status, PLUGIN_STATUS_ROLE );
|
|
mypDetailItem->setData( error, PLUGIN_ERROR_ROLE );
|
|
mypDetailItem->setData( description, PLUGIN_DESCRIPTION_ROLE );
|
|
mypDetailItem->setData( author, PLUGIN_AUTHOR_ROLE );
|
|
mypDetailItem->setData( it->value( "tags" ), PLUGIN_TAGS_ROLE );
|
|
mypDetailItem->setData( it->value( "downloads" ).rightJustified( 10, '0' ), PLUGIN_DOWNLOADS_ROLE );
|
|
mypDetailItem->setData( it->value( "zip_repository" ), PLUGIN_REPOSITORY_ROLE );
|
|
mypDetailItem->setData( it->value( "average_vote" ), PLUGIN_VOTE_ROLE );
|
|
|
|
if ( QFileInfo( iconPath ).isFile() )
|
|
{
|
|
mypDetailItem->setIcon( QPixmap( iconPath ) );
|
|
}
|
|
else
|
|
{
|
|
mypDetailItem->setIcon( QPixmap( QgsApplication::defaultThemePath() + "/plugin.png" ) );
|
|
}
|
|
|
|
mypDetailItem->setEditable( false );
|
|
|
|
// set item display style
|
|
if ( ! it->value( "error" ).isEmpty() )
|
|
{
|
|
QBrush brush = mypDetailItem->foreground();
|
|
brush.setColor( Qt::red );
|
|
mypDetailItem->setForeground( brush );
|
|
}
|
|
if ( ! it->value( "error" ).isEmpty() || it->value( "status" ) == "upgradeable" || it->value( "status" ) == "new" )
|
|
{
|
|
QFont font = mypDetailItem->font();
|
|
font.setBold( true );
|
|
mypDetailItem->setFont( font );
|
|
}
|
|
|
|
// Set checkable if the plugin is installed and not disabled due to incompatibility.
|
|
// Broken plugins are checkable to to allow disabling them
|
|
mypDetailItem->setCheckable( it->value( "installed" ) == "true" && it->value( "error" ) != "incompatible" );
|
|
|
|
// Set ckeckState depending on the plugin is loaded or not.
|
|
// Initially mark all unchecked, then overwrite state of loaded ones with checked.
|
|
// Only do it with installed plugins, not not initialize checkboxes of not installed plugins at all.
|
|
if ( it->value( "installed" ) == "true" )
|
|
{
|
|
mypDetailItem->setCheckState( Qt::Unchecked );
|
|
}
|
|
|
|
if ( isPluginEnabled( it->value( "id" ) ) )
|
|
{
|
|
mypDetailItem->setCheckState( Qt::Checked );
|
|
}
|
|
|
|
// Add items to model
|
|
mModelPlugins->appendRow( mypDetailItem );
|
|
|
|
// Repaint the details view if the currently displayed data are changed.
|
|
if ( baseName == mCurrentlyDisplayedPlugin )
|
|
{
|
|
showPluginDetails( mypDetailItem );
|
|
}
|
|
}
|
|
}
|
|
|
|
// Add spacers for sort by status
|
|
if ( mPythonUtils && mPythonUtils->isEnabled() )
|
|
{
|
|
// TODO: implement better sort method instead of these dummy -Z statuses
|
|
mModelPlugins->appendRow( createSpacerItem( tr( "Only locally available", "category: plugins that are only locally available" ), "orphanZ" ) );
|
|
if ( hasReinstallablePlugins() ) mModelPlugins->appendRow( createSpacerItem( tr( "Reinstallable", "category: plugins that are installed and available" ) , "installedZ" ) );
|
|
if ( hasUpgradeablePlugins() ) mModelPlugins->appendRow( createSpacerItem( tr( "Upgradeable", "category: plugins that are installed and there is a newer version available" ), "upgradeableZ" ) );
|
|
if ( hasNewerPlugins() ) mModelPlugins->appendRow( createSpacerItem( tr( "Downgradeable", "category: plugins that are installed and there is an OLDER version available" ), "newerZ" ) );
|
|
}
|
|
|
|
updateTabTitle();
|
|
|
|
buttonUpgradeAll->setEnabled( hasUpgradeablePlugins() );
|
|
|
|
// Disable tabs that are empty because of no suitable plugins in the model.
|
|
mOptionsListWidget->item( 1 )->setHidden( ! hasAvailablePlugins() );
|
|
mOptionsListWidget->item( 2 )->setHidden( ! hasUpgradeablePlugins() );
|
|
mOptionsListWidget->item( 3 )->setHidden( ! hasNewPlugins() );
|
|
mOptionsListWidget->item( 4 )->setHidden( ! hasInvalidPlugins() );
|
|
}
|
|
|
|
|
|
|
|
void QgsPluginManager::pluginItemChanged( QStandardItem * item )
|
|
{
|
|
QString id = item->data( PLUGIN_BASE_NAME_ROLE ).toString();
|
|
|
|
if ( item->checkState() )
|
|
{
|
|
if ( mPluginsAreEnabled && ! isPluginEnabled( id ) )
|
|
{
|
|
QgsDebugMsg( " Loading plugin: " + id );
|
|
loadPlugin( id );
|
|
}
|
|
else
|
|
{
|
|
// only enable the plugin, as we're in --noplugins mode
|
|
QgsDebugMsg( " Enabling plugin: " + id );
|
|
savePluginState( id, true );
|
|
}
|
|
}
|
|
else if ( ! item->checkState() )
|
|
{
|
|
QgsDebugMsg( " Unloading plugin: " + id );
|
|
unloadPlugin( id );
|
|
}
|
|
}
|
|
|
|
|
|
|
|
void QgsPluginManager::showPluginDetails( QStandardItem * item )
|
|
{
|
|
const QMap<QString, QString> * metadata = pluginMetadata( item->data( PLUGIN_BASE_NAME_ROLE ).toString() );
|
|
|
|
if ( ! metadata ) return;
|
|
|
|
QString html = "";
|
|
|
|
// // A future mockup for install/uninstall html controls
|
|
// "<table bgcolor=\"#CCCCCC\" cellspacing=\"5\" cellpadding=\"10\" width=\"100%\" height=\"100px\">"
|
|
// "<tr><td colspan=\"3\"><b>This plugin is installed</b></td></tr>"
|
|
// "<tr>"
|
|
// " <td></td>"
|
|
// " <td bgcolor=\"#AAFFAA\" align=\"right\"><a href=\"foo\" style=\"text-decoration:none;\" ><b>REINSTALL</b></a></td>"
|
|
// " <td bgcolor=\"#FFFFAA\" align=\"right\"><a href=\"foo\" style=\"text-decoration:none;\" ><b>UNINSTALL</b></a></td>"
|
|
// "</tr>"
|
|
// "</table><br/>";
|
|
// // -----------------------------------------
|
|
// // Print all tags for debug purposes
|
|
// QList<QString> keys = metadata->keys();
|
|
// for ( int i=0; i < keys.size(); ++i )
|
|
// {
|
|
// html += QString( "%1: %2 <br/>" ).arg( keys.at( i ) ).arg( metadata->value( keys.at( i ) ) );
|
|
// }
|
|
|
|
// First prepare message box(es)
|
|
if ( ! metadata->value( "error" ).isEmpty() )
|
|
{
|
|
QString errorMsg;
|
|
if ( metadata->value( "error" ) == "incompatible" )
|
|
{
|
|
errorMsg = QString( "<b>%1</b><br/>%2" ).arg( tr( "This plugin is incompatible with this version of QGIS" ) ).arg( tr( "Plugin designed for QGIS %1", "compatible QGIS version(s)" ).arg( metadata->value( "error_details" ) ) );
|
|
}
|
|
else if ( metadata->value( "error" ) == "dependent" )
|
|
{
|
|
errorMsg = QString( "<b>%1:</b><br/>%2" ).arg( tr( "This plugin requires a missing module" ) ).arg( metadata->value( "error_details" ) );
|
|
}
|
|
else
|
|
{
|
|
errorMsg = QString( "<b>%1</b><br/>%2" ).arg( tr( "This plugin is broken" ) ).arg( metadata->value( "error_details" ) );
|
|
}
|
|
html += QString( "<table bgcolor=\"#FFFF88\" cellspacing=\"2\" cellpadding=\"6\" width=\"100%\">"
|
|
" <tr><td width=\"100%\" style=\"color:#CC0000\">%1</td></tr>"
|
|
"</table>" ).arg( errorMsg );
|
|
}
|
|
if ( metadata->value( "status" ) == "upgradeable" )
|
|
{
|
|
html += QString( "<table bgcolor=\"#FFFFAA\" cellspacing=\"2\" cellpadding=\"6\" width=\"100%\">"
|
|
" <tr><td width=\"100%\" style=\"color:#880000\"><b>%1</b></td></tr>"
|
|
"</table>" ).arg( tr( "There is a new version available" ) );
|
|
};
|
|
if ( metadata->value( "status" ) == "new" )
|
|
{
|
|
html += QString( "<table bgcolor=\"#CCFFCC\" cellspacing=\"2\" cellpadding=\"6\" width=\"100%\">"
|
|
" <tr><td width=\"100%\" style=\"color:#008800\"><b>%1</b></td></tr>"
|
|
"</table>" ).arg( tr( "This is a new plugin" ) );
|
|
};
|
|
if ( metadata->value( "status" ) == "newer" )
|
|
{
|
|
html += QString( "<table bgcolor=\"#FFFFCC\" cellspacing=\"2\" cellpadding=\"6\" width=\"100%\">"
|
|
" <tr><td width=\"100%\" style=\"color:#550000\"><b>%1</b></td></tr>"
|
|
"</table>" ).arg( tr( "Installed version of this plugin is higher than any version found in repository" ) );
|
|
};
|
|
if ( metadata->value( "experimental" ) == "true" )
|
|
{
|
|
html += QString( "<table bgcolor=\"#EEEEBB\" cellspacing=\"2\" cellpadding=\"2\" width=\"100%\">"
|
|
" <tr><td width=\"100%\" style=\"color:#660000\">"
|
|
" <img src=\":/images/themes/default/pluginExperimental.png\" width=\"32\"><b>%1</b>"
|
|
" </td></tr>"
|
|
"</table>" ).arg( tr( "This plugin is experimental" ) );
|
|
};
|
|
if ( metadata->value( "deprecated" ) == "true" )
|
|
{
|
|
html += QString( "<table bgcolor=\"#EEBBCC\" cellspacing=\"2\" cellpadding=\"2\" width=\"100%\">"
|
|
" <tr><td width=\"100%\" style=\"color:#660000\">"
|
|
" <img src=\":/images/themes/default/pluginDeprecated.png\" width=\"32\"><b>%1</b>"
|
|
" </td></tr>"
|
|
"</table>" ).arg( tr( "This plugin is deprecated" ) );
|
|
};
|
|
// if ( metadata->value( "status" ) == t.b.d. )
|
|
// {
|
|
// html += QString( "<table bgcolor=\"#CCCCFF\" cellspacing=\"2\" cellpadding=\"6\" width=\"100%\">"
|
|
// " <tr><td width=\"100%\" style=\"color:#000088\"><b>%1</b></td></tr>"
|
|
// "</table>" ).arg( tr( "Installing..." ) );
|
|
// };
|
|
|
|
// Now the metadata
|
|
html += "<table cellspacing=\"10\" width=\"100%\"><tr><td>";
|
|
html += QString( "<h1>%1</h1>" ).arg( metadata->value( "name" ) );
|
|
|
|
if ( QFileInfo( metadata->value( "icon" ) ).isFile() )
|
|
{
|
|
html += QString( "<img src=\"%1\" style=\"float:right;\">" ).arg( metadata->value( "icon" ) );
|
|
}
|
|
|
|
html += QString( "<h3>%1</h3>" ).arg( metadata->value( "description" ) );
|
|
|
|
if ( ! metadata->value( "about" ).isEmpty() )
|
|
{
|
|
html += metadata->value( "about" );
|
|
}
|
|
|
|
html += "<table><tr><td align='right' width='100%'>";
|
|
if ( ! metadata->value( "average_vote" ).isEmpty() && metadata->value( "average_vote" ).toFloat() )
|
|
{
|
|
// draw stars
|
|
int stars = qRound( metadata->value( "average_vote" ).toFloat() );
|
|
for ( int i = 0; i < stars; i++ )
|
|
{
|
|
html += "<img src=\":/images/themes/default/mIconNew.png\">";
|
|
}
|
|
html += tr( "<br/>%1 rating vote(s)<br/>" ).arg( metadata->value( "rating_votes" ) );
|
|
}
|
|
else if ( ! metadata->value( "downloads" ).isEmpty() )
|
|
{
|
|
// spacer between description and downloads
|
|
html += "<br/>";
|
|
}
|
|
if ( ! metadata->value( "downloads" ).isEmpty() )
|
|
{
|
|
html += tr( "%1 downloads" ).arg( metadata->value( "downloads" ) );
|
|
}
|
|
html += "</td></tr></table><br/>";
|
|
|
|
if ( ! metadata->value( "category" ).isEmpty() )
|
|
{
|
|
html += QString( "%1: %2 <br/>" ).arg( tr( "Category" ) ).arg( metadata->value( "category" ) );
|
|
}
|
|
if ( ! metadata->value( "tags" ).isEmpty() )
|
|
{
|
|
html += QString( "%1: %2 <br/>" ).arg( tr( "Tags" ) ).arg( metadata->value( "tags" ) );
|
|
}
|
|
if ( ! metadata->value( "homepage" ).isEmpty() || ! metadata->value( "tracker" ).isEmpty() || ! metadata->value( "code_repository" ).isEmpty() )
|
|
{
|
|
html += QString( "%1: " ).arg( tr( "More info" ) );
|
|
if ( ! metadata->value( "homepage" ).isEmpty() )
|
|
{
|
|
html += QString( "<a href='%1'>%2</a> " ).arg( metadata->value( "homepage" ) ).arg( tr( "homepage" ) );
|
|
}
|
|
if ( ! metadata->value( "tracker" ).isEmpty() )
|
|
{
|
|
html += QString( "<a href='%1'>%2</a> " ).arg( metadata->value( "tracker" ) ).arg( tr( "tracker" ) );
|
|
}
|
|
if ( ! metadata->value( "code_repository" ).isEmpty() )
|
|
{
|
|
html += QString( "<a href='%1'>%2</a>" ).arg( metadata->value( "code_repository" ) ).arg( tr( "code_ repository" ) );
|
|
}
|
|
html += "<br/>";
|
|
}
|
|
html += "<br/>" ;
|
|
|
|
if ( ! metadata->value( "author_email" ).isEmpty() )
|
|
{
|
|
html += QString( "%1: <a href='mailto:%2'>%3</a>" ).arg( tr( "Author" ) ).arg( metadata->value( "author_email" ) ).arg( metadata->value( "author_name" ) );
|
|
html += "<br/><br/>" ;
|
|
}
|
|
else if ( ! metadata->value( "author_name" ).isEmpty() )
|
|
{
|
|
html += QString( "%1: %2" ).arg( tr( "Author" ) ).arg( metadata->value( "author_name" ) );
|
|
html += "<br/><br/>" ;
|
|
}
|
|
|
|
if ( ! metadata->value( "version_installed" ).isEmpty() )
|
|
{
|
|
QString ver = metadata->value( "version_installed" );
|
|
if ( ver == "-1" ) ver = "?";
|
|
html += tr( "Installed version: %1 (in %2)<br/>" ).arg( ver ).arg( metadata->value( "library" ) );
|
|
}
|
|
if ( ! metadata->value( "version_available" ).isEmpty() )
|
|
{
|
|
html += tr( "Available version: %1 (in %2)<br/>" ).arg( metadata->value( "version_available" ) ).arg( metadata->value( "zip_repository" ) );
|
|
}
|
|
|
|
if ( ! metadata->value( "changelog" ).isEmpty() )
|
|
{
|
|
html += "<br/>" ;
|
|
QString changelog = tr( "changelog:<br/>%1 <br/>" ).arg( metadata->value( "changelog" ) );
|
|
html += changelog.replace( "\n", "<br/>" );
|
|
}
|
|
|
|
html += "</td></tr></table>";
|
|
|
|
tbDetails->setHtml( html );
|
|
|
|
// Set buttonInstall text (and sometimes focus)
|
|
buttonInstall->setDefault( false );
|
|
if ( metadata->value( "status" ) == "upgradeable" )
|
|
{
|
|
buttonInstall->setText( tr( "Upgrade plugin" ) );
|
|
buttonInstall->setDefault( true );
|
|
}
|
|
else if ( metadata->value( "status" ) == "newer" )
|
|
{
|
|
buttonInstall->setText( tr( "Downgrade plugin" ) );
|
|
}
|
|
else if ( metadata->value( "status" ) == "not installed" || metadata->value( "status" ) == "new" )
|
|
{
|
|
buttonInstall->setText( tr( "Install plugin" ) );
|
|
}
|
|
else
|
|
{
|
|
// Default (will be grayed out if not available for reinstallation)
|
|
buttonInstall->setText( tr( "Reinstall plugin" ) );
|
|
}
|
|
|
|
// Enable/disable buttons
|
|
buttonInstall->setEnabled( metadata->value( "pythonic" ).toUpper() == "TRUE" && metadata->value( "status" ) != "orphan" );
|
|
buttonUninstall->setEnabled( metadata->value( "pythonic" ).toUpper() == "TRUE" && metadata->value( "readonly" ) != "true" && metadata->value( "status" ) != "not installed" && metadata->value( "status" ) != "new" );
|
|
buttonUninstall->setHidden( metadata->value( "status" ) == "not installed" || metadata->value( "status" ) == "new" );
|
|
|
|
// Store the id of the currently displayed plugin
|
|
mCurrentlyDisplayedPlugin = metadata->value( "id" );
|
|
}
|
|
|
|
|
|
|
|
void QgsPluginManager::selectTabItem( int idx )
|
|
{
|
|
mOptionsListWidget->setCurrentRow( idx );
|
|
}
|
|
|
|
|
|
|
|
void QgsPluginManager::clearPythonPluginMetadata( )
|
|
{
|
|
for ( QMap<QString, QMap<QString, QString> >::iterator it = mPlugins.begin();
|
|
it != mPlugins.end();
|
|
)
|
|
{
|
|
if ( it->value( "pythonic" ) == "true" )
|
|
{
|
|
it = mPlugins.erase( it );
|
|
}
|
|
else
|
|
{
|
|
++it;
|
|
}
|
|
}
|
|
}
|
|
|
|
|
|
|
|
void QgsPluginManager::addPluginMetadata( QString key, QMap<QString, QString> metadata )
|
|
{
|
|
mPlugins.insert( key, metadata );
|
|
}
|
|
|
|
|
|
|
|
const QMap<QString, QString> * QgsPluginManager::pluginMetadata( QString key ) const
|
|
{
|
|
QMap<QString, QMap<QString, QString> >::const_iterator it = mPlugins.find( key );
|
|
if ( it != mPlugins.end() )
|
|
{
|
|
return &it.value();
|
|
}
|
|
return NULL;
|
|
}
|
|
|
|
|
|
|
|
//! Clear the repository listWidget
|
|
void QgsPluginManager::clearRepositoryList()
|
|
{
|
|
treeRepositories->clear();
|
|
buttonRefreshRepos->setEnabled( false );
|
|
buttonEditRep->setEnabled( false );
|
|
buttonDeleteRep->setEnabled( false );
|
|
foreach ( QAction * action, treeRepositories->actions() )
|
|
{
|
|
treeRepositories->removeAction( action );
|
|
}
|
|
}
|
|
|
|
|
|
|
|
//! Add repository to the repository listWidget
|
|
void QgsPluginManager::addToRepositoryList( QMap<QString, QString> repository )
|
|
{
|
|
// If it's the second item on the tree, change the button text to plural form and add the filter context menu
|
|
if ( buttonRefreshRepos->isEnabled() && treeRepositories->actions().count() < 1 )
|
|
{
|
|
buttonRefreshRepos->setText( tr( "Reload all repositories" ) );
|
|
QAction* actionEnableThisRepositoryOnly = new QAction( tr( "Only show plugins from selected repository" ), treeRepositories );
|
|
treeRepositories->addAction( actionEnableThisRepositoryOnly );
|
|
connect( actionEnableThisRepositoryOnly, SIGNAL( triggered() ), this, SLOT( setRepositoryFilter() ) );
|
|
treeRepositories->setContextMenuPolicy( Qt::ActionsContextMenu );
|
|
QAction* actionClearFilter = new QAction( tr( "Clear filter" ), treeRepositories );
|
|
actionClearFilter->setEnabled( repository.value( "inspection_filter" ) == "true" );
|
|
treeRepositories->addAction( actionClearFilter );
|
|
connect( actionClearFilter, SIGNAL( triggered( ) ), this, SLOT( clearRepositoryFilter() ) );
|
|
}
|
|
|
|
QString key = repository.value( "name" );
|
|
if ( ! key.isEmpty() )
|
|
{
|
|
QTreeWidgetItem * a = new QTreeWidgetItem( treeRepositories );
|
|
a->setText( 1, key );
|
|
a->setText( 2, repository.value( "url" ) );
|
|
if ( repository.value( "enabled" ) == "true" && repository.value( "valid" ) == "true" )
|
|
{
|
|
if ( repository.value( "state" ) == "2" )
|
|
{
|
|
a->setText( 0, tr( "connected" ) );
|
|
a->setIcon( 0, QIcon( ":/images/themes/default/repositoryConnected.png" ) );
|
|
a->setToolTip( 0, tr( "The repository is connected" ) );
|
|
}
|
|
else
|
|
{
|
|
a->setText( 0, tr( "unavailable" ) );
|
|
a->setIcon( 0, QIcon( ":/images/themes/default/repositoryUnavailable.png" ) );
|
|
a->setToolTip( 0, tr( "The repository is enabled, but unavailable" ) );
|
|
}
|
|
}
|
|
else
|
|
{
|
|
a->setText( 0, tr( "disabled" ) );
|
|
a->setIcon( 0, QIcon( ":/images/themes/default/repositoryDisabled.png" ) );
|
|
if ( repository.value( "valid" ) == "true" )
|
|
{
|
|
a->setToolTip( 0, tr( "The repository is disabled" ) );
|
|
}
|
|
else
|
|
{
|
|
a->setToolTip( 0, tr( "The repository is blocked due to incompatibility with your QGIS version" ) );
|
|
}
|
|
|
|
QBrush grayBrush = QBrush( QColor( Qt::gray ) );
|
|
a->setForeground( 0, grayBrush );
|
|
a->setForeground( 1, grayBrush );
|
|
a->setForeground( 2, grayBrush );
|
|
}
|
|
}
|
|
treeRepositories->resizeColumnToContents( 0 );
|
|
treeRepositories->resizeColumnToContents( 1 );
|
|
treeRepositories->resizeColumnToContents( 2 );
|
|
treeRepositories->sortItems( 1, Qt::AscendingOrder );
|
|
buttonRefreshRepos->setEnabled( true );
|
|
}
|
|
|
|
|
|
|
|
// SLOTS ///////////////////////////////////////////////////////////////////
|
|
|
|
|
|
// "Close" button clicked
|
|
void QgsPluginManager::reject()
|
|
{
|
|
if ( mPythonUtils && mPythonUtils->isEnabled() )
|
|
{
|
|
// get the QSettings group from the installer
|
|
QString settingsGroup;
|
|
QgsPythonRunner::eval( "pyplugin_installer.instance().exportSettingsGroup()", settingsGroup );
|
|
QSettings settings;
|
|
settings.setValue( settingsGroup + "/checkOnStart", QVariant( ckbCheckUpdates->isChecked() ) );
|
|
settings.setValue( settingsGroup + "/checkOnStartInterval", QVariant( mCheckingOnStartIntervals.value( comboInterval->currentIndex() ) ) );
|
|
QgsPythonRunner::run( "pyplugin_installer.instance().onManagerClose()" );
|
|
}
|
|
done( 1 );
|
|
}
|
|
|
|
|
|
|
|
void QgsPluginManager::setCurrentTab( int idx )
|
|
{
|
|
if ( idx == ( mOptionsListWidget->count() - 1 ) )
|
|
{
|
|
QgsDebugMsg( "Switching current tab to Settings" );
|
|
mOptionsStackedWidget->setCurrentIndex( 1 );
|
|
}
|
|
else
|
|
{
|
|
QgsDebugMsg( "Switching current tab to Plugins" );
|
|
mOptionsStackedWidget->setCurrentIndex( 0 );
|
|
|
|
QStringList acceptedStatuses;
|
|
QString tabTitle;
|
|
switch ( idx )
|
|
{
|
|
case 0:
|
|
// installed (statuses ends with Z are for spacers to always sort properly)
|
|
acceptedStatuses << "installed" << "orphan" << "newer" << "upgradeable" << "installedZ" << "upgradeableZ" << "orphanZ" << "newerZZ" << "" ;
|
|
tabTitle = "installed_plugins";
|
|
break;
|
|
case 1:
|
|
// not installed (get more)
|
|
acceptedStatuses << "not installed" << "new" ;
|
|
tabTitle = "get_more_plugins";
|
|
break;
|
|
case 2:
|
|
// upgradeable
|
|
acceptedStatuses << "upgradeable" ;
|
|
tabTitle = "upgradeable_plugins";
|
|
break;
|
|
case 3:
|
|
// new
|
|
acceptedStatuses << "new" ;
|
|
tabTitle = "new_plugins";
|
|
break;
|
|
case 4:
|
|
// invalid
|
|
acceptedStatuses << "invalid" ;
|
|
tabTitle = "invalid_plugins";
|
|
break;
|
|
}
|
|
mModelProxy->setAcceptedStatuses( acceptedStatuses );
|
|
|
|
updateTabTitle();
|
|
|
|
// load tab description HTML to the detail browser
|
|
QString tabInfoHTML = "";
|
|
QMap<QString, QString>::iterator it = mTabDescriptions.find( tabTitle );
|
|
if ( it != mTabDescriptions.end() )
|
|
{
|
|
QString myStyle = QgsApplication::reportStyleSheet();
|
|
tabInfoHTML += "<style>" + myStyle + "</style>";
|
|
tabInfoHTML = it.value();
|
|
}
|
|
tbDetails->setHtml( tabInfoHTML );
|
|
|
|
// disable buttons
|
|
buttonInstall->setEnabled( false );
|
|
buttonUninstall->setEnabled( false );
|
|
}
|
|
}
|
|
|
|
|
|
|
|
void QgsPluginManager::currentPluginChanged( const QModelIndex & theIndex )
|
|
{
|
|
if ( theIndex.column() == 0 )
|
|
{
|
|
// If the model has been filtered, the index row in the proxy won't match the index row in the underlying model
|
|
// so we need to jump through this little hoop to get the correct item
|
|
QModelIndex realIndex = mModelProxy->mapToSource( theIndex );
|
|
QStandardItem* mypItem = mModelPlugins->itemFromIndex( realIndex );
|
|
showPluginDetails( mypItem );
|
|
}
|
|
}
|
|
|
|
|
|
|
|
void QgsPluginManager::on_vwPlugins_clicked( const QModelIndex &theIndex )
|
|
{
|
|
if ( theIndex.column() == 0 )
|
|
{
|
|
// If the model has been filtered, the index row in the proxy won't match the index row in the underlying model
|
|
// so we need to jump through this little hoop to get the correct item
|
|
QModelIndex realIndex = mModelProxy->mapToSource( theIndex );
|
|
QStandardItem* mypItem = mModelPlugins->itemFromIndex( realIndex );
|
|
if ( !mypItem->isEnabled() )
|
|
{
|
|
//The item is inactive (uncompatible or broken plugin), so it can't be selected. Display it's data anyway.
|
|
vwPlugins->clearSelection();
|
|
showPluginDetails( mypItem );
|
|
}
|
|
}
|
|
}
|
|
|
|
|
|
|
|
void QgsPluginManager::on_vwPlugins_doubleClicked( const QModelIndex & theIndex )
|
|
{
|
|
if ( theIndex.column() == 0 )
|
|
{
|
|
// If the model has been filtered, the index row in the proxy won't match the index row in the underlying model
|
|
// so we need to jump through this little hoop to get the correct item
|
|
QModelIndex realIndex = mModelProxy->mapToSource( theIndex );
|
|
QStandardItem* mypItem = mModelPlugins->itemFromIndex( realIndex );
|
|
if ( mypItem->isCheckable() )
|
|
{
|
|
if ( mypItem->checkState() == Qt::Checked )
|
|
{
|
|
mypItem->setCheckState( Qt::Unchecked );
|
|
}
|
|
else
|
|
{
|
|
mypItem->setCheckState( Qt::Checked );
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
|
|
|
|
void QgsPluginManager::on_leFilter_textChanged( QString theText )
|
|
{
|
|
QgsDebugMsg( "PluginManager filter changed to :" + theText );
|
|
QRegExp::PatternSyntax mySyntax = QRegExp::PatternSyntax( QRegExp::RegExp );
|
|
Qt::CaseSensitivity myCaseSensitivity = Qt::CaseInsensitive;
|
|
QRegExp myRegExp( theText, myCaseSensitivity, mySyntax );
|
|
mModelProxy->setFilterRegExp( myRegExp );
|
|
}
|
|
|
|
|
|
|
|
void QgsPluginManager::on_rbFilterNames_toggled( bool checked )
|
|
{
|
|
if ( checked )
|
|
{
|
|
mModelProxy->setFilterRole( Qt::DisplayRole );
|
|
}
|
|
}
|
|
|
|
|
|
|
|
void QgsPluginManager::on_rbFilterDescriptions_toggled( bool checked )
|
|
{
|
|
if ( checked )
|
|
{
|
|
mModelProxy->setFilterRole( PLUGIN_DESCRIPTION_ROLE );
|
|
}
|
|
}
|
|
|
|
|
|
|
|
void QgsPluginManager::on_rbFilterTags_toggled( bool checked )
|
|
{
|
|
if ( checked )
|
|
{
|
|
mModelProxy->setFilterRole( PLUGIN_TAGS_ROLE );
|
|
}
|
|
}
|
|
|
|
|
|
|
|
void QgsPluginManager::on_rbFilterAuthors_toggled( bool checked )
|
|
{
|
|
if ( checked )
|
|
{
|
|
mModelProxy->setFilterRole( PLUGIN_AUTHOR_ROLE );
|
|
}
|
|
}
|
|
|
|
|
|
|
|
void QgsPluginManager::on_buttonUpgradeAll_clicked( )
|
|
{
|
|
QgsPythonRunner::run( "pyplugin_installer.instance().upgradeAllUpgradeable()" );
|
|
}
|
|
|
|
|
|
|
|
void QgsPluginManager::on_buttonInstall_clicked( )
|
|
{
|
|
QgsPythonRunner::run( QString( "pyplugin_installer.instance().installPlugin('%1')" ).arg( mCurrentlyDisplayedPlugin ) );
|
|
}
|
|
|
|
|
|
|
|
void QgsPluginManager::on_buttonUninstall_clicked( )
|
|
{
|
|
QgsPythonRunner::run( QString( "pyplugin_installer.instance().uninstallPlugin('%1')" ).arg( mCurrentlyDisplayedPlugin ) );
|
|
}
|
|
|
|
|
|
|
|
void QgsPluginManager::on_treeRepositories_itemSelectionChanged( )
|
|
{
|
|
buttonEditRep->setEnabled( ! treeRepositories -> selectedItems().isEmpty() );
|
|
buttonDeleteRep->setEnabled( ! treeRepositories -> selectedItems().isEmpty() );
|
|
}
|
|
|
|
|
|
|
|
void QgsPluginManager::on_treeRepositories_doubleClicked( QModelIndex )
|
|
{
|
|
on_buttonEditRep_clicked();
|
|
}
|
|
|
|
|
|
|
|
void QgsPluginManager::setRepositoryFilter( )
|
|
{
|
|
QTreeWidgetItem * current = treeRepositories->currentItem();
|
|
if ( current )
|
|
{
|
|
QString key = current->text( 1 );
|
|
key = key.replace( "\'", "\\\'" ).replace( "\"", "\\\"" );
|
|
QgsDebugMsg( "Disabling all repositories but selected: " + key );
|
|
QgsPythonRunner::run( QString( "pyplugin_installer.instance().setRepositoryInspectionFilter('%1')" ).arg( key ) );
|
|
}
|
|
}
|
|
|
|
|
|
|
|
void QgsPluginManager::clearRepositoryFilter( )
|
|
{
|
|
QgsDebugMsg( "Enabling all repositories back" );
|
|
QgsPythonRunner::run( QString( "pyplugin_installer.instance().setRepositoryInspectionFilter()" ) );
|
|
}
|
|
|
|
|
|
|
|
void QgsPluginManager::on_buttonRefreshRepos_clicked( )
|
|
{
|
|
QgsDebugMsg( "Refreshing repositories..." );
|
|
QgsPythonRunner::run( "pyplugin_installer.instance().reloadAndExportData()" );
|
|
}
|
|
|
|
|
|
|
|
void QgsPluginManager::on_buttonAddRep_clicked( )
|
|
{
|
|
QgsDebugMsg( "Adding repository connection..." );
|
|
QgsPythonRunner::run( "pyplugin_installer.instance().addRepository()" );
|
|
}
|
|
|
|
|
|
|
|
void QgsPluginManager::on_buttonEditRep_clicked( )
|
|
{
|
|
QTreeWidgetItem * current = treeRepositories->currentItem();
|
|
if ( current )
|
|
{
|
|
QString key = current->text( 1 );
|
|
key = key.replace( "\'", "\\\'" ).replace( "\"", "\\\"" );
|
|
QgsDebugMsg( "Editing repository connection: " + key );
|
|
QgsPythonRunner::run( QString( "pyplugin_installer.instance().editRepository('%1')" ).arg( key ) );
|
|
}
|
|
}
|
|
|
|
|
|
|
|
void QgsPluginManager::on_buttonDeleteRep_clicked( )
|
|
{
|
|
QTreeWidgetItem * current = treeRepositories->currentItem();
|
|
if ( current )
|
|
{
|
|
QString key = current->text( 1 );
|
|
key = key.replace( "\'", "\\\'" ).replace( "\"", "\\\"" );
|
|
QgsDebugMsg( "Deleting repository connection: " + key );
|
|
QgsPythonRunner::run( QString( "pyplugin_installer.instance().deleteRepository('%1')" ).arg( key ) );
|
|
}
|
|
}
|
|
|
|
|
|
|
|
void QgsPluginManager::on_ckbExperimental_toggled( bool state )
|
|
{
|
|
QString settingsGroup;
|
|
QgsPythonRunner::eval( "pyplugin_installer.instance().exportSettingsGroup()", settingsGroup );
|
|
QSettings settings;
|
|
settings.setValue( settingsGroup + "/allowExperimental", QVariant( state ) );
|
|
QgsPythonRunner::run( "pyplugin_installer.installer_data.plugins.rebuild()" );
|
|
QgsPythonRunner::run( "pyplugin_installer.instance().exportPluginsToManager()" );
|
|
}
|
|
|
|
|
|
|
|
// PRIVATE METHODS ///////////////////////////////////////////////////////////////////
|
|
|
|
|
|
bool QgsPluginManager::isPluginEnabled( QString key )
|
|
{
|
|
const QMap<QString, QString>* plugin = pluginMetadata( key );
|
|
if ( plugin->isEmpty() )
|
|
{
|
|
// No such plugin in the metadata registry
|
|
return false;
|
|
}
|
|
|
|
QSettings mySettings;
|
|
if ( plugin->value( "pythonic" ) != "true" )
|
|
{
|
|
// Trim "cpp:" prefix from cpp plugin id
|
|
key = key.mid( 4 );
|
|
return ( mySettings.value( "/Plugins/" + key, QVariant( false ) ).toBool() );
|
|
}
|
|
else
|
|
{
|
|
return ( plugin->value( "installed" ) == "true" && mySettings.value( "/PythonPlugins/" + key, QVariant( false ) ).toBool() );
|
|
}
|
|
}
|
|
|
|
|
|
|
|
bool QgsPluginManager::hasAvailablePlugins( )
|
|
{
|
|
for ( QMap<QString, QMap<QString, QString> >::iterator it = mPlugins.begin();
|
|
it != mPlugins.end();
|
|
it++ )
|
|
{
|
|
if ( it->value( "status" ) == "not installed" || it->value( "status" ) == "new" )
|
|
{
|
|
return true;
|
|
}
|
|
}
|
|
|
|
return false;
|
|
}
|
|
|
|
|
|
|
|
bool QgsPluginManager::hasReinstallablePlugins( )
|
|
{
|
|
for ( QMap<QString, QMap<QString, QString> >::iterator it = mPlugins.begin();
|
|
it != mPlugins.end();
|
|
it++ )
|
|
{
|
|
// plugins marked as "installed" are available for download (otherwise they are marked "orphans")
|
|
if ( it->value( "status" ) == "installed" )
|
|
{
|
|
return true;
|
|
}
|
|
}
|
|
|
|
return false;
|
|
}
|
|
|
|
|
|
|
|
bool QgsPluginManager::hasUpgradeablePlugins( )
|
|
{
|
|
for ( QMap<QString, QMap<QString, QString> >::iterator it = mPlugins.begin();
|
|
it != mPlugins.end();
|
|
it++ )
|
|
{
|
|
if ( it->value( "status" ) == "upgradeable" )
|
|
{
|
|
return true;
|
|
}
|
|
}
|
|
|
|
return false;
|
|
}
|
|
|
|
|
|
|
|
bool QgsPluginManager::hasNewPlugins( )
|
|
{
|
|
for ( QMap<QString, QMap<QString, QString> >::iterator it = mPlugins.begin();
|
|
it != mPlugins.end();
|
|
it++ )
|
|
{
|
|
if ( it->value( "status" ) == "new" )
|
|
{
|
|
return true;
|
|
}
|
|
}
|
|
|
|
return false;
|
|
}
|
|
|
|
|
|
|
|
bool QgsPluginManager::hasNewerPlugins( )
|
|
{
|
|
for ( QMap<QString, QMap<QString, QString> >::iterator it = mPlugins.begin();
|
|
it != mPlugins.end();
|
|
it++ )
|
|
{
|
|
if ( it->value( "status" ) == "newer" )
|
|
{
|
|
return true;
|
|
}
|
|
}
|
|
|
|
return false;
|
|
}
|
|
|
|
|
|
|
|
bool QgsPluginManager::hasInvalidPlugins( )
|
|
{
|
|
for ( QMap<QString, QMap<QString, QString> >::iterator it = mPlugins.begin();
|
|
it != mPlugins.end();
|
|
it++ )
|
|
{
|
|
if ( ! it->value( "error" ).isEmpty() )
|
|
{
|
|
return true;
|
|
}
|
|
}
|
|
|
|
return false;
|
|
}
|
|
|
|
|
|
|
|
void QgsPluginManager::updateTabTitle()
|
|
{
|
|
lbStatusFilter->setText( QString( " %1 > %2 (%3)" ).arg( tr( "Plugins" ) )
|
|
.arg( mOptionsListWidget->currentItem()->text() )
|
|
.arg( mModelProxy->countWithCurrentStatus() ) );
|
|
}
|
|
|