// -*- Mode: C++; c-basic-offset: 2; indent-tabs-mode: nil; tab-width: 2; -*- /** * pluginmanager.cpp * Most of this code has been lifted from Martijn's KopetePluginManager class * * Copyright (C) 2004 Zack Rusin * * This library is free software; you can redistribute it and/or * modify it under the terms of the GNU Lesser General Public * License as published by the Free Software Foundation; either * version 2.1 of the License, or (at your option) any later version. * * This library is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU * Lesser General Public License for more details. * * You should have received a copy of the GNU Lesser General Public * License along with this library; if not, write to the Free Software * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA * 02110-1301 USA */ #include "pluginmanager.h" #include "plugin.h" #include #include #include #include #include #include #include #include #include #include #include #include #include #include namespace Komposer { class PluginManager::Private { public: // All available plugins, regardless of category, and loaded or not TQValueList plugins; // Dict of all currently loaded plugins, mapping the KPluginInfo to // a plugin TQMap loadedPlugins; // The plugin manager's mode. The mode is StartingUp until loadAllPlugins() // has finished loading the plugins, after which it is set to Running. // ShuttingDown and DoneShutdown are used during Komposer shutdown by the // async unloading of plugins. enum ShutdownMode { StartingUp, Running, ShuttingDown, DoneShutdown }; ShutdownMode shutdownMode; KSharedConfig::Ptr config; // Plugins pending for loading TQValueStack pluginsToLoad; }; PluginManager::PluginManager( TQObject *tqparent ) : TQObject( tqparent ) { d = new Private; // We want to add a reference to the application's event loop so we // can remain in control when all windows are removed. // This way we can unload plugins asynchronously, which is more // robust if they are still doing processing. kapp->ref(); d->shutdownMode = Private::StartingUp; KSettings::Dispatcher::self()->registerInstance( KGlobal::instance(), this, TQT_SLOT( loadAllPlugins() ) ); d->plugins = KPluginInfo::fromServices( KTrader::self()->query( TQString::tqfromLatin1( "Komposer/Plugin" ), TQString::tqfromLatin1( "[X-Komposer-Version] == 1" ) ) ); } PluginManager::~PluginManager() { if ( d->shutdownMode != Private::DoneShutdown ) { slotShutdownTimeout(); #if 0 kdWarning() << k_funcinfo << "Destructing plugin manager without going through " << "the shutdown process!" << endl << kdBacktrace(10) << endl; #endif } // Quick cleanup of the remaining plugins, hope it helps TQMap::ConstIterator it; for ( it = d->loadedPlugins.begin(); it != d->loadedPlugins.end(); /* EMPTY */ ) { // Remove causes the iterator to become invalid, so pre-increment first TQMap::ConstIterator nextIt( it ); ++nextIt; kdWarning() << k_funcinfo << "Deleting stale plugin '" << it.data()->name() << "'" << endl; delete it.data(); it = nextIt; } delete d; } TQValueList PluginManager::availablePlugins( const TQString &category ) const { if ( category.isEmpty() ) return d->plugins; TQValueList result; TQValueList::ConstIterator it; for ( it = d->plugins.begin(); it != d->plugins.end(); ++it ) { if ( ( *it )->category() == category ) result.append( *it ); } return result; } TQMap PluginManager::loadedPlugins( const TQString &category ) const { TQMap result; TQMap::ConstIterator it; for ( it = d->loadedPlugins.begin(); it != d->loadedPlugins.end(); ++it ) { if ( category.isEmpty() || it.key()->category() == category ) result.insert( it.key(), it.data() ); } return result; } void PluginManager::shutdown() { d->shutdownMode = Private::ShuttingDown; // Remove any pending plugins to load, we're shutting down now :) d->pluginsToLoad.clear(); // Ask all plugins to unload if ( d->loadedPlugins.empty() ) { d->shutdownMode = Private::DoneShutdown; } else { TQMap::ConstIterator it; for ( it = d->loadedPlugins.begin(); it != d->loadedPlugins.end(); /* EMPTY */ ) { // Remove causes the iterator to become invalid, so pre-increment first TQMap::ConstIterator nextIt( it ); ++nextIt; it.data()->aboutToUnload(); it = nextIt; } } TQTimer::singleShot( 3000, this, TQT_SLOT(slotShutdownTimeout()) ); } void PluginManager::slotPluginReadyForUnload() { // Using TQObject::sender() is on purpose here, because otherwise all // plugins would have to pass 'this' as parameter, which makes the API // less clean for plugin authors Plugin* plugin = dynamic_cast( const_cast( sender() ) ); if ( !plugin ) { kdWarning() << k_funcinfo << "Calling object is not a plugin!" << endl; return; } kdDebug()<<"manager unloading"<deleteLater(); } void PluginManager::slotShutdownTimeout() { // When we were already done the timer might still fire. // Do nothing in that case. if ( d->shutdownMode == Private::DoneShutdown ) return; #ifndef NDEBUG TQStringList remaining; for ( TQMap::ConstIterator it = d->loadedPlugins.begin(); it != d->loadedPlugins.end(); ++it ) remaining.append( it.key()->pluginName() ); kdWarning() << k_funcinfo << "Some plugins didn't shutdown in time!" << endl << "Remaining plugins: " << remaining.join( TQString::tqfromLatin1( ", " ) ) << endl << "Forcing Komposer shutdown now." << endl; #endif slotShutdownDone(); } void PluginManager::slotShutdownDone() { d->shutdownMode = Private::DoneShutdown; kapp->deref(); } void PluginManager::loadAllPlugins() { // FIXME: We need session management here - Martijn if ( !d->config ) d->config = KSharedConfig::openConfig( "komposerrc" ); TQMap entries = d->config->entryMap( TQString::tqfromLatin1( "Plugins" ) ); TQMap::Iterator it; for ( it = entries.begin(); it != entries.end(); ++it ) { TQString key = it.key(); if ( key.endsWith( TQString::tqfromLatin1( "Enabled" ) ) ) { key.setLength( key.length() - 7 ); //kdDebug() << k_funcinfo << "Set " << key << " to " << it.data() << endl; if ( it.data() == TQString::tqfromLatin1( "true" ) ) { if ( !plugin( key ) ) d->pluginsToLoad.push( key ); } else { // FIXME: Does this ever happen? As loadAllPlugins is only called on startup // I'd say 'no'. If it does, it should be made async // though. - Martijn if ( plugin( key ) ) unloadPlugin( key ); } } } // Schedule the plugins to load TQTimer::singleShot( 0, this, TQT_SLOT( slotLoadNextPlugin() ) ); } void PluginManager::slotLoadNextPlugin() { if ( d->pluginsToLoad.isEmpty() ) { if ( d->shutdownMode == Private::StartingUp ) { d->shutdownMode = Private::Running; emit allPluginsLoaded(); } return; } TQString key = d->pluginsToLoad.pop(); loadPluginInternal( key ); // Schedule the next run unconditionally to avoid code duplication on the // allPluginsLoaded() signal's handling. This has the added benefit that // the signal is delayed one event loop, so the accounts are more likely // to be instantiated. TQTimer::singleShot( 0, this, TQT_SLOT( slotLoadNextPlugin() ) ); } Plugin* PluginManager::loadPlugin( const TQString &pluginId, PluginLoadMode mode /* = LoadSync */ ) { if ( mode == LoadSync ) { return loadPluginInternal( pluginId ); } else { d->pluginsToLoad.push( pluginId ); TQTimer::singleShot( 0, this, TQT_SLOT( slotLoadNextPlugin() ) ); return 0; } } Plugin* PluginManager::loadPluginInternal( const TQString &pluginId ) { KPluginInfo* info = infoForPluginId( pluginId ); if ( !info ) { kdWarning() << k_funcinfo << "Unable to find a plugin named '" << pluginId << "'!" << endl; return 0; } if ( d->loadedPlugins.tqcontains( info ) ) return d->loadedPlugins[ info ]; int error = 0; Plugin *plugin = KParts::ComponentFactory::createInstanceFromQuery( TQString::tqfromLatin1( "Komposer/Plugin" ), TQString::tqfromLatin1( "[X-KDE-PluginInfo-Name]=='%1'" ).arg( pluginId ), this, 0, TQStringList(), &error ); if ( plugin ) { d->loadedPlugins.insert( info, plugin ); info->setPluginEnabled( true ); connect( plugin, TQT_SIGNAL(destroyed(TQObject*)), this, TQT_SLOT(slotPluginDestroyed(TQObject*)) ); connect( plugin, TQT_SIGNAL(readyForUnload()), this, TQT_SLOT(slotPluginReadyForUnload()) ); kdDebug() << k_funcinfo << "Successfully loaded plugin '" << pluginId << "'" << endl; emit pluginLoaded( plugin ); } else { switch ( error ) { case KParts::ComponentFactory::ErrNoServiceFound: kdDebug() << k_funcinfo << "No service implementing the given mimetype " << "and fullfilling the given constraint expression can be found." << endl; break; case KParts::ComponentFactory::ErrServiceProvidesNoLibrary: kdDebug() << "the specified service provides no shared library." << endl; break; case KParts::ComponentFactory::ErrNoLibrary: kdDebug() << "the specified library could not be loaded." << endl; break; case KParts::ComponentFactory::ErrNoFactory: kdDebug() << "the library does not export a factory for creating components." << endl; break; case KParts::ComponentFactory::ErrNoComponent: kdDebug() << "the factory does not support creating components " << "of the specified type." << endl; break; } kdDebug() << k_funcinfo << "Loading plugin '" << pluginId << "' failed, KLibLoader reported error: '" << KLibLoader::self()->lastErrorMessage() << "'" << endl; } return plugin; } bool PluginManager::unloadPlugin( const TQString &spec ) { TQMap::ConstIterator it; for ( it = d->loadedPlugins.begin(); it != d->loadedPlugins.end(); ++it ) { if ( it.key()->pluginName() == spec ) { it.data()->aboutToUnload(); return true; } } return false; } void PluginManager::slotPluginDestroyed( TQObject *plugin ) { TQMap::Iterator it; for ( it = d->loadedPlugins.begin(); it != d->loadedPlugins.end(); ++it ) { if ( it.data() == plugin ) { d->loadedPlugins.erase( it ); break; } } if ( d->shutdownMode == Private::ShuttingDown && d->loadedPlugins.isEmpty() ) { // Use a timer to make sure any pending deleteLater() calls have // been handled first TQTimer::singleShot( 0, this, TQT_SLOT(slotShutdownDone()) ); } } Plugin* PluginManager::plugin( const TQString &pluginId ) const { KPluginInfo *info = infoForPluginId( pluginId ); if ( !info ) return 0; if ( d->loadedPlugins.tqcontains( info ) ) return d->loadedPlugins[ info ]; else return 0; } TQString PluginManager::pluginName( const Plugin *plugin ) const { TQMap::ConstIterator it; for ( it = d->loadedPlugins.begin(); it != d->loadedPlugins.end(); ++it ) { if ( it.data() == plugin ) return it.key()->name(); } return TQString::tqfromLatin1( "Unknown" ); } TQString PluginManager::pluginId( const Plugin *plugin ) const { TQMap::ConstIterator it; for ( it = d->loadedPlugins.begin(); it != d->loadedPlugins.end(); ++it ) { if ( it.data() == plugin ) return it.key()->pluginName(); } return TQString::tqfromLatin1( "unknown" ); } TQString PluginManager::pluginIcon( const Plugin *plugin ) const { TQMap::ConstIterator it; for ( it = d->loadedPlugins.begin(); it != d->loadedPlugins.end(); ++it ) { if ( it.data() == plugin ) return it.key()->icon(); } return TQString::tqfromLatin1( "Unknown" ); } KPluginInfo* PluginManager::infoForPluginId( const TQString &pluginId ) const { TQValueList::ConstIterator it; for ( it = d->plugins.begin(); it != d->plugins.end(); ++it ) { if ( ( *it )->pluginName() == pluginId ) return *it; } return 0; } bool PluginManager::setPluginEnabled( const TQString &pluginId, bool enabled /* = true */ ) { if ( !d->config ) d->config = KSharedConfig::openConfig( "komposerrc" ); d->config->setGroup( "Plugins" ); if ( !infoForPluginId( pluginId ) ) return false; d->config->writeEntry( pluginId + TQString::tqfromLatin1( "Enabled" ), enabled ); d->config->sync(); return true; } } #include "pluginmanager.moc"