Fixed map jumping when starting a drag
[situare] / src / map / mapview.cpp
index 827cc7c..8cdcf29 100644 (file)
@@ -3,6 +3,7 @@
    Copyright (C) 2010  Ixonos Plc. Authors:
 
        Sami Rämö - sami.ramo@ixonos.com
+       Pekka Nissinen - pekka.nissinen@ixonos.com
 
    Situare is free software; you can redistribute it and/or
    modify it under the terms of the GNU General Public License
 
 #include <QDebug>
 #include <QMouseEvent>
+#include <QParallelAnimationGroup>
 
-#ifndef Q_WS_MAEMO_5
-    #include <QGLWidget>
-#endif // Q_WS_MAEMO_5
-
-#ifdef Q_WS_MAEMO_5
-    #include <QAbstractKineticScroller>
-#endif // Q_WS_MAEMO_5
-
+#include "coordinates/scenecoordinate.h"
 #include "mapcommon.h"
+#include "mapscroller.h"
+
 #include "mapview.h"
 
-MapView::MapView(QWidget *parent) : QGraphicsView(parent)
+const qreal MS_PER_S = 1000;
+
+// const values for tuning the kinetic scroll effect
+const int KINETIC_MIN_DRAG_LENGTH_VIEW_PIXELS = 30;
+const int KINETIC_MAX_TIME_FROM_LAST_MOUSE_EVENT_MS = 100;
+const qreal KINETIC_MAX_VIEW_DISTANCE_FACTOR = 0.8;
+const int KINETIC_SCROLL_TIME_MS = 750;
+const qreal KINETIC_SPEED_TO_DISTANCE_FACTOR = 0.15 * sqrt(KINETIC_SCROLL_TIME_MS / MS_PER_S);
+
+const qreal ZOOM_TIME_MS = 350; ///< Length of the zoom effect (ms)
+
+MapView::MapView(QWidget *parent)
+    : QGraphicsView(parent),
+      m_doubleTapZoomRunning(false),
+      m_panelIsOpen(false)
 {
-/**
-  * Use OpenGL for desktop to gain some performance in map view.
-  * OpenGL can't be used in scratchbox.
-  */
-#ifndef Q_WS_MAEMO_5
-    setViewport(new QGLWidget);
-#endif // !Q_WS_MAEMO_5
-
-/**
-  * Use kinetic scrolling for Maemo5 and QGraphicsViews drag mode
-  * ScrollHandDrag for other environments
-  */
-#ifdef Q_WS_MAEMO_5
-    QAbstractKineticScroller *scroller = property("kineticScroller")
-                                         .value<QAbstractKineticScroller *>();
-    if (scroller)
-        scroller->setEnabled(true);
-#else
-    setDragMode(QGraphicsView::ScrollHandDrag);
-#endif // Q_WS_MAEMO_5
+    qDebug() << __PRETTY_FUNCTION__;
 
     setHorizontalScrollBarPolicy(Qt::ScrollBarAlwaysOff);
     setVerticalScrollBarPolicy(Qt::ScrollBarAlwaysOff);
+
+    m_zoomAnimation = new QPropertyAnimation(this, "viewScale", this);
+    connect(m_zoomAnimation, SIGNAL(finished()),
+        this, SIGNAL(viewZoomFinished()));
+    setOptimizationFlag(QGraphicsView::DontAdjustForAntialiasing);
+
+    m_scroller = &MapScroller::getInstance();
+
+    m_scrollAndZoomAnimation = new QParallelAnimationGroup();
+    m_scrollAndZoomAnimation->addAnimation(m_scroller);
+    m_scrollAndZoomAnimation->addAnimation(m_zoomAnimation);
+    connect(m_scrollAndZoomAnimation, SIGNAL(finished()),
+            this, SLOT(doubleTapZoomFinished()));
 }
 
-void MapView::setZoomLevel(int zoomLevel)
+void MapView::centerToSceneCoordinates(const SceneCoordinate &coordinate, bool isUserDragAction)
 {
-    m_zoomTargetScale = pow(2, zoomLevel - MAX_MAP_ZOOM_LEVEL);
-    m_zoomScaleDelta = (m_zoomTargetScale - currentScale()) / (ZOOM_FPS * ZOOM_TIME);
+    qDebug() << __PRETTY_FUNCTION__ << "coordinate" << coordinate;
+
+    QPointF target = coordinate.toPointF();
 
-    startTimer(1000/ZOOM_FPS);
+    if (!isUserDragAction) {
+        target += m_centerHorizontalShift;
+    }
+
+    centerOn(target);
+}
+
+void MapView::disableCenterShift()
+{
+    qWarning() << __PRETTY_FUNCTION__;
+
+    m_panelIsOpen = false;
+    updateCenterShift();
+
+    ///< @todo Update center position
 }
 
