Как запустить QThread из QWizardPage и получить доступ к полю ()

Мне нужен совет для доступа к полю (QString name) переменная в QWizardPage из QThread, Я создаю какой-то инсталлятор и хочу выполнить установку в отдельном потоке.

Моя цель: когда я достиг страницы фиксации / установки, я хочу выполнить код для "установки" и обновить QWizardPage с моим прогрессом, пока не закончил.

Функция установки зависит от многих field() переменные из других QWizardPages, Поэтому я попытался выполнить эту функцию установки из QThread, который определен во внутреннем классе из моего QWizardPage, Проблема в том, field()-функция является нестатическим членом, и поэтому он не работает. И поэтому у меня нет идей запускать мою функцию установки параллельно моей WizardPage.

Я попробовал что-то вроде этого:

InstallPage.h

class InstallPage : public QWizardPage
{
    Q_OBJECT
    class WorkerThread : public QThread
    {
        Q_OBJECT
        void run() override;
    };

public:
    InstallPage(QWidget *parent = 0);

private: 
    QLabel *lProgress;
    WorkerThread *installer;
    void install(); 
};


InstallPage.c

InstallPage::InstallPage(QWidget *parent)
    : QWizardPage(parent)
{
...
    installer = new WorkerThread(this);
    installer->start();
}

void InstallPage::WorkerThread::run()
{
    if(field("checkBox1").ToBool()) 
    {
         doStuff();
    }
}

//QT-Creator says at field("checkBox1"): 
//error: call to non-static member function without an object argument

Я также открыт для любой другой идеи, чтобы заставить мой установщик работать. Может, кто-то знает что-то, о чем я не думал.

1 ответ

Другой подход заключается в создании работника (QObject), который живет в другом потоке, который выполняет тяжелую задачу и уведомляет о состоянии этой задачи с помощью сигналов:

#include <QtWidgets>

class InitialPage: public QWizardPage
{
public:
    InitialPage(QWidget *parent = nullptr): QWizardPage(parent)
    {
        QSpinBox *spinbox = new QSpinBox;
        QLineEdit *lineedit = new QLineEdit;
        QVBoxLayout *lay = new QVBoxLayout(this);
        lay->addWidget(spinbox);
        lay->addWidget(lineedit);

        registerField("value1", spinbox);
        registerField("value2", lineedit);
    }

};

class InstallWorker: public QObject
{
    Q_OBJECT
public:
    InstallWorker(QObject *parent=nullptr): QObject(parent)
    {

    }
public Q_SLOTS:
    void install(int param1, const QString & param2)
    {
        Q_EMIT started();
        for(int i=0; i < 100; i++){
            qDebug() << __PRETTY_FUNCTION__ << i << param1 << param2;
            QThread::msleep(100);
            Q_EMIT progressChanged(i);
        }
        qDebug()<< __PRETTY_FUNCTION__ << "finished";
        Q_EMIT finished();
    }
Q_SIGNALS:
    void started();
    void progressChanged(int value);
    void finished();
};

class InstallPage: public QWizardPage
{
    Q_OBJECT
public:
    InstallPage(QWidget *parent = nullptr): QWizardPage(parent),
        label(new QLabel), progressbar(new QProgressBar)
    {

        QVBoxLayout *lay = new QVBoxLayout(this);
        lay->addWidget(label);
        lay->addWidget(progressbar);

        progressbar->setMinimum(0);
        progressbar->setMaximum(100);

        thread = new QThread(this);
        worker.moveToThread(thread);
        connect(&worker, &InstallWorker::started, this, &InstallPage::onStarted);
        connect(&worker, &InstallWorker::finished, this, &InstallPage::onFinished);
        connect(&worker, &InstallWorker::progressChanged, this, &InstallPage::onProgressChanged);
        thread->start();
    }
    ~InstallPage(){
        thread->quit();
        thread->wait();
    }
    void initializePage(){
        start_install();
    }
private Q_SLOTS:
    void start_install(){
        int param1 = field("value1").toInt();; 
        QString param2 = field("value2").toString();
        QMetaObject::invokeMethod(&worker, "install", Qt::QueuedConnection, Q_ARG(int, param1), Q_ARG(QString, param2));
    }
    void onStarted(){
        for(QWizard::WizardButton which: {QWizard::BackButton, QWizard::NextButton, QWizard::CancelButton})
            if(QAbstractButton * button = wizard()->button(which))
                button->setEnabled(false);
    }
    void onFinished(){
        for(QWizard::WizardButton which: {QWizard::BackButton, QWizard::NextButton, QWizard::CancelButton})
            if(QAbstractButton * button = wizard()->button(which))
                button->setEnabled(true);
        wizard()->next();
    }
    void onProgressChanged(int value){
        progressbar->setValue(value);
        label->setNum(value);
    }
private:
    InstallWorker worker;
    QThread *thread;
    QLabel *label;
    QProgressBar *progressbar;
};

class FinalPage: public QWizardPage
{
public:
    FinalPage(QWidget *parent = nullptr): QWizardPage(parent)
    {

    }
};

#include "main.moc"

int main(int argc, char *argv[]) {
    QApplication app(argc, argv);
    QWizard wizard;
    wizard.addPage(new InitialPage);
    wizard.addPage(new InstallPage);
    wizard.addPage(new FinalPage);
    wizard.show();
    return app.exec();
}
Другие вопросы по тегам