Implemented Popup dialog widget
authorJoel Holdsworth <joel@airwebreahe.org.uk>
Wed, 25 Sep 2013 07:18:28 +0000 (16:18 +0900)
committerJoel Holdsworth <joel@airwebreathe.org.uk>
Sun, 13 Oct 2013 10:05:22 +0000 (11:05 +0100)
CMakeLists.txt
pv/widgets/popup.cpp [new file with mode: 0644]
pv/widgets/popup.h [new file with mode: 0644]

index 74b91af9601c3102e06d38da89f27f5e2b8fe15a..6fa08273ef73ed8d865d971b4e1da42d8784c722 100644 (file)
@@ -142,6 +142,7 @@ set(pulseview_SOURCES
        pv/view/view.cpp
        pv/view/viewport.cpp
        pv/view/decode/annotation.cpp
+       pv/widgets/popup.cpp
 )
 
 # This list includes only QObject derrived class headers
diff --git a/pv/widgets/popup.cpp b/pv/widgets/popup.cpp
new file mode 100644 (file)
index 0000000..f89e38a
--- /dev/null
@@ -0,0 +1,218 @@
+/*
+ * This file is part of the PulseView project.
+ *
+ * Copyright (C) 2013 Joel Holdsworth <joel@airwebreathe.org.uk>
+ *
+ * This program is free software; you can redistribute it and/or modify
+ * it under the terms of the GNU General Public License as published by
+ * the Free Software Foundation; either version 2 of the License, or
+ * (at your option) any later version.
+ *
+ * This program is distributed in the hope that it will be useful,
+ * but WITHOUT ANY WARRANTY; without even the implied warranty of
+ * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
+ * GNU General Public License for more details.
+ *
+ * You should have received a copy of the GNU General Public License
+ * along with this program; if not, write to the Free Software
+ * Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA  02110-1301 USA
+ */
+
+#include <algorithm>
+
+#include <QtGui>
+
+#include "popup.h"
+
+using namespace std;
+
+namespace pv {
+namespace widgets {
+
+const unsigned int Popup::ArrowLength = 15;
+const unsigned int Popup::ArrowOverlap = 3;
+const unsigned int Popup::MarginWidth = 10;
+
+Popup::Popup(QWidget *parent) :
+       QWidget(parent, Qt::Popup | Qt::FramelessWindowHint),
+       _point(),
+       _pos(Left)
+{
+}
+
+const QPoint& Popup::point() const
+{
+       return _point;
+}
+
+Popup::Position Popup::position() const
+{
+       return _pos;
+}
+
+void Popup::set_position(const QPoint point, Position pos)
+{
+       _point = point, _pos = pos;
+
+       setContentsMargins(
+               MarginWidth + ((pos == Right) ? ArrowLength : 0),
+               MarginWidth + ((pos == Bottom) ? ArrowLength : 0),
+               MarginWidth + ((pos == Left) ? ArrowLength : 0),
+               MarginWidth + ((pos == Top) ? ArrowLength : 0));
+
+}
+
+QPolygon Popup::arrow_polygon() const
+{
+       QPolygon poly;
+
+       const QPoint p = mapFromGlobal(_point);
+       const int l = ArrowLength + ArrowOverlap; 
+
+       switch (_pos)
+        {
+       case Right:
+               poly << QPoint(p.x() + l, p.y() - l);
+               break;
+
+       case Bottom:
+               poly << QPoint(p.x() - l, p.y() + l);
+               break;
+
+        case Left:
+       case Top:
+               poly << QPoint(p.x() - l, p.y() - l);
+               break;
+       }
+
+       poly << p;
+
+       switch (_pos)
+        {
+       case Right:
+       case Bottom:
+               poly << QPoint(p.x() + l, p.y() + l);
+               break;
+
+        case Left:
+               poly << QPoint(p.x() - l, p.y() + l);
+               break;
+               
+       case Top:
+               poly << QPoint(p.x() + l, p.y() - l);
+               break;
+       }
+
+       return poly;
+}
+
+QRegion Popup::arrow_region() const
+{
+       return QRegion(arrow_polygon());
+}
+
+QRect Popup::bubble_rect() const
+{
+       return QRect(
+               QPoint((_pos == Right) ? ArrowLength : 0,
+                       (_pos == Bottom) ? ArrowLength : 0),
+               QSize(width() - ((_pos == Left || _pos == Right) ?
+                               ArrowLength : 0),
+                       height() - ((_pos == Top || _pos == Bottom) ?
+                               ArrowLength : 0)));
+}
+
+QRegion Popup::bubble_region() const
+{
+       const QRect rect(bubble_rect());
+
+       const unsigned int r = MarginWidth;
+       const unsigned int d = 2 * r;
+       return QRegion(rect.adjusted(r, 0, -r, 0)).united(
+               QRegion(rect.adjusted(0, r, 0, -r))).united(
+               QRegion(rect.left(), rect.top(), d, d,
+                       QRegion::Ellipse)).united(
+               QRegion(rect.right() - d, rect.top(), d, d,
+                       QRegion::Ellipse)).united(
+               QRegion(rect.left(), rect.bottom() - d, d, d,
+                       QRegion::Ellipse)).united(
+               QRegion(rect.right() - d, rect.bottom() - d, d, d,
+                       QRegion::Ellipse));
+}
+
+QRegion Popup::popup_region() const
+{
+       return arrow_region().united(bubble_region());
+}
+
+void Popup::reposition_widget()
+{
+       QPoint o;
+
+       if (_pos == Right || _pos == Left)
+               o.ry() = -height() / 2;
+       else
+               o.rx() = -width() / 2;
+
+       if (_pos == Left)
+               o.rx() = -width();
+       else if(_pos == Top)
+               o.ry() = -height();
+
+       move(_point + o);
+}
+
+void Popup::paintEvent(QPaintEvent*)
+{
+       QPainter painter(this);
+       painter.setRenderHint(QPainter::Antialiasing);
+
+       const QColor outline_color(QApplication::palette().color(
+               QPalette::Dark));
+
+       const QRegion b = bubble_region();
+       const QRegion bubble_outline = QRegion(rect()).subtracted(
+               b.translated(1, 0).intersected(b.translated(0, 1).intersected(
+               b.translated(-1, 0).intersected(b.translated(0, -1)))));
+
+       painter.setPen(Qt::NoPen);
+       painter.setBrush(QApplication::palette().brush(QPalette::Window));
+       painter.drawRect(rect());
+
+       const QPoint ArrowOffsets[] = {
+               QPoint(1, 0), QPoint(0, -1), QPoint(-1, 0), QPoint(0, 1)};
+
+       const QRegion a(arrow_region());
+       const QRegion arrow_outline = a.subtracted(
+               a.translated(ArrowOffsets[_pos]));
+
+       painter.setClipRegion(bubble_outline.subtracted(a).united(
+               arrow_outline));
+       painter.setBrush(outline_color);
+       painter.drawRect(rect());
+}
+
+void Popup::resizeEvent(QResizeEvent*)
+{
+       reposition_widget();
+       setMask(popup_region());
+}
+
+void Popup::showEvent(QShowEvent*)
+{
+       reposition_widget();
+}
+
+void Popup::mouseReleaseEvent(QMouseEvent *e)
+{
+       assert(e);
+
+       // We need our own out-of-bounds click handler because QWidget counts
+       // the drop-shadow region as inside the widget
+       if(!bubble_rect().contains(e->pos()))
+               close();
+}
+
+} // namespace widgets
+} // namespace pv
+
diff --git a/pv/widgets/popup.h b/pv/widgets/popup.h
new file mode 100644 (file)
index 0000000..402fc6b
--- /dev/null
@@ -0,0 +1,82 @@
+/*
+ * This file is part of the PulseView project.
+ *
+ * Copyright (C) 2013 Joel Holdsworth <joel@airwebreathe.org.uk>
+ *
+ * This program is free software; you can redistribute it and/or modify
+ * it under the terms of the GNU General Public License as published by
+ * the Free Software Foundation; either version 2 of the License, or
+ * (at your option) any later version.
+ *
+ * This program is distributed in the hope that it will be useful,
+ * but WITHOUT ANY WARRANTY; without even the implied warranty of
+ * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
+ * GNU General Public License for more details.
+ *
+ * You should have received a copy of the GNU General Public License
+ * along with this program; if not, write to the Free Software
+ * Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA  02110-1301 USA
+ */
+
+#ifndef PULSEVIEW_PV_WIDGETS_POPUP_H
+#define PULSEVIEW_PV_WIDGETS_POPUP_H
+
+#include <QWidget>
+
+namespace pv {
+namespace widgets {
+
+class Popup : public QWidget
+{
+public:
+       enum Position
+       {
+               Right,
+               Top,
+               Left,
+               Bottom
+       };
+
+private:
+       static const unsigned int ArrowLength;
+       static const unsigned int ArrowOverlap;
+       static const unsigned int MarginWidth;
+
+public:
+       Popup(QWidget *parent);
+
+       const QPoint& point() const;
+       Position position() const;
+
+       void set_position(const QPoint point, Position pos);
+
+private:
+       QPolygon arrow_polygon() const;
+
+       QRegion arrow_region() const;
+
+       QRect bubble_rect() const;
+
+       QRegion bubble_region() const;
+
+       QRegion popup_region() const;
+
+       void reposition_widget();
+
+       void paintEvent(QPaintEvent*);
+
+       void resizeEvent(QResizeEvent*);
+
+       void showEvent(QShowEvent*);
+
+       void mouseReleaseEvent(QMouseEvent *e);
+
+private:
+       QPoint _point;
+       Position _pos;
+};
+
+} // namespace widgets
+} // namespace pv
+
+#endif // PULSEVIEW_PV_WIDGETS_POPUP_H