0

我想编写一个属性连接器,用于以 QML 方式连接 C++ Qt 中的属性。它可能是这样的:

new PropConnector ( receiver, "propName", sender, "propName" );

propConnector 类是一个小型 QObject 类,它使用发送者作为父类,侦听属性更改的信号并设置接收者的适当属性。像这样:

class PropConnector : public QObject
{
    Q_OBJECT
private:
    QObject *_sender;
    QObject *_receiver;
    QString _senderProp;
    QString _receiverProp;
public:
    PropConnector(QObject *sender, QObject *receiver, QString senderProp, QString receiverProp) : 
      QObject(sender)
      ,_sender(sender)
      ,_receiver(receiver)
      ,_senderProp(senderProp)
      ,_receiverProp(receiverProp)
    { 
        // Is it possible to get the proper property-changed notifier signal?
        //connect(sender, SIGNAL( on<PROPERTY>Changed() ), this, SLOT( forward ) );
    }

private slots:
    void forward()
    {
        _receiver->setProperty( receiverProp, _sender->property( senderProp ) );
    }
};

唯一的问题是,我怎样才能找到我想听的属性名称的正确信号名称?我想,这是可能的,但我直到现在才通过阅读元对象文档找到答案。

编辑:在凯文的建议下,我能够实施该课程。

propconnector.h

#ifndef PROPCONNECTOR_H
#define PROPCONNECTOR_H

#include <QObject>
#include <QString>

class PropConnector : public QObject
{
    Q_OBJECT
private:
    QObject *_sender;
    QObject *_receiver;
    QString _senderProperty;
    QString _receiverProperty;
public:
    PropConnector(QObject *sender, QString senderProperty, QObject *receiver, QString receiverProperty);

private slots:
    void forward();
};

#endif // PROPCONNECTOR_H

propconnector.cpp

#include "propconnector.h"
#include <QMetaObject>
#include <QMetaProperty>

PropConnector::PropConnector(QObject *sender, QString senderProperty, QObject *receiver, QString receiverProperty) : 
  QObject(sender)
  ,_sender(sender)
  ,_receiver(receiver)
  ,_senderProperty(senderProperty)
  ,_receiverProperty(receiverProperty)
{ 
    const QMetaObject *senderMeta = sender->metaObject();
    const int index = senderMeta->indexOfProperty(senderProperty.toUtf8().constData());
    if (index != -1) {
        const QMetaProperty p = senderMeta->property(index);
        if ( p.hasNotifySignal() ) {
            const QMetaMethod s = p.notifySignal();

            QString sig = QString("2%1").arg(s.signature());
            const char *ssig = SLOT(forward());
            bool ok = connect(sender, sig.toStdString().c_str() , this, SLOT(forward()));   
            int i=0;
            i++;
        }
    }       
}

void PropConnector::forward()
{
    _receiver->setProperty( _receiverProperty.toStdString().c_str(), _sender->property( _senderProperty.toStdString().c_str() ) );
}
4

1 回答 1

3

您可以从QMetaProperty

QMetaObject *senderMeta = sender->metaObject();
const int index = senderMeta->indexOfProperty(senderProperty.toUtf8().constData());
if (index != -1) {
    const QMetaProperty property = senderMeta->property(index);
    if (property.hasNotifySignal) {
        const QMetaMethod notifySignal = property.notifySignal();

        connect(sender, notifySignal, receiver, metaMethodOfSlot);     
    }
}

未经测试,可能无法编译。metaMethodOfSlot将是QMetaMethod接收器插槽,以类似的方式获得。

于 2017-01-08T12:34:37.567 回答