/**************************************************************************** ** ** Copyright (C) 2016 The Qt Company Ltd. ** Contact: https://www.qt.io/licensing/ ** ** This file is part of the test suite of the Qt Toolkit. ** ** $QT_BEGIN_LICENSE:GPL-EXCEPT$ ** Commercial License Usage ** Licensees holding valid commercial Qt licenses may use this file in ** accordance with the commercial license agreement provided with the ** Software or, alternatively, in accordance with the terms contained in ** a written agreement between you and The Qt Company. For licensing terms ** and conditions see https://www.qt.io/terms-conditions. For further ** information use the contact form at https://www.qt.io/contact-us. ** ** GNU General Public License Usage ** Alternatively, this file may be used under the terms of the GNU ** General Public License version 3 as published by the Free Software ** Foundation with exceptions as appearing in the file LICENSE.GPL3-EXCEPT ** included in the packaging of this file. Please review the following ** information to ensure the GNU General Public License requirements will ** be met: https://www.gnu.org/licenses/gpl-3.0.html. ** ** $QT_END_LICENSE$ ** ****************************************************************************/ #include #include #include #include #include #include #include #include #include #include #include #include enum TabletPointType { TabletButtonPress, TabletButtonRelease, TabletMove }; struct TabletPoint { TabletPoint(const QPointF &p = QPointF(), TabletPointType t = TabletMove, Qt::MouseButton b = Qt::LeftButton, QTabletEvent::PointerType pt = QTabletEvent::UnknownPointer, qreal prs = 0, qreal rotation = 0) : pos(p), type(t), button(b), ptype(pt), pressure(prs), angle(rotation) {} QPointF pos; TabletPointType type; Qt::MouseButton button; QTabletEvent::PointerType ptype; qreal pressure; qreal angle; }; class ProximityEventFilter : public QObject { Q_OBJECT public: explicit ProximityEventFilter(QObject *parent) : QObject(parent) { } bool eventFilter(QObject *, QEvent *event) override; static bool tabletPenProximity() { return m_tabletPenProximity; } signals: void proximityChanged(); private: static bool m_tabletPenProximity; }; bool ProximityEventFilter::eventFilter(QObject *, QEvent *event) { switch (event->type()) { case QEvent::TabletEnterProximity: case QEvent::TabletLeaveProximity: ProximityEventFilter::m_tabletPenProximity = event->type() == QEvent::TabletEnterProximity; emit proximityChanged(); qDebug() << event; break; default: break; } return false; } bool ProximityEventFilter::m_tabletPenProximity = false; class EventReportWidget : public QWidget { Q_OBJECT public: EventReportWidget(); public slots: void clearPoints() { m_points.clear(); update(); } signals: void stats(QString s, int timeOut = 0); protected: void mouseDoubleClickEvent(QMouseEvent *event) override { outputMouseEvent(event); } void mouseMoveEvent(QMouseEvent *event) override { outputMouseEvent(event); } void mousePressEvent(QMouseEvent *event) override { outputMouseEvent(event); } void mouseReleaseEvent(QMouseEvent *event) override { outputMouseEvent(event); } void tabletEvent(QTabletEvent *) override; bool event(QEvent *event) override; void paintEvent(QPaintEvent *) override; void timerEvent(QTimerEvent *) override; private: void outputMouseEvent(QMouseEvent *event); bool m_lastIsMouseMove = false; bool m_lastIsTabletMove = false; Qt::MouseButton m_lastButton = Qt::NoButton; QVector m_points; QVector m_touchPoints; QPointF m_tabletPos; int m_tabletMoveCount = 0; int m_paintEventCount = 0; }; EventReportWidget::EventReportWidget() { setAttribute(Qt::WA_AcceptTouchEvents); startTimer(1000); } void EventReportWidget::paintEvent(QPaintEvent *) { QPainter p(this); int lineSpacing = fontMetrics().lineSpacing(); int halfLineSpacing = lineSpacing / 2; const QRectF geom = QRectF(QPoint(0, 0), size()); p.fillRect(geom, Qt::white); p.drawRect(QRectF(geom.topLeft(), geom.bottomRight() - QPointF(1,1))); p.setPen(Qt::white); QPainterPath ellipse; ellipse.addEllipse(0, 0, halfLineSpacing * 5, halfLineSpacing); for (const TabletPoint &t : qAsConst(m_points)) { if (geom.contains(t.pos)) { QPainterPath pp; pp.addEllipse(t.pos, halfLineSpacing, halfLineSpacing); QRectF pointBounds(t.pos.x() - halfLineSpacing, t.pos.y() - halfLineSpacing, lineSpacing, lineSpacing); switch (t.type) { case TabletButtonPress: p.fillPath(pp, Qt::darkGreen); if (t.button != Qt::NoButton) p.drawText(pointBounds, Qt::AlignCenter, QString::number(t.button)); break; case TabletButtonRelease: p.fillPath(pp, Qt::red); if (t.button != Qt::NoButton) p.drawText(pointBounds, Qt::AlignCenter, QString::number(t.button)); break; case TabletMove: if (t.pressure > 0.0) { p.setPen(t.ptype == QTabletEvent::Eraser ? Qt::red : Qt::black); if (t.angle != 0.0) { p.save(); p.translate(t.pos); p.scale(t.pressure, t.pressure); p.rotate(t.angle); p.drawPath(ellipse); p.restore(); } else { p.drawEllipse(t.pos, t.pressure * halfLineSpacing, t.pressure * halfLineSpacing); } p.setPen(Qt::white); } else { p.fillRect(t.pos.x() - 2, t.pos.y() - 2, 4, 4, Qt::black); } break; } } } // Draw haircross when tablet pen is in proximity if (ProximityEventFilter::tabletPenProximity() && geom.contains(m_tabletPos)) { p.setPen(Qt::black); p.drawLine(QPointF(0, m_tabletPos.y()), QPointF(geom.width(), m_tabletPos.y())); p.drawLine(QPointF(m_tabletPos.x(), 0), QPointF(m_tabletPos.x(), geom.height())); } p.setPen(Qt::blue); for (QPointF t : m_touchPoints) { p.drawLine(t.x() - 40, t.y(), t.x() + 40, t.y()); p.drawLine(t.x(), t.y() - 40, t.x(), t.y() + 40); } ++m_paintEventCount; } void EventReportWidget::tabletEvent(QTabletEvent *event) { QWidget::tabletEvent(event); bool isMove = false; m_tabletPos = event->posF(); switch (event->type()) { case QEvent::TabletMove: m_points.push_back(TabletPoint(m_tabletPos, TabletMove, m_lastButton, event->pointerType(), event->pressure(), event->rotation())); update(); isMove = true; ++m_tabletMoveCount; break; case QEvent::TabletPress: m_points.push_back(TabletPoint(m_tabletPos, TabletButtonPress, event->button(), event->pointerType(), event->rotation())); m_lastButton = event->button(); update(); break; case QEvent::TabletRelease: m_points.push_back(TabletPoint(m_tabletPos, TabletButtonRelease, event->button(), event->pointerType(), event->rotation())); update(); break; default: Q_ASSERT(false); break; } if (!(isMove && m_lastIsTabletMove)) { QDebug d = qDebug(); d << event << " global position = " << event->globalPos() << " cursor at " << QCursor::pos(); if (event->button() != Qt::NoButton) d << " changed button " << event->button(); } m_lastIsTabletMove = isMove; } bool EventReportWidget::event(QEvent *event) { switch (event->type()) { case QEvent::TouchBegin: case QEvent::TouchUpdate: event->accept(); m_touchPoints.clear(); for (const QTouchEvent::TouchPoint &p : static_cast(event)->touchPoints()) m_touchPoints.append(p.pos()); update(); break; case QEvent::TouchEnd: m_touchPoints.clear(); update(); break; default: return QWidget::event(event); } return true; } void EventReportWidget::outputMouseEvent(QMouseEvent *event) { if (event->type() == QEvent::MouseMove) { if (m_lastIsMouseMove) return; // only show one move to keep things readable m_lastIsMouseMove = true; } qDebug() << event; } void EventReportWidget::timerEvent(QTimerEvent *) { emit stats(QString("%1 moves/sec, %2 frames/sec").arg(m_tabletMoveCount).arg(m_paintEventCount)); m_tabletMoveCount = 0; m_paintEventCount = 0; } int main(int argc, char *argv[]) { QApplication app(argc, argv); ProximityEventFilter *proximityEventFilter = new ProximityEventFilter(&app); app.installEventFilter(proximityEventFilter); QMainWindow mainWindow; mainWindow.setWindowTitle(QString::fromLatin1("Tablet Test %1").arg(QT_VERSION_STR)); EventReportWidget *widget = new EventReportWidget; QObject::connect(proximityEventFilter, &ProximityEventFilter::proximityChanged, widget, QOverload::of(&QWidget::update)); widget->setMinimumSize(640, 480); QMenu *fileMenu = mainWindow.menuBar()->addMenu("File"); fileMenu->addAction("Clear", widget, &EventReportWidget::clearPoints); QObject::connect(widget, &EventReportWidget::stats, mainWindow.statusBar(), &QStatusBar::showMessage); QAction *quitAction = fileMenu->addAction("Quit", qApp, &QCoreApplication::quit); quitAction->setShortcut(Qt::CTRL + Qt::Key_Q); mainWindow.setCentralWidget(widget); mainWindow.show(); return app.exec(); } #include "main.moc"