【问题标题】:Qt Designer - How to connect a signal to a static function?Qt Designer - 如何将信号连接到静态函数?
【发布时间】:2013-06-13 17:34:16
【问题描述】:

好的,我正在使用 Qt Designer 构建 GUI。我已经设法弄清楚如何制作 menuBar 并且我已经向该栏添加了一些操作,但现在我需要连接这些操作以使它们执行某些操作。

具体来说,在我的文件菜单上,我有简单的打开操作。我希望此操作运行调用我的 QFileDialog 等的函数,但我不知道如何执行此操作。

那么,如何将我的 actionOpen 连接到我的静态函数?

我使用的是最新的 Qt,5.0.2

我在这里有点沮丧。这显然是某人可能需要做的最基本的事情之一,但我在网络上的任何地方都找不到任何真正的解决方案。从缺少的Qt wiki,到其他人的问题,似乎没有人真正有一个明确的答案。旧版本的 Qt 有答案,但在那些旧版本中,信号显然无法连接到静态函数,所以这些是无关紧要的。似乎没有人知道如何通过 Qt Designer 做到这一点。此外,没有人明确说明在哪里放置什么。

我的 main.cpp 文件中有这一行:

QObject::connect(actionOpen, &actionOpen::triggered, fileOpen)

我在 Qt Designer 中创建了一个名为“actionOpen”的对象,有一个名为“触发”的信号,并且我在 main.cpp 中的 main 下面定义了一个名为“fileOpen”的函数。这似乎遵循正确的语法,但它会引发许多错误。

另外,我可以在 Qt Creator 中反复单击构建,每次它出现不同数量的错误,消失和重新出现,我什至不接触代码。我开始认为这个 IDE 有点像 POS。

编辑:

这是我的文件。也许这会有所帮助。

主窗口.h

#ifndef MAINWINDOW_H
#define MAINWINDOW_H

#include <QMainWindow>

namespace Ui {
class MainWindow;
}

class MainWindow : public QMainWindow
{
    Q_OBJECT

public:
    explicit MainWindow(QWidget *parent = 0);
    ~MainWindow();

public slots:
    void fileOpen();

private:
    Ui::MainWindow *ui;
};

#endif // MAINWINDOW_H

main.cpp

#include "mainwindow.h"
#include <QApplication>
#include <iostream>
#include <fstream>

using namespace std;


int main(int argc, char *argv[])
{
    QApplication a(argc, argv);
    MainWindow w;

    w.show();


    return a.exec();
}

主窗口.cpp

#include "mainwindow.h"
#include "ui_mainwindow.h"
#include <iostream>

using namespace std;



MainWindow::MainWindow(QWidget *parent) :
    QMainWindow(parent),
    ui(new Ui::MainWindow)
{
    QObject::connect(ui->actionOpen, &QAction::triggered, &MainWindow::fileOpen);
    ui->setupUi(this);
}

MainWindow::~MainWindow()
{
    delete ui;

}

void fileOpen()
{

    /*
    QString fileName = QFileDialog::getOpenFileName(this, tr("Open File"), QString(),
        tr("Text Files (*.txt);;C++ Files (*.cpp *.h)"));

    if (!fileName.isEmpty()) {
        QFile file(fileName);
        if (!file.open(QIODevice::ReadOnly)) {
            QMessageBox::critical(this, tr("Error"), tr("Could not open file"));
            return;
        }
        QTextStream in(&file);
        ui->textEdit->setText(in.readAll());
        file.close();
    }
    */

    cout << "Hello!";
}

【问题讨论】:

  • 这不是“最基本的东西之一”。信号通常连接到插槽,这些插槽是 QObject 派生类的非静态成员函数。我建议您使用常规插槽。如果确实需要,应使用连接到静态函数。
  • @Riateche 那么我该怎么做呢?你是说我需要为actionOpen对象创建自己的类,继承自QObject,并给它一个成员函数来运行?这似乎有点假,必须为我想使用的 Qt Designer 中的每个对象和操作创建一个类。我可以看到通常信号从一个对象发送到另一个对象的插槽,但是对于一个简单的打开文件命令,没有接收器对象。我不明白在这种情况下连接它们的模型是如何工作的。
  • @TylerJandreau 不,我已经阅读了那篇文章,但我的问题仍未得到解答。那篇文章指的是旧语法,而最佳答案链接到新语法的页面,我也读过,但仍然感到困惑。
  • 对不起,伙计,我从未尝试过这样做。我希望我能帮忙。

