当您选择一个项目(使用箭头键)并按回车时使用 PopupCompletion 模式 - lineEdit 应该变为空(我在按下回车时清除 lineEdit),但 lineEdit 不会变为空。 (如果您再次按“Enter”,它将清空 lineEdit)。所以我认为按回车键会清除 lineEdit,但按回车键也会告诉 QCompleter 将选定的项目插入到 lineEdit 中,所以看起来什么也没发生。
但是,如果您单击项目而不是使用箭头选择它 - 一切正常。
我试图在互联网上找到解决方案,但我发现只有一个人有同样的问题:http://lists.trolltech.com/qt-interest/2006-10/thread00985-0.html .遗憾的是没有答案。请阅读他的问题,因为这将有助于理解我的问题。
如何在 QCompleter 插入所选项目后清理 LineEdit? (捕获激活信号无济于事)
最佳答案
这里的问题是,完成器实际上包含一个弹出窗口,它实际上是一个单独的 QAbstractItemView
小部件(请参阅 QCompleter::popup() 文档)。因此,当您在 QCompleter 上按“Enter”时,按键事件实际上会转到弹出窗口而不是行编辑。
有两种不同的方法可以解决您的问题:
选项1
将完成者的激活信号连接到行编辑的清除槽,但作为 QueuedConnection
执行:
QObject::connect(completer, SIGNAL(activated(const QString&)),
lineEdit, SLOT(clear()),
Qt::QueuedConnection);
使用直接连接不起作用的原因是因为您基本上依赖于从信号调用槽的顺序。使用 QueuedConnection
可以解决这个问题。从代码维护的角度来看,我不太喜欢这种解决方案,因为仅通过查看代码并不清楚您的意图。
选项 2
在弹出窗口周围编写一个事件过滤器,以过滤掉“Enter”键以明确清除行编辑。您的事件过滤器最终会看起来像这样:
class EventFilter : public QObject
{
Q_OBJECT
public:
EventFilter(QLineEdit* lineEdit, QObject* parent = NULL)
:QObject(parent)
,mLineEdit(lineEdit)
{ }
virtual ~EventFilter()
{ }
bool eventFilter(QObject* watched, QEvent* event)
{
QAbstractItemView* view = qobject_cast<QAbstractItemView*>(watched);
if (event->type() == QEvent::KeyPress)
{
QKeyEvent* keyEvent = dynamic_cast<QKeyEvent*>(event);
if (keyEvent->key() == Qt::Key_Return ||
keyEvent->key() == Qt::Key_Enter)
{
mLineEdit->clear();
view->hide();
return true;
}
}
return false;
}
private:
QLineEdit* mLineEdit;
};
然后您将在完成者的弹出窗口中安装事件过滤器:
EventFilter* filter = new EventFilter(lineEdit);
completer->popup()->installEventFilter(filter);
此选项需要更多工作,但更清楚您在做什么。此外,如果您愿意,您可以通过这种方式执行额外的自定义。
关于c++ - 从 QCompleter 中选择项目后无法清除 QLineEdit,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/11865129/