
由于最近需要使用ActiveX,一般来说可以使用微软提供的MFC或者ATL框架来开发,由于我个人对这部分内容不是很熟悉,好在Qt也提供对于ActiveX的支持。本文主要记录个人学习ActiveX的一些内容,方便日后查阅。本文以Qt5(5.3.1)提供的ActiveX为参考,但是由于ActiveX这部分比较稳定,因此Qt4应该也是一样的。
- 概述
- 支持将已有的COM或者ActiveX空间引入到Qt的应用程序中
- 支持将Qt应用程序或者Qt的对象导出成COM对象或者ActiveX控件供他人使用
- 使用QAxContainer模块,通过QAxObject和QAxWidget分别支持COM对象和ActiveX控件的开发,可以通过这两个对象将外部的COM或者ActiveX组件接入到Qt应用程序
- 使用QAxServer模块,通过QAxAggregated、QAxBindable和QAxFactory类,通过了进程内和可执行程序exe两种方式的COM Server模式,用来将Qt写的内容导出为COM或者ActiveX供他人使用。

- 使用QtActiveX创建COM或ActiveX Server
- Qt作为Server支持的模式
- TEMPLATE = app
- QT += axserver
- RC_FILE = qaxserver.rc
当作为进程内dll的时候,需要这样写.pro文件
- TEMPLATE = lib
- QT += axserver
- CONFIG += dll
- DEF_FILE = qaxserver.def
- RC_FILE = qaxserver.rc
- ...
但是我们不用去操心这些内容,因为一般我们是用向导来生成Qt工程的,当你勾选ActiveX Server模式的时候,向导已经帮你写好这些内容了。
- TEMPLATE = lib
- VERSION = 2.5
- ...
- COM进程外和进程内使用方式的比较
- 开发Server的过程
- #include <QWidget>
- class MyActiveX : public QWidget
- {
- Q_OBJECT
只里面Q_OBJECT宏不能少,它提供了关于类MyActiveX的一些元数据信息, 接下来继续添加ActiveX的一些信息
- Q_CLASSINFO("ClassID", "{1D9928BD-4453-4bdd-903D-E525ED17FDE5}")
- Q_CLASSINFO("InterfaceID", "{99F6860E-2C5A-42ec-87F2-43396F4BE389}")
- Q_CLASSINFO("EventsID", "{0A3E9F27-E4F1-45bb-9E47-63099BCCD0E3}")
Q_CLASSINFO定义了COM组件的一些信息,这里面ClassID和InterfaceID是必须的,当你需要使用COM中的事件时,EventsID就需要添加进来。里面的128位的字符串使用GUID.exe工具生成(微软提供的一个小工具,可以生成全球唯一的标识符,可以在系统文件夹或者visual Studio的工具菜单中找到),工具提供了多种模式,可以选择需要的模式复制添加到代码中:

名称 | 值和含义 |
Version | 类的版本号,默认值是1.0 |
Description | 类的描述 |
ClassID | 类的ID(COM中用来唯一确定一个类的方式) |
InterfaceID | 接口ID(COM中用来唯一确定接口的方式) |
EventsID | 事件ID |
DefaultProperty | 默认属性 |
DefaultSignal | 默认的时间 |
LicenseKey | 类的许可号,默认未开启,如果开启使用类需要许可号 |
StockEvents | TODO:??? |
ToSuperClass | 暴露父类的接口 |
Insertable | 设置"yes"后可以被列到OLE2容器中,默认未设置 |
Aggregatable | 默认是"yes",COM支持聚合 |
Creatable | 设置为“no”调用者不能使用该类 |
RegisterObject | 仅能用在进程外方式的COM中 |
MIME | 该COM控件支持的文件格式描述 |
CoClassAlias | 类的名称在IDL中被修改为CoClassAlias指定的名字 |
继续上面的代码,接下来可以添加一些属性到COM组件中,可以使用另一个宏Q_PROPERTY
- Q_PROPERTY(int value READ value WRITE setValue)
之后可以像写Qt程序那样来完成。
- public:
- MyActiveX(QWidget *parent = 0)
- ...
- int value() const;
- public slots:
- void setValue(int v);
- ...
- signals:
- void valueChange(int v);
- ...
- };
Qt的ActiveX框架会将Qt类中的要素转换为COM中的标准要素供其他调用者使用,具体来说:
Qt数据类型 | COM 属性数据类型 |
bool | VARIANT_BOOL |
QString | BSTR |
int | int |
uint | unsigned int |
double | double |
qlonglong | CY |
qulonglong | CY |
QColor | OLE_COLOR |
QDate | DATE |
QDateTime | DATE |
QTime | DATE |
QFont | IFontDisp* |
QPixmap | IPixtureDisp* |
QVariant | VARIANT |
QVariantList | SAFEARRAY(VARIANT) |
QStringList | SAFEARRAY(BSTR) |
QByteArray | SAFEARRAY(BYTE) |
QRect | User defined type |
QSize | User defined type |
QPoint | User defined type |
Qt数据类型 | 对应COM的数据类型 |
bool | [in] VARIANT_BOOL |
bool& | [in,out] VARIANT_BOOL* |
QString, const Qtring& | [in] BSTR |
QString& | [in, out] BSTR* |
QStinrg& | [in, out] BSTR* |
int | [in] int |
int& | [in, out]int |
uint | [in,out]unsigned int |
uint& | [int, out] unsigned int* |
double | [in] double |
QColor, const QColor& | [in] OLE_COLOR |
QColor& | [in,out] OLE_COLOR* |
QDate, const QDate& | [in] DATE |
QDate& | [in,out]DATE* |
QDateTime, const QDateTime& | [in] DATE |
QDateTime& | [in,out]DATE* |
QFont, const QFont& | [in] IFontDisp* |
QFont& | [in,out]IFontDisp** |
QPixmap, const QPixmap& | [in]IPictureDisp* |
QPixmap& | [in,out]IPictureDisp** |
QList<QVariant> | [in]SAFEARRAY(VARIANT) |
QList<QVariant>& | [in,out]SAFEARRAY(VARIANT)* |
QObject* | [in] IDispatch* |
- 发布COM控件
- QAXFACTORY_BEGIN("{ad90301a-849e-4e8b-9a91-0a6dc5f6461f}",
- "{a8f21901-7ff7-4f6a-b939-789620c03d83}")
- QAXCLASS(MyWidget)
- QAXCLASS(MyWidget2)
- QAXTYPE(MySubType)
- QAXFACTORY_END()
上面这段代码把MyWidget和MyWidget2导出为可供外部调用的COM对象,并注册MySubType类型可供MyWidget和MyWidget2中的属性和参数使用。
- 进程外COM控件的编写
- #include <QApplication>
- #include <QAxFactory>
- int main(int argc, char *argv[])
- {
- QApplication app(argc, argv);
- if (!QAxFactory::isServer()) {
- // create and show main window
- }
- return app.exec();
- }
http://blog.****.net/csxiaoshui/article/category/5705027
http://download.****.net/detail/csxiaoshui/9728972