Qt我可以在构造函数中将信号/插槽连接到self吗?

时间:2012-07-25 00:56:42

标签: c++ qt signals-slots

编辑:与信号/插槽/连接无关。问题是构造函数调用构造函数。

可能有更好的方法可以做到这一点 - 我有兴趣听到这些......

我有一个源自QLabel的MyClass。我希望在信号中传递有关派生类的更多数据,而不是基本信号。所以我创建了一个插槽来拦截customContextMenuRequested信号并发出一个包含更多数据的修改后的信号。

当我尝试在构造函数中连接此信号时,我的插槽永远不会被调用。但是,如果我移动策略并将线连接到父窗口小部件(而不是类层次结构父窗口),以便它们在MyClass完全构造之后执行,那么我的插槽将被调用。但是我总是希望这个类能够连接起来,这似乎是我想要它的构造函数,而不是指望父类记住这样做。

我有什么问题吗?或者更好的方法来向信号添加数据?

MyClass::MyClass() : QLabel()
{
    QFont currFont = font();
    currFont.setPointSize(15);
    setFont(currFont);

    setBackgroundRole(QPalette::Mid);

    std::cout << "connecting customContextMenuRequested" << std::endl;
    /** PROBLEM START */
    setContextMenuPolicy(Qt::CustomContextMenu);

    // Is there anything wrong with connecting from "this" to "this" in a constructor?

    QObject::connect(this, SIGNAL(customContextMenuRequested(const QPoint&)),
                     this, SLOT(addCellDataToMenuContextRequest(const QPoint&)));
     /* PROBLEM END **/
}


MyClass::MyClass(QString &cellString, int row, int col)
    : QLabel(cellString)
{
    MyClass();
    setRow(row);
    setCol(col);

}

// This one is a slot
void MyClass::addCellDataToMenuContextRequest(const QPoint& pos)
{
        // This doesn't get printed if I connect in my constructor,
        // but it does print if I do the same connect from a parent widget.
    std::cout << "called addCellDataToMenuContextRequest" << std::endl;

    emit customContextMenuRequestedForCell(pos, _row, _col);
}

所以我希望父窗口小部件只查找customContextMenuRequestedForCell,但是现在,父窗口小部件似乎也需要负责customContextMenuRequested。

2 个答案:

答案 0 :(得分:3)

实际上,如果你使用的是C ++ 11,你可以调用(排序)另一个构造函数。它被称为delegating constructor。但我不认为这会使问题消失。您的问题似乎是在调用connect()时未完全构造元对象。此外,您可能需要转到Qt 5才能使用C ++ 11。

解决方案是延迟连接,直到完全构造对象。您可以start a timer间隔为零。它将在下一个事件循环处理时触发,这肯定会在您的对象完全构建之后。

然后在你的timerEvent中,建立连接并终止计时器。

编辑:没看到您的编辑。看起来你找到了解决方案。然后忽略这一点。 :)

顺便说一句。你没有调用另一个构造函数。您创建了一个临时的MyClass对象。

答案 1 :(得分:0)

使这种“更干净”的方法是在MyClass中重新实现QWidget :: mouseReleaseEvent()。实现它的方法是,如果传递给mouseReleaseEvent的QMouseEvent类型不是右键单击鼠标释放,则调用QLabel :: mouseReleaseEvent(event)。如果是右键单击鼠标释放事件,则可以发出自定义信号。这样可以使用QLabel / QWidget提供的现有鼠标按钮释放处理代码,同时允许您拦截要发出自定义信号的一种情况。

编辑哦,并确保在mouseReleaseEvent处理自定义案例后调用event-&gt; accept()。