移动无边框窗体(设置标志位更流畅,或者发送WM_SYSCOMMAND和SC_MOVE + HTCAPTION消息)

移动无边框窗体的代码网上很多,其原理都是一样的,但是是有问题的,我这里只是对其修正一下

网上的代码仅仅实现了两个事件

  1. void EditDialog::mousePressEvent(QMouseEvent *event)
  2. {
  3. if (event->button() == Qt::LeftButton) {
  4. m_DragPosition = event->globalPos() - this->pos();
  5. event->accept();
  6. }
  7. }
  8. void EditDialog::mouseMoveEvent(QMouseEvent *event)
  9. {
  10. if (event->buttons() && Qt::LeftButton) {
  11. move(event->globalPos() - m_DragPosition);
  12. event->accept();
  13. }
  14. }

但是这样就会有一个问题,就是当鼠标在一个实现了mousePressEvent的类上点击时(比如QPushButton)会被该类优先处理此事件

而不会将事件传递到窗体的mousePressEvent中。继续,当移动鼠标到这个按钮外时(假设点在了QPushButton上)会触发窗体的mouseMoveEvent

从而导致计算坐标时发生错误,此时你就会看到窗体闪了一下,变动了位置,鼠标也没有停在前面按下的按钮之上。

解决办法也很简单,就是再多声明一个bool变量来判断,并实现mouseReleaseEvent即可

  1. void EditDialog::mousePressEvent(QMouseEvent *event)
  2. {
  3. if (event->button() == Qt::LeftButton) {
  4. m_Drag = true;
  5. m_DragPosition = event->globalPos() - this->pos();
  6. event->accept();
  7. }
  8. }
  9. void EditDialog::mouseMoveEvent(QMouseEvent *event)
  10. {
  11. if (m_Drag && (event->buttons() && Qt::LeftButton)) {
  12. move(event->globalPos() - m_DragPosition);
  13. event->accept();
  14. }
  15. }
  16. void EditDialog::mouseReleaseEvent(QMouseEvent *)
  17. {
  18. m_Drag = false;
  19. }

这样,就完成了无边框窗体的拖动。可是,这样做的效率并不高,因为鼠标每次move时都会触发事件,计算位置,移动窗口,重绘窗口……

当窗体上有QWebView部件时,特别是网页中有图片,Flash时,你就会发现用上面的方案去移动窗体时会非常不流畅。

如果不考虑跨平台,只针对Windows平台,那么我建议用Windows下的标准方法,模拟标题栏移动消息,既简单又高效

  1. void MainWindow::mousePressEvent(QMouseEvent *event)
  2. {
  3. if (ReleaseCapture())
  4. SendMessage(HWND(this->winId()), WM_SYSCOMMAND, SC_MOVE + HTCAPTION, 0);
  5. event->ignore();
  6. }

这样,在拖动窗体时只会在松开鼠标时才将窗体移动过去,这样就避免了第一种方法的低效率问题

http://blog.csdn.net/aqtata/article/details/8902889

上一篇:(转)C#中Trim()、TrimStart()、TrimEnd()的用法 .


下一篇:【Android进度条】三种方式实现自定义圆形进度条ProgressBar