需求
根据业务需求,在输入框(QlineEdit)下面做一个模糊查找的弹窗(PopupWidget),当鼠标点击弹窗以外的地方时,这个窗口关闭,当光标在输入框重新输入内容时,再次弹框。
实现方法
方法一
在网上查找到的方法;设置窗口标志位:
setWindowFlags(Qt::FramelessWindowHint | Qt::Popup);
此方法可以达到弹出式窗口的效果,但是出现了一个致命的问题,但我再次show()这个窗口时(popupWidget),程序会闪退,网上找了很久才找到解决办法。
重写popupWidget的closeEvent()函数:
void PopupWidget::closeEvent(QCloseEvent *event)
{
event->ignore();
this->hide();
}
这里我只是在鼠标点击PopupWidget窗口以外的地方时,把这个窗口隐藏起来。
方法二
安装事件过滤器,捕捉鼠标点击事件,发出信号,在PopupWidget注册该信号,关闭窗口。
这种方法不推荐。虽然可以到达想要的效果,但是比较繁琐,你需要在每个窗口都重写eventFillter()函数或者重写keyPressedEvent()事件,特别是在一个界面比较复杂,子窗口比较多的情况下,会相当繁琐,所以这不是一个明智的选择。
具体实现省略…
方法三
为什么会有方法三呢? 其实在第一个方法已经完全可以实现弹出式窗口的效果了,由于我需要在股票symbol输入框输入内容时,(公司做的是一个股票期货的客户端)弹出模糊查找的结果,但是在弹框出现后,symbol输入框失去焦点,当你再次点击输入框想要重新获得焦点时,弹框又隐藏起来了。
这里想要达成的效果是:弹框出现后,焦点一直都在symbol输入框内,模糊查找返回的弹框也要实现一个弹出式窗口的效果。
这个是富途牛牛的效果。
解决焦点不在symbol输入框的问题:
1、让弹框始终没有焦点
this->setFocusPolicy(Qt::NoFocus);//方法无效。
另外说一下,要给一个窗口设置焦点前,得先让它是一个活跃窗口
this->activateWindow();
this->setFocus(); //单单这句是无效的
设置窗口无焦点:
this->setAttribute(Qt::WA_ShowWithoutActivating);//Show the widget without making it active.
this->setAttribute(Qt::WA_X11DoNotAcceptFocus);//对非X11平台不生效
以上方法都试过了,都没有效果。
猜测可能是由于设置了setWindowFlags(Qt::FramelessWindowHint | Qt::Popup);此窗口是一个弹出式窗口,所以无效。
2、去掉setWindowFlags(Qt::Popup),让PopupWidget为一个普通窗口,并设置
this->setAttribute(Qt::WA_ShowWithoutActivating);/
果然,窗口弹出之后,焦点还是在symbol输入框内。
既然PopupWidget是一个普通窗口,那我要怎样实现弹窗的效果呢?
自定义QlineEdit控件,重写焦点事件:
void focusInEvent(QFocusEvent *event) override;
void focusOutEvent(QFocusEvent *event) override;
void MySymbolLineEdit::focusInEvent(QFocusEvent *event)
{
QLineEdit::focusInEvent(event);
emit focusInSignal();
}
void MySymbolLineEdit::focusOutEvent(QFocusEvent *event)
{
QLineEdit::focusOutEvent(event);
emit focuaOutSignal();
}
然后再PopupWidget注册信号槽就可以实现弹框效果了。