From 616e680aa8af8f5056b5133dd44258c252ca656f Mon Sep 17 00:00:00 2001 From: aqua Date: Tue, 16 Aug 2022 16:19:04 +0300 Subject: Turn WebEngine into a plugin --- CMakeLists.txt | 1 + plugins/webengine/CMakeLists.txt | 12 +++++++++++ plugins/webengine/WebEngine.json | 4 ++++ plugins/webengine/webengineplugin.cpp | 13 ++++++++++++ plugins/webengine/webengineplugin.hpp | 23 ++++++++++++++++++++ plugins/webengine/webview.cpp | 25 ++++++++++++++++++++++ plugins/webengine/webview.h | 25 ++++++++++++++++++++++ src/CMakeLists.txt | 9 +++++--- src/application.cpp | 37 +++++++++++++++++++++++++++++--- src/application.h | 4 ++++ src/main.cpp | 11 +++++++++- src/plugins/CMakeLists.txt | 7 ++++++ src/plugins/pluginloader.cpp | 17 +++++++++++++++ src/plugins/pluginloader.h | 25 ++++++++++++++++++++++ src/plugins/rplugininterface.hpp | 23 ++++++++++++++++++++ src/plugins/rview.hpp | 25 ++++++++++++++++++++++ src/plugins/test/pluginloader.cpp | 40 +++++++++++++++++++++++++++++++++++ src/rview.cpp | 12 ----------- src/rview.h | 25 ---------------------- src/webengine/webview.cpp | 25 ---------------------- src/webengine/webview.h | 25 ---------------------- 21 files changed, 294 insertions(+), 94 deletions(-) create mode 100644 plugins/webengine/CMakeLists.txt create mode 100644 plugins/webengine/WebEngine.json create mode 100644 plugins/webengine/webengineplugin.cpp create mode 100644 plugins/webengine/webengineplugin.hpp create mode 100644 plugins/webengine/webview.cpp create mode 100644 plugins/webengine/webview.h create mode 100644 src/plugins/CMakeLists.txt create mode 100644 src/plugins/pluginloader.cpp create mode 100644 src/plugins/pluginloader.h create mode 100644 src/plugins/rplugininterface.hpp create mode 100644 src/plugins/rview.hpp create mode 100644 src/plugins/test/pluginloader.cpp delete mode 100644 src/rview.cpp delete mode 100644 src/rview.h delete mode 100644 src/webengine/webview.cpp delete mode 100644 src/webengine/webview.h diff --git a/CMakeLists.txt b/CMakeLists.txt index 025c40e0..ee184c79 100644 --- a/CMakeLists.txt +++ b/CMakeLists.txt @@ -102,6 +102,7 @@ feature_summary(WHAT ALL) ADD_SUBDIRECTORY( src ) ADD_SUBDIRECTORY( icons ) +add_subdirectory(plugins/webengine) # ================================================================================ diff --git a/plugins/webengine/CMakeLists.txt b/plugins/webengine/CMakeLists.txt new file mode 100644 index 00000000..8a11bc76 --- /dev/null +++ b/plugins/webengine/CMakeLists.txt @@ -0,0 +1,12 @@ +add_library(WebEnginePlugin MODULE + webengineplugin.cpp webengineplugin.hpp + webview.cpp webview.h + ${PROJECT_SOURCE_DIR}/src/plugins/rplugininterface.hpp + ${PROJECT_SOURCE_DIR}/src/plugins/rview.hpp +) +target_include_directories(WebEnginePlugin PRIVATE ${PROJECT_SOURCE_DIR}/src/plugins) +target_link_libraries(WebEnginePlugin PUBLIC Qt6::Core Qt6::WebEngineWidgets) + +if(TESTING) + add_test(NAME "load_plugin WebEnginePlugin" COMMAND load_plugin $) +endif() diff --git a/plugins/webengine/WebEngine.json b/plugins/webengine/WebEngine.json new file mode 100644 index 00000000..03e2112c --- /dev/null +++ b/plugins/webengine/WebEngine.json @@ -0,0 +1,4 @@ +{ + "name": "WebEngine", + "schemes": [ "file", "http", "https" ] +} diff --git a/plugins/webengine/webengineplugin.cpp b/plugins/webengine/webengineplugin.cpp new file mode 100644 index 00000000..cf6811dc --- /dev/null +++ b/plugins/webengine/webengineplugin.cpp @@ -0,0 +1,13 @@ +/* ============================================================ + * The rekonq project + * ============================================================ + * SPDX-License-Identifier: GPL-3.0-only + * Copyright (C) 2022 aqua + * ============================================================ + * Description: WebEngine plugin View factory + * ============================================================ */ + +#include "webengineplugin.hpp" +#include "webview.h" + +rView *WebEnginePlugin::view(const QUrl &url) { return new WebView(url, nullptr); } diff --git a/plugins/webengine/webengineplugin.hpp b/plugins/webengine/webengineplugin.hpp new file mode 100644 index 00000000..3ee63ff8 --- /dev/null +++ b/plugins/webengine/webengineplugin.hpp @@ -0,0 +1,23 @@ +/* ============================================================ + * The rekonq project + * ============================================================ + * SPDX-License-Identifier: GPL-3.0-only + * Copyright (C) 2022 aqua + * ============================================================ + * Description: WebEngine plugin View factory + * ============================================================ */ + +#pragma once + +#include + +class WebEnginePlugin final : public RekonqPluginInterface { + Q_OBJECT + Q_PLUGIN_METADATA(IID RekonqPluginInterface_iid FILE "WebEngine.json") + Q_INTERFACES(RekonqPluginInterface) + +public: + WebEnginePlugin() = default; + + rView *view(const QUrl &url) override; +}; diff --git a/plugins/webengine/webview.cpp b/plugins/webengine/webview.cpp new file mode 100644 index 00000000..1aef01e2 --- /dev/null +++ b/plugins/webengine/webview.cpp @@ -0,0 +1,25 @@ +/* ============================================================ + * The rekonq project + * ============================================================ + * SPDX-License-Identifier: GPL-3.0-only + * Copyright (C) 2022 aqua + * ============================================================ + * Description: Qt WebEngine View + * ============================================================ */ + +#include "webview.h" +#include +#include + +WebView::WebView(const QUrl &url, QWidget *parent) : rView(url, parent), view(new QWebEngineView(this)) +{ + auto *layout = new QVBoxLayout; + layout->setContentsMargins(0, 0, 0, 0); + layout->addWidget(view); + setLayout(layout); + + connect(view, &QWebEngineView::iconChanged, this, [this](const QIcon &icon) { emit iconChanged(icon); }); + connect(view, &QWebEngineView::urlChanged, this, [this](const QUrl &url) { emit urlChanged(url); }); + connect(view, &QWebEngineView::titleChanged, this, [this](const QString &title) { emit titleChanged(title); }); + view->load(url); +} diff --git a/plugins/webengine/webview.h b/plugins/webengine/webview.h new file mode 100644 index 00000000..0b8d0dae --- /dev/null +++ b/plugins/webengine/webview.h @@ -0,0 +1,25 @@ +/* ============================================================ + * The rekonq project + * ============================================================ + * SPDX-License-Identifier: GPL-3.0-only + * Copyright (C) 2022 aqua + * ============================================================ + * Description: Qt WebEngine View + * ============================================================ */ + +#pragma once + +#include "rview.hpp" + +class QWebEngineView; + +class WebView final : public rView { + Q_OBJECT + +public: + explicit WebView(const QUrl &url = QUrl(), QWidget *parent = nullptr); + ~WebView() final = default; + +private: + QWebEngineView *view; +}; diff --git a/src/CMakeLists.txt b/src/CMakeLists.txt index 4aa141c9..5f154e3f 100644 --- a/src/CMakeLists.txt +++ b/src/CMakeLists.txt @@ -1,6 +1,7 @@ ### ------- sub dirs ------- ADD_SUBDIRECTORY( data ) +add_subdirectory(plugins) ### ------- SETTING REKONQ FILES.. @@ -216,15 +217,17 @@ ADD_DEFINITIONS ( ${KDE4_DEFINITIONS} ) add_executable(rekonq main.cpp application.cpp application.h # ---------------------------------------- - rview.cpp rview.h webengine/webview.cpp webengine/webview.h + plugins/rplugininterface.hpp plugins/rview.hpp ) ### --------------- TARGETTING LINK LIBRARIES... target_link_libraries(rekonq - Qt6::Widgets Qt6::WebEngineWidgets - SingleApplication::SingleApplication) + Qt6::Widgets + SingleApplication::SingleApplication + pluginloader +) # Nepomuk optional target link libraries IF(SOPRANO_FOUND AND NepomukCore_FOUND) diff --git a/src/application.cpp b/src/application.cpp index 54a7db88..9e34e5e2 100644 --- a/src/application.cpp +++ b/src/application.cpp @@ -10,8 +10,9 @@ * ============================================================ */ #include "application.h" -#include "rview.h" -#include "webengine/webview.h" +#include "plugins/pluginloader.h" +#include "plugins/rplugininterface.hpp" +#include // --------------------------------------------------------------------------------------------------------------- // Ctor and Dtor @@ -42,8 +43,28 @@ Application::~Application() // Destroy all web apps for (auto *webApp : m_webApps) delete webApp; + + // Unload all plugins + for (auto *plugin : m_plugins) { + plugin->unload(); + delete plugin; + } } +// --------------------------------------------------------------------------------------------------------------- +// Plugin management + +bool Application::registerPlugin(const QString &path) +{ + auto *loader = new PluginLoader(path); + if (!loader->load()) { + delete loader; + return false; + } + + m_plugins.append(loader); + return true; +} // --------------------------------------------------------------------------------------------------------------- /* int Application::newInstance() @@ -388,7 +409,17 @@ void Application::setWindowInfo(RekonqWindow *w) rView *Application::newWebApp(const QUrl &url) { - auto *view = new WebView(url); + RekonqPluginInterface *interface = nullptr; + for (auto *plugin : m_plugins) + if (plugin->hasScheme(url.scheme())) { + interface = plugin->interface(); + break; + } + + if (interface == nullptr) return nullptr; + + auto *view = interface->view(url); + Q_CHECK_PTR(view); // tab->installEventFilter(this); m_webApps.append(view); diff --git a/src/application.h b/src/application.h index 761bc89b..407a8681 100644 --- a/src/application.h +++ b/src/application.h @@ -18,6 +18,7 @@ // Forward Declarations class rView; +class PluginLoader; // class RekonqWindow; // class WebTab; @@ -38,6 +39,8 @@ public: Application(int &argc, char *argv[]); ~Application() override; + bool registerPlugin(const QString &path); + // int newInstance(); static Application *instance() { return (qobject_cast(QCoreApplication::instance())); } @@ -107,6 +110,7 @@ private slots: // void pageCreated(WebPage *); private: + QList m_plugins; // RekonqWindowList m_rekonqWindows; QList m_webApps; }; diff --git a/src/main.cpp b/src/main.cpp index 20c30c68..6db6f36c 100644 --- a/src/main.cpp +++ b/src/main.cpp @@ -19,6 +19,9 @@ static const char *description = "A lightweight Web Browser based on Qt WebEngin int main(int argc, char **argv) { + // When loading QtWebEngine from a plugin, set Qt::AA_ShareOpenGLContexts using QCoreApplication::setAttribute + QCoreApplication::setAttribute(Qt::AA_ShareOpenGLContexts, true); + Application app(argc, argv); // Set application data QCoreApplication::setApplicationName(QL1S("rekonq")); @@ -34,13 +37,19 @@ int main(int argc, char **argv) QCommandLineOption options_incognito("incognito", QCoreApplication::translate("main", "Open in incognito mode")); QCommandLineOption options_webapp("webapp", QCoreApplication::translate("main", "Open URL as web app (in a simple window)")); - parser.addOptions({options_incognito, options_webapp}); + QCommandLineOption options_plugin({"l", "load"}, QCoreApplication::translate("main", "Add plugin to load path"), + "path"); + parser.addOptions({options_incognito, options_webapp, options_plugin}); // Define the positional arguments parser.addPositionalArgument("URL", QCoreApplication::translate("main", "Location to open")); parser.process(app); + if (parser.isSet(options_plugin)) { + for (const auto &plugin : parser.values(options_plugin)) app.registerPlugin(plugin); + } + const auto positionalArguments = parser.positionalArguments(); if (parser.isSet(options_webapp)) positionalArguments.isEmpty() ? app.newWebApp() : app.newWebApp(QUrl::fromUserInput(positionalArguments.first())); diff --git a/src/plugins/CMakeLists.txt b/src/plugins/CMakeLists.txt new file mode 100644 index 00000000..01b13ae1 --- /dev/null +++ b/src/plugins/CMakeLists.txt @@ -0,0 +1,7 @@ +add_library(pluginloader STATIC pluginloader.cpp pluginloader.h) +target_link_libraries(pluginloader PUBLIC Qt6::Core Qt6::Widgets) + +if(TESTING) + add_executable(load_plugin test/pluginloader.cpp) + target_link_libraries(load_plugin GTest::gtest GTest::gtest_main Qt6::Core) +endif() diff --git a/src/plugins/pluginloader.cpp b/src/plugins/pluginloader.cpp new file mode 100644 index 00000000..ad5dbc14 --- /dev/null +++ b/src/plugins/pluginloader.cpp @@ -0,0 +1,17 @@ +/* ============================================================ + * The rekonq project + * ============================================================ + * SPDX-License-Identifier: GPL-3.0-only + * Copyright (C) 2022 aqua + * ============================================================ + * Description: rekonq plugin loader + * ============================================================ */ + +#include "pluginloader.h" +#include + +PluginLoader::PluginLoader(const QString &fileName, QObject *parent) : QPluginLoader(fileName, parent) +{ + const auto metadata = metaData()["MetaData"].toObject(); + for (const auto &value : metadata["schemes"].toArray()) { m_schemes.append(value.toString()); } +} diff --git a/src/plugins/pluginloader.h b/src/plugins/pluginloader.h new file mode 100644 index 00000000..f764da9f --- /dev/null +++ b/src/plugins/pluginloader.h @@ -0,0 +1,25 @@ +/* ============================================================ + * The rekonq project + * ============================================================ + * SPDX-License-Identifier: GPL-3.0-only + * Copyright (C) 2022 aqua + * ============================================================ + * Description: rekonq plugin loader + * ============================================================ */ + +#pragma once + +#include "rplugininterface.hpp" +#include + +class PluginLoader : public QPluginLoader { + +public: + explicit PluginLoader(const QString &path, QObject *parent = nullptr); + [[nodiscard]] bool hasScheme(const QString &scheme) { return m_schemes.contains(scheme); } + + [[nodiscard]] RekonqPluginInterface *interface() { return qobject_cast(instance()); } + +private: + QStringList m_schemes; +}; diff --git a/src/plugins/rplugininterface.hpp b/src/plugins/rplugininterface.hpp new file mode 100644 index 00000000..16186591 --- /dev/null +++ b/src/plugins/rplugininterface.hpp @@ -0,0 +1,23 @@ +/* ============================================================ + * The rekonq project + * ============================================================ + * SPDX-License-Identifier: GPL-3.0-only + * Copyright (C) 2022 aqua + * ============================================================ + * Description: rekonq plugin interface + * ============================================================ */ + +#pragma once + +#include "rview.hpp" +#include + +class RekonqPluginInterface : public QObject { + Q_OBJECT + +public: + virtual rView *view(const QUrl &url) = 0; +}; + +#define RekonqPluginInterface_iid "rekonq.3.RekongPluginInterface" +Q_DECLARE_INTERFACE(RekonqPluginInterface, RekonqPluginInterface_iid) diff --git a/src/plugins/rview.hpp b/src/plugins/rview.hpp new file mode 100644 index 00000000..71bf5ddf --- /dev/null +++ b/src/plugins/rview.hpp @@ -0,0 +1,25 @@ +/* ============================================================ + * The rekonq project + * ============================================================ + * SPDX-License-Identifier: GPL-3.0-only + * Copyright (C) 2022 aqua + * ============================================================ + * Description: View Interface + * ============================================================ */ + +#pragma once + +#include +#include + +class rView : public QWidget { + Q_OBJECT + +public: + explicit rView(const QUrl &url = QUrl(), QWidget *parent = nullptr) : QWidget(parent) {} + +signals: + void iconChanged(const QIcon &); + void urlChanged(const QUrl &); + void titleChanged(const QString &); +}; diff --git a/src/plugins/test/pluginloader.cpp b/src/plugins/test/pluginloader.cpp new file mode 100644 index 00000000..83b30c40 --- /dev/null +++ b/src/plugins/test/pluginloader.cpp @@ -0,0 +1,40 @@ +#include +#include +#include +#include + +char *pluginPath; + +class PluginLoaderTest : public ::testing::Test { + +protected: + void SetUp() override { loader = new QPluginLoader; } + void TearDown() override { delete loader; } + + QPluginLoader *loader = nullptr; +}; + +TEST_F(PluginLoaderTest, InterfaceTest) +{ + // Ensure plugin loads + loader->setFileName(pluginPath); + EXPECT_TRUE(loader->load()) << "For plugin: " << pluginPath << '\n' << qUtf8Printable(loader->errorString()); + + // Ensure plugin has metadata + const auto metadata = loader->metaData()["MetaData"].toObject(); + EXPECT_TRUE(metadata.contains("name")); + EXPECT_FALSE(metadata["name"].toString().isEmpty()); + EXPECT_TRUE(metadata.contains("schemes")); + EXPECT_FALSE(metadata["schemes"].toArray().isEmpty()); +} + +int main(int argc, char **argv) +{ + ::testing::InitGoogleTest(&argc, argv); + if (argc != 2) { + std::cerr << "Usage: " << argv[0] << " [path to plugin]" << std::endl; + return -1; + } + pluginPath = argv[1]; + return RUN_ALL_TESTS(); +} diff --git a/src/rview.cpp b/src/rview.cpp deleted file mode 100644 index 2da3096c..00000000 --- a/src/rview.cpp +++ /dev/null @@ -1,12 +0,0 @@ -/* ============================================================ - * The rekonq project - * ============================================================ - * SPDX-License-Identifier: GPL-3.0-only - * Copyright (C) 2022 aqua - * ============================================================ - * Description: View Interface - * ============================================================ */ - -#include "rview.h" - -rView::rView(const QUrl &url, QWidget *parent) : QWidget(parent) {} diff --git a/src/rview.h b/src/rview.h deleted file mode 100644 index 4f4cc897..00000000 --- a/src/rview.h +++ /dev/null @@ -1,25 +0,0 @@ -/* ============================================================ - * The rekonq project - * ============================================================ - * SPDX-License-Identifier: GPL-3.0-only - * Copyright (C) 2022 aqua - * ============================================================ - * Description: View Interface - * ============================================================ */ - -#pragma once - -#include -#include - -class rView : public QWidget { - Q_OBJECT - -public: - explicit rView(const QUrl &url = QUrl(), QWidget *parent = nullptr); - -signals: - void iconChanged(const QIcon &); - void urlChanged(const QUrl &); - void titleChanged(const QString &); -}; diff --git a/src/webengine/webview.cpp b/src/webengine/webview.cpp deleted file mode 100644 index 1aef01e2..00000000 --- a/src/webengine/webview.cpp +++ /dev/null @@ -1,25 +0,0 @@ -/* ============================================================ - * The rekonq project - * ============================================================ - * SPDX-License-Identifier: GPL-3.0-only - * Copyright (C) 2022 aqua - * ============================================================ - * Description: Qt WebEngine View - * ============================================================ */ - -#include "webview.h" -#include -#include - -WebView::WebView(const QUrl &url, QWidget *parent) : rView(url, parent), view(new QWebEngineView(this)) -{ - auto *layout = new QVBoxLayout; - layout->setContentsMargins(0, 0, 0, 0); - layout->addWidget(view); - setLayout(layout); - - connect(view, &QWebEngineView::iconChanged, this, [this](const QIcon &icon) { emit iconChanged(icon); }); - connect(view, &QWebEngineView::urlChanged, this, [this](const QUrl &url) { emit urlChanged(url); }); - connect(view, &QWebEngineView::titleChanged, this, [this](const QString &title) { emit titleChanged(title); }); - view->load(url); -} diff --git a/src/webengine/webview.h b/src/webengine/webview.h deleted file mode 100644 index b838c5ac..00000000 --- a/src/webengine/webview.h +++ /dev/null @@ -1,25 +0,0 @@ -/* ============================================================ - * The rekonq project - * ============================================================ - * SPDX-License-Identifier: GPL-3.0-only - * Copyright (C) 2022 aqua - * ============================================================ - * Description: Qt WebEngine View - * ============================================================ */ - -#pragma once - -#include "rview.h" - -class QWebEngineView; - -class WebView final : public rView { - Q_OBJECT - -public: - explicit WebView(const QUrl &url = QUrl(), QWidget *parent = nullptr); - ~WebView() final = default; - -private: - QWebEngineView *view; -}; -- cgit v1.2.1