如何创建 Qt 插件?

简单三部曲

  1. 定义接口类或接口基类并使用 Q_DECLARE_INTERFACE 宏进行声明
  2. 所有的插件都需要继承该基类并继承 QObject(不带界面插件) or QWidget(带界面插件)
  3. 在插件类中添加 Q_PLUGIN_METADATA 导出元数据,使用 Q_INTERFACES 指定继承基类(该基类需要使用 Q_DECLARE_INTERFACE 声明过)

如何生成?

示例

*.pro 文件内容如下

# 插件依赖模块
QT += core gui widgets # 生成目标文件名称
TARGET = FlowWidget
TEMPLATE = lib
CONFIG += plugin # 插件输出目录
DESTDIR = $$[QT_INSTALL_PLUGINS]/generic DEFINES += QT_DEPRECATED_WARNINGS SOURCES += \
flowWidget.cpp HEADERS += \
flowWidget.h DISTFILES += FlowWidget.json include($$PWD/interface/interface.pri)

interface.pri 内容如下:

SOURCES += \

HEADERS += \
$$PWD/flowInterface.h INCLUDEPATH += $$PWD

纯接口类定义如下:

#include <QtPlugin>
class QImage;
class QString;
class QWidget;
class FlowInterface
{
public:
FlowInterface() {}
virtual ~FlowInterface() {}
virtual void process() = 0;
/*!
属性页
*/
virtual QWidget *propertyPage() = 0;
}; #define FlowInterface_iid "com.xiongwei.cheung.FlowInterface" Q_DECLARE_INTERFACE(FlowInterface, FlowInterface_iid)

注意:在插件类中我们需要实现接口中定义的纯虚函数

插件定义分为带界面和不带界面两种,采用多重继承的方式分别继承 QWidget 或 QObject 即可.

  1. 带界面插件定义如下:
class FlowWidget : public QWidget , FlowInterface
{
Q_OBJECT
Q_PLUGIN_METADATA(IID "com.xiongwei.cheung.FlowWidget" FILE "FlowWidget.json")
Q_INTERFACES(FlowInterface)
public:
explicit FlowWidget(QWidget *parent = nullptr);
void process();
QWidget *propertyPage();
signals: public slots: private: }; FlowWidget::FlowWidget(QWidget *parent) : QWidget(parent)
{ } void FlowWidget::process()
{
// to do something
} QWidget *FlowWidget::propertyPage()
{
return this;
}

FlowWidget.json 文件是可选的,可通过该文件实现密钥插件

{
"Keys" : [ "FlowWidget" ]
}
  1. 不带界面插件定义如下:
class FlowObject : public QObject , FlowInterface
{
Q_OBJECT
Q_PLUGIN_METADATA(IID "com.xiongwei.cheung.FlowObject")
Q_INTERFACES(FlowInterface)
public:
explicit FlowObject(QObject *parent = nullptr);
void process();
QWidget *propertyPage();
signals: public slots: private: }; FlowObject::FlowObject(QObject *parent) : QObject(parent)
{ } void FlowObject::process()
{
// to do something
} QWidget *FlowObject::propertyPage()
{
return nullptr;
}

如何使用?

main.cpp 文件中添加插件加载路径

QCoreApplication::applicationDirPath().append("\generic");

在工程文件中包含插件接口基类文件 flowInterface.h

#include <QPluginLoader>
#include "flowInterface.h" void load()
{
// 添加一个枚举目录所有动态库文件即可加载所有插件
QPluginLoader pluginLoader("FlowWidget.dll");
// 打印插件元数据,即 json 文件内容中的键值数据
qDebug() << pluginLoader.metaData();
// 加载该插件
QObject *plugin = pluginLoader.instance();
if (plugin) {
// 转换成接口基类指针
FlowInterface *p = qobject_cast<FlowInterface *>(plugin);
if (p){
// 显示界面
p->propertyPage()->show();
} }
}

注意一个插件类中只能存在一个 Q_PLUGIN_METADATA 宏,使用该宏会导出 qt_plugin_instanceqt_plugin_query_metadata 俩个函数。

显然如果使用 2 次 Q_PLUGIN_METADATA 宏会导致函数重定义.

实际上如果向导出多个插件的可以使用 QGenericPlugin 的手法根据 keys and `` 创建不同的对象.