标签: c++ qt signals-slots qt-designer


【解决方案1】:

第二个参数不正确。您应该指定类名,而不是对象名。所以应该是:

QObject::connect(actionOpen, &QAction::triggered, fileOpen);

完整的工作示例(经过测试):

void fileOpen() {
  qDebug() << "ok";
}

int main(int argc, char *argv[]) {
  QApplication a(argc, argv);
  QMenu menu;
  QAction* actionOpen = menu.addAction("test");
  QObject::connect(actionOpen, &QAction::triggered, fileOpen);
  menu.show();
  return a.exec();
}

【讨论】:

  • 好吧,这是有道理的。现在我已经设法消除了除一个之外的所有错误。现在我得到“actionOpen”:未声明的标识符。我猜这是因为 actionOpen 没有在代码中的任何地方声明。我在 ui 文件中创建了它,所以它只存在于 'mainwindow.ui' xml 类型文件中。那么如何引用该文件中的内容呢?还是需要单独在代码中声明?
  • 我将连接调用移动到 mainwindow.cpp 文件而不是 main.cpp,这似乎是我可以使用 ui->fileOpen 访问 fileOpen 的唯一地方,然后我收到错误 MainWindow: :fileOpen '函数调用缺少参数列表,使用 &MainWindow::fileOpen 创建一个指针,我尝试使用它,它给出了 'void QtPrivate::FunctionPointer::call...无法将参数 2 从 QObject* 转换为主窗口。我不知道发生了什么事。我是 C++ 的菜鸟,这对我来说太混乱了。我要做的就是连接到名为fileOpen的简单函数,为什么不能这么简单。
  • 你写过MainWindow::fileOpen吗?这是不正确的。或者您要编译的确切代码是什么?
  • 两种方法我都试过了,函数只用'void fileOpen'声明,还有'void MainWindow::fileOpen',它们都不起作用。
  • 我编辑了我的原始帖子并粘贴了我文件中的代码。也许您现在可以发现错误。
【解决方案2】:

1.) 创建常规槽并调用静态函数。

2.) 我想您可以创建一个单例 Q_OBJECT 类并连接到它 - 当然,就像选项 1 一样,您可以调用任何静态/全局函数。

/**
 * Pseudo-code!, the singleton should be in its own header file 
 * not in the global main.cpp file.
 */
class Singleton : public QObject
{
Q_OBJECT
public:
  Singleton() : QObject() {}
  static Singleton* getInstance() {
    if(!_instance) _instance = new Singleton();
    return _instance;
  }
public slots:
  void mySlot() { qDebug() << __FILE__ << " " << __FUNCTION__ << __LINE__;
  }
private:
  static Singleton* _instance;
};

Singleton* Singleton::_instance = NULL;

int main(int argc, char *argv[])
{
  QApplication a(argc, argv);
  MainWindow w;
  w.show();

  Singleton* instance = Singleton::getInstance();
  QObject::connect(&w, SIGNAL(destroyed()), instance, SLOT(mySlot()));

  return a.exec();
}

我只是尽力回答这个问题,但就像上面的许多 cmets - 这不是我需要做的事情。我可以说 QtCreator/Designer 是一个非常了不起的工具,当你克服一些学习曲线时,它会变得不那么令人沮丧。

【讨论】:

  • 在 Qt 5 中没有必要。您可以在没有任何 QObject 的情况下使用静态函数。
  • @Pavel Strakhov 如果我们不能使用this 并且函数在另一个类中?
猜你喜欢
  • 1970-01-01
  • 1970-01-01
  • 1970-01-01
  • 1970-01-01
  • 1970-01-01
  • 1970-01-01
  • 2018-09-07
  • 1970-01-01
  • 1970-01-01
相关资源
最近更新 更多