1

当您选择一个项目(使用箭头键)并按回车时使用 PopupCompletion 模式 - lineEdit 应该为空(按下回车时我清除 lineEdit),但 lineEdit 不会为空。(如果您再次按“Enter”,它将清空 lineEdit)。所以我认为按回车确实清除了lineEdit,但按回车也告诉QCompleter将所选项目插入lineEdit,所以似乎什么也没发生。

但是,如果您单击使用箭头选择它的项目 - 一切正常。

我试图在互联网上找到解决方案,但我发现只有一个人有同样的问题: http: //lists.trolltech.com/qt-interest/2006-10/thread00985-0.html。遗憾的是没有答案。请阅读他的问题,因为这将有助于理解我的问题。

QCompleter 插入所选项目后如何清理 LineEdit?(捕捉激活信号无济于事)

4

1 回答 1

11

这里的问题是完成器实际上包含一个弹出窗口,它实际上是一个单独的QAbstractItemView小部件(请参阅QCompleter::popup()文档)。因此,当您在 QCompleter 上按“Enter”时,关键事件实际上会进入弹出窗口而不是行编辑。

有两种不同的方法可以解决您的问题:

选项1

将完成者的激活信号连接到行编辑的清除槽,但将其作为QueuedConnection

QObject::connect(completer, SIGNAL(activated(const QString&)),
                 lineEdit, SLOT(clear()),
                 Qt::QueuedConnection);

使用直接连接不起作用的原因是因为您本质上取决于从信号调用插槽的顺序。使用 aQueuedConnection可以解决这个问题。从代码维护的角度来看,我真的不喜欢这个解决方案,因为仅仅通过查看代码并不清楚你的意图是什么。

选项 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);

这个选项工作量更大,但更清楚你在做什么。此外,如果您愿意,您可以通过这种方式执行其他自定义。

于 2012-08-10T16:48:55.090 回答