-void MapView::timerEvent(QTimerEvent *event)
+void MapView::doubleTapZoomFinished()
 {
-    qreal scaleFactor = currentScale();
+    qDebug() << __PRETTY_FUNCTION__;
 
-//    qDebug() << __PRETTY_FUNCTION__
-//             << "abs(m_zoomTargetScale - scaleFactor)" << fabs(m_zoomTargetScale - scaleFactor)
-//             << "abs(m_zoomScaleDelta)" << fabs(m_zoomScaleDelta);
+    m_doubleTapZoomRunning = false;
+    emit zoomIn();
+}
+
+void MapView::enableCenterShift()
+{
+    qWarning() << __PRETTY_FUNCTION__;
 
-    if (fabs(m_zoomTargetScale - scaleFactor) <= fabs(m_zoomScaleDelta)) {
-        scaleFactor = m_zoomTargetScale;
-        killTimer(event->timerId());
+    m_panelIsOpen = true;
+    updateCenterShift();
+
+    ///< @todo Update center position
+}
+
+void MapView::mouseDoubleClickEvent(QMouseEvent *event)
+{
+    qDebug() << __PRETTY_FUNCTION__;
+
+    if (m_zoomLevel + 1 <= OSM_MAX_ZOOM_LEVEL) {
+        QPoint pressPosition = mapToScene(event->pos()).toPoint();
+        QPoint viewCenterPosition = mapToScene(width() / 2 - 1, height() / 2 - 1).toPoint();
+        QPoint zoomPosition = viewCenterPosition - ((viewCenterPosition - pressPosition) / 2);
+
+        m_scrollAndZoomAnimation->stop();
+        m_doubleTapZoomRunning = true;
+
+        m_scroller->setEasingCurve(QEasingCurve::Linear);
+        m_scroller->setDuration(ZOOM_TIME_MS);
+        m_scroller->setStartValue(SceneCoordinate(m_scenePosition.x(), m_scenePosition.y()));
+        m_scroller->setEndValue(SceneCoordinate(zoomPosition.x(), zoomPosition.y()));
+
+        m_zoomAnimation->setEasingCurve(QEasingCurve::InQuad);
+        m_zoomAnimation->setDuration(ZOOM_TIME_MS);
+        m_zoomAnimation->setStartValue(viewScale());
+        m_zoomAnimation->setEndValue(pow(2, m_zoomLevel+1 - OSM_MAX_ZOOM_LEVEL));
+
+        m_scrollAndZoomAnimation->start();
     }
-    else {
-        scaleFactor += m_zoomScaleDelta;
+}
+
+void MapView::mouseMoveEvent(QMouseEvent *event)
+{
+    qDebug() << __PRETTY_FUNCTION__;
+
+    if (m_doubleTapZoomRunning)
+        return;
+
+    m_scenePosition += m_mouseLastScenePosition - mapToScene(event->pos()).toPoint();
+
+    if (m_index >= VALUES)
+        m_index = 0;
+
+    m_dragMovement[m_index] = m_mouseLastViewPosition - event->pos();
+    m_dragTime[m_index] = m_time.elapsed();
+    m_time.start();
+    m_index++;
+
+    emit viewScrolled(SceneCoordinate(m_scenePosition.x(), m_scenePosition.y()), true);
+
+    m_mouseLastScenePosition = mapToScene(event->pos()).toPoint();
+    m_mouseLastViewPosition = event->pos();
+}
+
+void MapView::mousePressEvent(QMouseEvent *event)
+{
+    qDebug() << __PRETTY_FUNCTION__;
+
+    if (m_doubleTapZoomRunning)
+        return;
+
+    m_time.start();
+
+    m_scroller->stop();
+
+    QGraphicsView::mousePressEvent(event);
+
+    m_mouseLastScenePosition = mapToScene(event->pos()).toPoint();
+    m_mouseLastViewPosition = event->pos();
+    m_scenePosition = mapToScene(width() / 2 - 1, height() / 2 - 1).toPoint();
+
+    for (int i = 0; i < VALUES; i++) {
+        m_dragMovement[i] = QPoint();
+        m_dragTime[i] = 0;
+    }
+    m_index = 0;
+}
+
+void MapView::mouseReleaseEvent(QMouseEvent *event)
+{
+    qDebug() << __PRETTY_FUNCTION__;
+
+    if (m_doubleTapZoomRunning)
+        return;
+
+    int elapsed = m_time.elapsed();
+
+    QGraphicsView::mouseReleaseEvent(event);
+
+    // start kinetic scroll only if there isn't too much time elapsed from the last mouse move event
+    if (elapsed <= KINETIC_MAX_TIME_FROM_LAST_MOUSE_EVENT_MS) {
+        QPointF dragViewSpeed;
+        int dragLength = 0;
+        int values = 0;
+        for (int i = 0; i < VALUES; i++) {
+            if (m_dragTime[i] > 0) {
+                dragViewSpeed += m_dragMovement[i] / (m_dragTime[i] / MS_PER_S);
+                dragLength += m_dragMovement[i].manhattanLength();
+                values++;
+            }
+        }
+
+        if (dragLength >= KINETIC_MIN_DRAG_LENGTH_VIEW_PIXELS) {
+            dragViewSpeed /= values;
+            QPointF effectViewDistance = dragViewSpeed * KINETIC_SPEED_TO_DISTANCE_FACTOR;
+
+            // limit the scroll distance in screen pixels
+            qreal biggerDistance = qMax(abs(effectViewDistance.x()), abs(effectViewDistance.y()));
+            if (biggerDistance > m_kineticMaxViewDistance)
+                effectViewDistance /= biggerDistance / m_kineticMaxViewDistance;
+
+            QPointF effectSceneDistance = effectViewDistance
+                                          * (1 << (OSM_MAX_ZOOM_LEVEL - m_zoomLevel));
+
+            m_scroller->setEasingCurve(QEasingCurve::OutCirc);
+            m_scroller->setDuration(KINETIC_SCROLL_TIME_MS);
+            m_scroller->setStartValue(SceneCoordinate(m_scenePosition.x(), m_scenePosition.y()));
+            QPointF endValue = QPointF(m_scenePosition) + effectSceneDistance;
+            m_scroller->setEndValue(SceneCoordinate(endValue.x(), endValue.y()));
+            m_scroller->start();
+        }
     }
+}
 
-//    qDebug() << __PRETTY_FUNCTION__ << "currentScale:" << currentScale()
-//                                    << "m_zoomScaleDelta" << m_zoomScaleDelta
-//                                    << "scaleFactor:" << scaleFactor;
+void MapView::resizeEvent(QResizeEvent *event)
+{
+    qDebug() << __PRETTY_FUNCTION__ << "Resize:" << event->size();
+
+    m_kineticMaxViewDistance = qMax(width(), height()) * KINETIC_MAX_VIEW_DISTANCE_FACTOR;
+
+    emit viewResized(event->size());
+
+    updateCenterShift();
+}
+
+void MapView::setViewScale(qreal viewScale)
+{
+    qDebug() << __PRETTY_FUNCTION__;
 
     QTransform transform;
-    transform.scale(scaleFactor, scaleFactor);
+    transform.scale(viewScale, viewScale);
     setTransform(transform);
 }
 
-qreal MapView::currentScale()
+void MapView::setZoomLevel(int zoomLevel)
 {
-    QTransform currentTransform = transform();
-    return currentTransform.m11();
+    qDebug() << __PRETTY_FUNCTION__;
+
+    m_zoomLevel = zoomLevel;
+
+    if (m_zoomAnimation) {
+        m_zoomAnimation->stop();
+        m_zoomAnimation->setEasingCurve(QEasingCurve::InQuad);
+        m_zoomAnimation->setDuration(ZOOM_TIME_MS);
+        m_zoomAnimation->setStartValue(viewScale());
+        m_zoomAnimation->setEndValue(pow(2, zoomLevel - OSM_MAX_ZOOM_LEVEL));
+
+        m_zoomAnimation->start();
+    }
+
+    updateCenterShift();
 }
 
-void MapView::scrollContentsBy (int dx, int dy)
+void MapView::updateCenterShift()
 {
-//    qDebug() << __PRETTY_FUNCTION__;
+    qDebug() << __PRETTY_FUNCTION__;
 
-    QGraphicsView::scrollContentsBy(dx, dy);
+    int shift = 0;
 
-    QPoint centerInScene = mapToScene(frameRect().center()).toPoint();
-//    qDebug() << __PRETTY_FUNCTION__ << "centerInScene:" << centerInScene;
-    emit viewScrolled(centerInScene);
+    if (m_panelIsOpen)
+        shift = 200;
+
+    m_centerHorizontalShift = QPointF(shift * (1 << (OSM_MAX_ZOOM_LEVEL - m_zoomLevel)), 0);
 }
 
-void MapView::centerToSceneCoordinates(QPoint sceneCoordinate)
+qreal MapView::viewScale()
 {
-//    qDebug() << __PRETTY_FUNCTION__ << "sceneCoordinate" << sceneCoordinate;
-    centerOn(sceneCoordinate);
+    qDebug() << __PRETTY_FUNCTION__;
+
+    return transform().m11();
 }
 
-void MapView::resizeEvent(QResizeEvent *event)
+MapView::~MapView()
 {
-    //qDebug() << "Resize event: " << event->size();
-    emit viewResized(event->size());
+    qDebug() << __PRETTY_FUNCTION__;
+
+    m_scrollAndZoomAnimation->removeAnimation(m_scroller);
+    delete m_scrollAndZoomAnimation;
 }