Merge branch 'master' into map_double_click_zoom
[situare] / src / map / mapview.cpp
1 /*
2    Situare - A location system for Facebook
3    Copyright (C) 2010  Ixonos Plc. Authors:
4
5        Sami Rämö - sami.ramo@ixonos.com
6        Pekka Nissinen - pekka.nissinen@ixonos.com
7
8    Situare is free software; you can redistribute it and/or
9    modify it under the terms of the GNU General Public License
10    version 2 as published by the Free Software Foundation.
11
12    Situare is distributed in the hope that it will be useful,
13    but WITHOUT ANY WARRANTY; without even the implied warranty of
14    MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
15    GNU General Public License for more details.
16
17    You should have received a copy of the GNU General Public License
18    along with Situare; if not, write to the Free Software
19    Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA  02110-1301,
20    USA.
21 */
22
23 #include <cmath>
24
25 #include <QDebug>
26 #include <QMouseEvent>
27 #include <QParallelAnimationGroup>
28
29 #include "mapcommon.h"
30 #include "mapscroller.h"
31
32 #include "mapview.h"
33
34 const qreal MS_PER_S = 1000;
35
36 // const values for tuning the kinetic scroll effect
37 const int KINETIC_MIN_DRAG_LENGTH_VIEW_PIXELS = 30;
38 const int KINETIC_MAX_TIME_FROM_LAST_MOUSE_EVENT_MS = 100;
39 const int KINETIC_SCROLL_TIME_MS = 750;
40 const qreal KINETIC_SPEED_TO_DISTANCE_FACTOR = 0.15 * sqrt(KINETIC_SCROLL_TIME_MS / MS_PER_S);
41
42 MapView::MapView(QWidget *parent)
43     : QGraphicsView(parent)
44 {
45     qDebug() << __PRETTY_FUNCTION__;
46
47     setHorizontalScrollBarPolicy(Qt::ScrollBarAlwaysOff);
48     setVerticalScrollBarPolicy(Qt::ScrollBarAlwaysOff);
49
50     m_zoomAnimation = new QPropertyAnimation(this, "viewScale", this);
51     connect(m_zoomAnimation, SIGNAL(finished()),
52         this, SIGNAL(viewZoomFinished()));
53
54     setOptimizationFlag(QGraphicsView::DontAdjustForAntialiasing);
55
56     m_scroller = &MapScroller::getInstance();
57
58     m_groupAnimation = new QParallelAnimationGroup(this);
59     m_groupAnimation->addAnimation(m_scroller);
60     m_groupAnimation->addAnimation(m_zoomAnimation);
61 }
62
63 void MapView::centerToSceneCoordinates(QPoint sceneCoordinate)
64 {
65     qDebug() << __PRETTY_FUNCTION__ << "sceneCoordinate" << sceneCoordinate;
66
67     centerOn(sceneCoordinate);
68 }
69
70 void MapView::mouseDoubleClickEvent(QMouseEvent *event)
71 {
72     QPoint pressPos = mapToScene(event->pos()).toPoint();
73     QPoint centerPos = mapToScene(width() / 2 - 1, height() / 2 - 1).toPoint();
74     QPoint zoomPoint = centerPos - ((centerPos - pressPos) / 2);
75
76     m_groupAnimation->stop();
77
78     m_scroller->setEasingCurve(QEasingCurve::OutCirc);
79     m_scroller->setDuration(KINETIC_SCROLL_TIME_MS);
80     m_scroller->setStartValue(m_scenePosition);
81     m_scroller->setEndValue(zoomPoint);
82
83     m_zoomAnimation->setEasingCurve(QEasingCurve::OutCirc);
84     m_zoomAnimation->setDuration(KINETIC_SCROLL_TIME_MS);
85     m_zoomAnimation->setStartValue(viewScale());
86     m_zoomAnimation->setEndValue(pow(2, ++m_zoomLevel - MAX_MAP_ZOOM_LEVEL));
87
88
89     m_groupAnimation->start();
90 }
91
92 void MapView::mouseMoveEvent(QMouseEvent *event)
93 {
94     m_scenePosition += m_mouseLastScenePosition - mapToScene(event->pos()).toPoint();
95
96     qWarning() << __PRETTY_FUNCTION__;
97
98     qDebug() << __PRETTY_FUNCTION__ << "m_scenePosition:" << m_scenePosition;
99
100     if (m_index >= VALUES)
101         m_index = 0;
102
103     m_dragMovement[m_index] = m_mouseLastViewPosition - event->pos();
104     m_dragTime[m_index] = m_time.elapsed();
105     m_time.start();
106     m_index++;
107
108     emit viewScrolled(m_scenePosition);
109
110     m_mouseLastScenePosition = mapToScene(event->pos()).toPoint();
111     m_mouseLastViewPosition = event->pos();
112 }
113
114 void MapView::mousePressEvent(QMouseEvent *event)
115 {
116     qWarning() << __PRETTY_FUNCTION__;
117     qDebug() << __PRETTY_FUNCTION__;
118
119     m_time.start();
120
121     m_scroller->stop();
122
123     QGraphicsView::mousePressEvent(event);
124
125     m_mouseLastScenePosition = mapToScene(event->pos()).toPoint();
126     m_mouseLastViewPosition = event->pos();
127     m_scenePosition = mapToScene(width() / 2 - 1, height() / 2 - 1).toPoint();
128
129     for (int i = 0; i < VALUES; i++) {
130         m_dragMovement[i] = QPoint();
131         m_dragTime[i] = 0;
132     }
133     m_index = 0;
134 }
135
136 void MapView::mouseReleaseEvent(QMouseEvent *event)
137 {
138     qDebug() << __PRETTY_FUNCTION__;
139
140     int elapsed = m_time.elapsed();
141
142     QGraphicsView::mouseReleaseEvent(event);
143
144     // start kinetic scroll only if there isn't too much time elapsed from the last mouse move event
145     if (elapsed <= KINETIC_MAX_TIME_FROM_LAST_MOUSE_EVENT_MS) {
146         QPointF dragViewSpeed;
147         int dragLength = 0;
148         int values = 0;
149         for (int i = 0; i < VALUES; i++) {
150             if (m_dragTime[i] > 0) {
151                 dragViewSpeed += m_dragMovement[i] / (m_dragTime[i] / MS_PER_S);
152                 dragLength += m_dragMovement[i].manhattanLength();
153                 values++;
154             }
155         }
156
157         if (dragLength >= KINETIC_MIN_DRAG_LENGTH_VIEW_PIXELS) {
158             dragViewSpeed /= values;
159             QPointF effectViewDistance= dragViewSpeed * KINETIC_SPEED_TO_DISTANCE_FACTOR;
160             QPointF effectSceneDistance = effectViewDistance
161                                           * (1 << (MAX_MAP_ZOOM_LEVEL - m_zoomLevel));
162
163             m_scroller->setEasingCurve(QEasingCurve::OutCirc);
164             m_scroller->setDuration(KINETIC_SCROLL_TIME_MS);
165             m_scroller->setStartValue(m_scenePosition);
166             m_scroller->setEndValue(m_scenePosition + effectSceneDistance.toPoint());
167             m_scroller->start();
168         }
169     }
170 }
171
172 void MapView::resizeEvent(QResizeEvent *event)
173 {
174     qDebug() << __PRETTY_FUNCTION__ << "Resize:" << event->size();
175
176     emit viewResized(event->size());
177 }
178
179 void MapView::setViewScale(qreal viewScale)
180 {
181     qDebug() << __PRETTY_FUNCTION__;
182
183     QTransform transform;
184     transform.scale(viewScale, viewScale);
185     setTransform(transform);
186 }
187
188 void MapView::setZoomLevel(int zoomLevel)
189 {
190     qDebug() << __PRETTY_FUNCTION__;
191
192     m_zoomLevel = zoomLevel;
193
194     if (m_zoomAnimation && (m_zoomAnimation->state() == QAbstractAnimation::Stopped)) {
195         m_zoomAnimation->stop();
196         m_zoomAnimation->setDuration(ZOOM_TIME);
197         m_zoomAnimation->setStartValue(viewScale());
198         m_zoomAnimation->setEndValue(pow(2, zoomLevel - MAX_MAP_ZOOM_LEVEL));
199
200         m_zoomAnimation->start();
201     }
202 }
203
204 qreal MapView::viewScale()
205 {
206     qDebug() << __PRETTY_FUNCTION__;
207
208     return transform().m11();
209 }