cutter/src/plugins/PluginManager.cpp

160 lines
4.4 KiB
C++
Raw Normal View History

2019-02-09 13:05:06 +00:00
#ifdef CUTTER_ENABLE_PYTHON
#include <Python.h>
#include <cutterbindings_python.h>
2019-02-09 13:05:06 +00:00
#endif
#include "PluginManager.h"
#include "PythonManager.h"
#include "CutterPlugin.h"
#include <QDir>
#include <QCoreApplication>
#include <QPluginLoader>
#include <QStandardPaths>
2019-02-09 13:05:06 +00:00
Q_GLOBAL_STATIC(PluginManager, uniqueInstance)
PluginManager *PluginManager::getInstance()
{
return uniqueInstance;
}
PluginManager::PluginManager()
{
}
PluginManager::~PluginManager()
{
}
void PluginManager::loadPlugins()
{
assert(plugins.isEmpty());
QStringList locations = QStandardPaths::standardLocations(QStandardPaths::AppDataLocation);
if (locations.isEmpty()) {
qCritical() << "Failed to get a standard path to load plugins from.";
return;
2019-02-09 13:05:06 +00:00
}
QDir pluginsDir(locations.first());
2019-02-12 09:49:24 +00:00
pluginsDir.mkpath(".");
pluginsDir.mkdir("plugins");
2019-02-09 13:05:06 +00:00
if (!pluginsDir.cd("plugins")) {
return;
}
QDir nativePluginsDir = pluginsDir;
nativePluginsDir.mkdir("native");
2019-02-09 13:05:06 +00:00
if (nativePluginsDir.cd("native")) {
loadNativePlugins(nativePluginsDir);
}
QDir pythonPluginsDir = pluginsDir;
pythonPluginsDir.mkdir("python");
2019-02-09 13:05:06 +00:00
if (pythonPluginsDir.cd("python")) {
loadPythonPlugins(pythonPluginsDir.absolutePath());
}
qInfo() << "Loaded" << plugins.length() << "plugin(s).";
}
void PluginManager::destroyPlugins()
{
for (CutterPlugin *plugin : plugins) {
delete plugin;
}
}
void PluginManager::loadNativePlugins(const QDir &directory)
{
for (const QString &fileName : directory.entryList(QDir::Files)) {
QPluginLoader pluginLoader(directory.absoluteFilePath(fileName));
QObject *plugin = pluginLoader.instance();
if (!plugin) {
continue;
}
CutterPlugin *cutterPlugin = qobject_cast<CutterPlugin *>(plugin);
if (!cutterPlugin) {
continue;
}
cutterPlugin->setupPlugin();
2019-02-09 13:05:06 +00:00
plugins.append(cutterPlugin);
}
}
#ifdef CUTTER_ENABLE_PYTHON
void PluginManager::loadPythonPlugins(const QDir &directory)
{
Python()->addPythonPath(directory.absolutePath().toLocal8Bit().data());
for (const QString &fileName : directory.entryList(QDir::Dirs | QDir::Files | QDir::NoDotAndDotDot)) {
if (fileName == "__pycache__") {
continue;
}
2019-02-09 13:05:06 +00:00
QString moduleName;
if (fileName.endsWith(".py")) {
QStringList l = fileName.split(".py");
moduleName = l[0];
} else {
moduleName = fileName;
2019-02-09 13:05:06 +00:00
}
CutterPlugin *cutterPlugin = loadPythonPlugin(moduleName.toLocal8Bit().constData());
2019-02-09 13:05:06 +00:00
if (!cutterPlugin) {
continue;
}
cutterPlugin->setupPlugin();
2019-02-09 13:05:06 +00:00
plugins.append(cutterPlugin);
}
PythonManager::ThreadHolder threadHolder;
}
CutterPlugin *PluginManager::loadPythonPlugin(const char *moduleName)
{
PythonManager::ThreadHolder threadHolder;
PyObject *pluginModule = PyImport_ImportModule(moduleName);
if (!pluginModule) {
qWarning() << "Couldn't load module for plugin:" << QString(moduleName);
PyErr_Print();
return nullptr;
2019-02-09 13:05:06 +00:00
}
PyObject *createPluginFunc = PyObject_GetAttrString(pluginModule, "create_cutter_plugin");
if (!createPluginFunc || !PyCallable_Check(createPluginFunc)) {
qWarning() << "Plugin module does not contain create_cutter_plugin() function:" << QString(moduleName);
if (createPluginFunc) {
Py_DECREF(createPluginFunc);
}
Py_DECREF(pluginModule);
return nullptr;
}
PyObject *pluginObject = PyObject_CallFunction(createPluginFunc, nullptr);
Py_DECREF(createPluginFunc);
Py_DECREF(pluginModule);
if (!pluginObject) {
qWarning() << "Plugin's create_cutter_plugin() function failed.";
PyErr_Print();
return nullptr;
}
PythonToCppFunc pythonToCpp = Shiboken::Conversions::isPythonToCppPointerConvertible(reinterpret_cast<SbkObjectType *>(SbkCutterBindingsTypes[SBK_CUTTERPLUGIN_IDX]), pluginObject);
if (!pythonToCpp) {
qWarning() << "Plugin's create_cutter_plugin() function did not return an instance of CutterPlugin:" << QString(moduleName);
return nullptr;
}
CutterPlugin *plugin;
pythonToCpp(pluginObject, &plugin);
if (!plugin) {
qWarning() << "Error during the setup of CutterPlugin:" << QString(moduleName);
return nullptr;
}
2019-02-09 13:05:06 +00:00
return plugin;
}
#endif