1

我有两个 ViewModelMainVMAddVM. 在main.cpp,MainVM中以这种方式声明:

MainVM *mvm;    
int main(int argc, char *argv[])
{
    ...
    mvm = new MainVM();
    ...
    engine.rootContext()->setContextProperty("mainContext", mvm);
    engine.rootContext()->setContextProperty("addContext", new AddVM());
    ...
}

MainVM,我有这个Q_PROPERTY

class MainVM : public QObject
{
    Q_OBJECT
    ...
    PROPERTY(QVector<Plot*>, plots)
    ...
public:
    ...
    QSqlDatabase db;
    int maxPlotId, maxSpaceId, maxTenantId, maxHeadId, maxLeaseId;
    ...
};

PROPERTY宏这样做:

#define PROPERTY(QType, name) \
    Q_PROPERTY(QType name READ name WRITE set##name NOTIFY name##Changed) \
    public: \
        QType name(){return m_##name;} \
        void set##name(QType value){if(m_##name != value){m_##name = value; emit name##Changed();}} \
        Q_SIGNAL void name##Changed(); \
    private: \
        QType m_##name;

在我的AddVM我有另一个Q_PROPERTYnewPlot 和一个Q_INVOKABLEaddNewPlot:

class AddVM : public QObject
{
    Q_OBJECT
    PROPERTY(Plot*, newPlot)
public:
    explicit AddVM(QObject *parent = nullptr);
    Q_INVOKABLE void addNewPlot();
}; 

在上面AddVM.cpp,我有这些:

#include "MainVM.h"
extern MainVM *mvm;

addNewPlot功能有这些说明:

void AddVM::addNewPlot()
{
    mvm->db.open();
    QSqlQuery query;
    query.prepare("INSERT INTO Plots (Name, Description) VALUES(:Name, :Description)");
    query.bindValue(":Name", newPlot()->name());
    query.bindValue(":Description", newPlot()->description());
    query.exec();
    mvm->db.close();

    mvm->plots().push_back(newPlot());
    setnewPlot(new Plot());
    newPlot()->setid(++mvm->maxPlotId);
}

此函数中的所有内容都按预期工作,除了mvm->plots().push_back(newPlot());行!这不会newPlotQVectorof 中添加MainVM

编辑

可能最好的方法是getter像这样在宏中重新定义:

QType& name(){return m_##name;} \

并且我现有的代码无需任何修改即可工作。

4

1 回答 1

0

当您调用时mvm->plots(),它会返回真实绘图向量的副本(这是我的猜测,因为您没有显示什么plots())。因此,将新图添加到副本而不是原始向量。所以,我认为最好的方法是添加一个MainVM名为“addPlot()”的函数,例如,您可以在内部直接添加到绘图向量中。

于 2020-07-09T07:31:12.777 回答