如何创建 Qt 插件?的更多相关文章

  1. [Qt Creator 快速入门] 第9章 国际化、帮助系统和Qt插件

    一.国际化 国际化的英文表述为Internationalization,通常简写为I18N(首尾字母加中间的字符数),一个应用程序的国际化就是使该应用程序可以让其他国家的用户使用的过程. Qt支持现在 ...

  2. [Qt插件]-03创建Qt Designer自定义部件

    如何创建自定义部件并添加到Qt Designer来爽快的拖动部件可视化界面设计?   Qt Designer基于插件的架构使得它可以使用用户设计或者第三方提供的自定义部件,就像使用标准的Qt部件一样. ...

  3. [Qt插件]-02创建应用程序插件(插件化开发的一种思路)

    本篇是学习Qt Creator快速入门,插件开发的笔记   分为两部分 创建插件 使用插件的应用程序(测试插件)   插件是被使用的应用程序加载使用的. 是使用插件的应用程序定义接口,插件按照接口来实 ...

  4. VS2013中使用QT插件后每次重新编译问题

    环境 系统:win7 64位旗舰版 软件:VS2013.QT5.5.1-32位.Qt5 Visual Studio Add-in1.2.4 概述 使用QT Visual Studio插件打开pro项目 ...

  5. VS2017上在线和离线安装Qt插件(在线安装)

    版权声明:如需转载,请告知博主并声明出处 https://blog.csdn.net/CLinuxF/article/details/88816436文章目录前言在线安装离线安装(推荐)前言很多朋友喜 ...

  6. 【Win10+eclipse+MinGW+QT安装教程】已有eclipse环境下配置QT插件出错详解

    (纪念一下装了一整天濒临绝望的自己[/doge]) (以下所有下载项一律使用32位) step1:下载安装eclipse.适配的jdk.MinGW和QT 1.下载32位eclipse for c++以 ...

  7. 设置qt插件路径

    1.在Qt中使用 WebKit 浏览器核心 使用 QtWebKit 需要在工程文件(*.pro)中加入: QT +=webkitQT += network 2.QtWebKit的flash支持 QtW ...

  8. Qt 插件学习(一)

    插件是什么 注意:这儿暂时不考虑静态插件(潜意识中总觉得它根本就不算插件). 插件是一个动态库(共享库).动态库是一个独立的文件中的独立模块,可被多个程序访问. 先看动态库的两种用法 1. 程序链接时 ...

  9. [Qt插件]-01Qt插件&&提升部件(自定义控件)

    本篇的文档为:How to Create Qt Plugins https://doc.qt.io/qt-5/plugins-howto.html   前言 Qt插件(Qt Plugin)就是一个共享 ...

随机推荐

  1. “没有找到mfc100u.dll”的解决方法

    现在需要安装 MindManager 2016 思维导图软件时,打开软件提示找不到 mfc100u.dll,无法执行程序.之前一直好好的,现在换电脑了安装提示这个问题,然后百度找的解决方案: 需要去微 ...

  2. 13Ajax和JQuery

    1.Ajax 1.1是什么? “Asynchronous Javascript And XML”(异步JavaScript和XML), 并不是新的技术,只是把原有的技术,整合到一起而已. 1.使用CS ...

  3. 在一个div上增加遮罩

    有一个需求是给一个视频增加遮罩 我研究了下 抽象出来就是给一个div增加遮罩 原理是:最外层的div使用relative定位     然后里面放两个子div  一个是不被遮的  另一个是遮罩(用abs ...

  4. App.after

    解释: App.after可以增加APP级的切面,触发的时机是在所拦截的对应生命周期方法执行之后. 方法参数:Object Object 参数说明: 参数名 类型 必填 默认值 说明 methods ...

  5. Codeforces 845D - Two TVs(贪心)

    原题链接:http://codeforces.com/problemset/problem/845/D 题意:一个人在驾照考试中,路边有“限速XX”.“没有限速”.“可以超车”.“不能超车”路牌, 以 ...

  6. if和switch的选择结构

    1. Java中的if选择结构,包括以下形式. *基本if选择结构:可以处理一单一或组合条件的情况. *if-else选择结构:可以处理简单的条件分支情况. *多重if选择结构:可以处理连续区间的条件 ...

  7. python cv2在验证码识别中的使用

    使用函数cv2.imread(filepath,flags)读入一副图片 filepath:要读入图片的完整路径 flags:读入图片的标志 cv2.IMREAD_COLOR:默认参数,读入一副彩色图 ...

  8. SHADER 用 step替代 if-else

    今天聊起这个问题,百度发现了这个优化方式: https://blog.csdn.net/liu_if_else/article/details/77455639

  9. [WCF REST] WebServiceHost 不依赖配置文件启动简单服务

    最近用WPF启动 WCF REST 服务,发现app.config 配置好烦,简单一个exe 可以到处搬动,还非得带一个累赘配置,不小心丢了程序就跑不起来. 最后决定,砍去WCF配置项,用WebSer ...

  10. DB-MDM:MDM/主数据管理 百科

    ylbtech-DB-MDM:MDM/主数据管理 百科 主数据管理(MDM Master Data Management)描述了一组规程.技术和解决方案,这些规程.技术和解决方案用于为所有利益相关方( ...