Chris@173
|
1 /* -*- c-basic-offset: 4 indent-tabs-mode: nil -*- vi:set ts=8 sts=4 sw=4: */
|
Chris@173
|
2
|
Chris@173
|
3 /*
|
Chris@173
|
4 Sonic Visualiser
|
Chris@173
|
5 An audio file viewer and annotation editor.
|
Chris@173
|
6 Centre for Digital Music, Queen Mary, University of London.
|
Chris@182
|
7 This file copyright 2006 Chris Cannam and QMUL.
|
Chris@173
|
8
|
Chris@173
|
9 This program is free software; you can redistribute it and/or
|
Chris@173
|
10 modify it under the terms of the GNU General Public License as
|
Chris@173
|
11 published by the Free Software Foundation; either version 2 of the
|
Chris@173
|
12 License, or (at your option) any later version. See the file
|
Chris@173
|
13 COPYING included with this distribution for more information.
|
Chris@173
|
14 */
|
Chris@173
|
15
|
Chris@173
|
16 #include "Overview.h"
|
Chris@173
|
17 #include "layer/Layer.h"
|
Chris@173
|
18 #include "data/model/Model.h"
|
Chris@173
|
19 #include "base/ZoomConstraint.h"
|
Chris@173
|
20
|
Chris@173
|
21 #include <QPaintEvent>
|
Chris@173
|
22 #include <QPainter>
|
Chris@173
|
23 #include <iostream>
|
Chris@173
|
24
|
Chris@643
|
25 //#define DEBUG_OVERVIEW 1
|
Chris@642
|
26
|
Chris@682
|
27
|
Chris@173
|
28 Overview::Overview(QWidget *w) :
|
Chris@173
|
29 View(w, false),
|
Chris@854
|
30 m_clickedInRange(false),
|
Chris@854
|
31 m_dragCentreFrame(0)
|
Chris@173
|
32 {
|
Chris@173
|
33 setObjectName(tr("Overview"));
|
Chris@173
|
34 m_followPan = false;
|
Chris@173
|
35 m_followZoom = false;
|
Chris@211
|
36 setPlaybackFollow(PlaybackIgnore);
|
Chris@274
|
37 m_modelTestTime.start();
|
Chris@965
|
38
|
Chris@965
|
39 bool light = hasLightBackground();
|
Chris@965
|
40 if (light) m_boxColour = Qt::darkGray;
|
Chris@965
|
41 else m_boxColour = Qt::lightGray;
|
Chris@173
|
42 }
|
Chris@173
|
43
|
Chris@173
|
44 void
|
Chris@908
|
45 Overview::modelChangedWithin(sv_frame_t startFrame, sv_frame_t endFrame)
|
Chris@173
|
46 {
|
Chris@1325
|
47 using namespace std::rel_ops;
|
Chris@1325
|
48
|
Chris@253
|
49 bool zoomChanged = false;
|
Chris@253
|
50
|
Chris@908
|
51 sv_frame_t frameCount = getModelsEndFrame() - getModelsStartFrame();
|
Chris@1183
|
52 ZoomLevel zoomLevel { ZoomLevel::FramesPerPixel, int(frameCount / width()) };
|
Chris@1183
|
53 if (zoomLevel.level < 1) zoomLevel.level = 1;
|
Chris@1325
|
54 zoomLevel = getZoomConstraintLevel(zoomLevel, ZoomConstraint::RoundUp);
|
Chris@253
|
55 if (zoomLevel != m_zoomLevel) {
|
Chris@253
|
56 zoomChanged = true;
|
Chris@253
|
57 }
|
Chris@253
|
58
|
Chris@253
|
59 if (!zoomChanged) {
|
Chris@274
|
60 if (m_modelTestTime.elapsed() < 1000) {
|
Chris@835
|
61 for (LayerList::const_iterator i = m_layerStack.begin();
|
Chris@835
|
62 i != m_layerStack.end(); ++i) {
|
Chris@274
|
63 if ((*i)->getModel() &&
|
Chris@389
|
64 (!(*i)->getModel()->isOK() ||
|
Chris@389
|
65 !(*i)->getModel()->isReady())) {
|
Chris@274
|
66 return;
|
Chris@274
|
67 }
|
Chris@253
|
68 }
|
Chris@274
|
69 } else {
|
Chris@274
|
70 m_modelTestTime.restart();
|
Chris@253
|
71 }
|
Chris@253
|
72 }
|
Chris@253
|
73
|
Chris@806
|
74 View::modelChangedWithin(startFrame, endFrame);
|
Chris@173
|
75 }
|
Chris@173
|
76
|
Chris@173
|
77 void
|
Chris@173
|
78 Overview::modelReplaced()
|
Chris@173
|
79 {
|
Chris@339
|
80 m_playPointerFrame = getAlignedPlaybackFrame();
|
Chris@173
|
81 View::modelReplaced();
|
Chris@173
|
82 }
|
Chris@173
|
83
|
Chris@173
|
84 void
|
Chris@211
|
85 Overview::registerView(View *view)
|
Chris@173
|
86 {
|
Chris@211
|
87 m_views.insert(view);
|
Chris@173
|
88 update();
|
Chris@173
|
89 }
|
Chris@173
|
90
|
Chris@173
|
91 void
|
Chris@211
|
92 Overview::unregisterView(View *view)
|
Chris@173
|
93 {
|
Chris@211
|
94 m_views.erase(view);
|
Chris@173
|
95 update();
|
Chris@173
|
96 }
|
Chris@173
|
97
|
Chris@173
|
98 void
|
Chris@908
|
99 Overview::globalCentreFrameChanged(sv_frame_t
|
Chris@806
|
100 #ifdef DEBUG_OVERVIEW
|
Chris@806
|
101 f
|
Chris@806
|
102 #endif
|
Chris@806
|
103 )
|
Chris@173
|
104 {
|
Chris@642
|
105 #ifdef DEBUG_OVERVIEW
|
Chris@682
|
106 cerr << "Overview::globalCentreFrameChanged: " << f << endl;
|
Chris@642
|
107 #endif
|
Chris@211
|
108 update();
|
Chris@211
|
109 }
|
Chris@173
|
110
|
Chris@211
|
111 void
|
Chris@908
|
112 Overview::viewCentreFrameChanged(View *v, sv_frame_t
|
Chris@806
|
113 #ifdef DEBUG_OVERVIEW
|
Chris@806
|
114 f
|
Chris@806
|
115 #endif
|
Chris@806
|
116 )
|
Chris@211
|
117 {
|
Chris@642
|
118 #ifdef DEBUG_OVERVIEW
|
Chris@682
|
119 cerr << "Overview[" << this << "]::viewCentreFrameChanged(" << v << "): " << f << endl;
|
Chris@642
|
120 #endif
|
Chris@211
|
121 if (m_views.find(v) != m_views.end()) {
|
Chris@1266
|
122 update();
|
Chris@173
|
123 }
|
Chris@211
|
124 }
|
Chris@173
|
125
|
Chris@173
|
126 void
|
Chris@806
|
127 Overview::viewZoomLevelChanged(View *v, int, bool)
|
Chris@173
|
128 {
|
Chris@222
|
129 if (v == this) return;
|
Chris@211
|
130 if (m_views.find(v) != m_views.end()) {
|
Chris@1266
|
131 update();
|
Chris@173
|
132 }
|
Chris@173
|
133 }
|
Chris@173
|
134
|
Chris@173
|
135 void
|
Chris@908
|
136 Overview::viewManagerPlaybackFrameChanged(sv_frame_t f)
|
Chris@173
|
137 {
|
Chris@642
|
138 #ifdef DEBUG_OVERVIEW
|
Chris@682
|
139 cerr << "Overview[" << this << "]::viewManagerPlaybackFrameChanged(" << f << "): " << f << endl;
|
Chris@642
|
140 #endif
|
Chris@642
|
141
|
Chris@173
|
142 bool changed = false;
|
Chris@173
|
143
|
Chris@339
|
144 f = getAlignedPlaybackFrame();
|
Chris@339
|
145
|
Chris@173
|
146 if (getXForFrame(m_playPointerFrame) != getXForFrame(f)) changed = true;
|
Chris@173
|
147 m_playPointerFrame = f;
|
Chris@173
|
148
|
Chris@173
|
149 if (changed) update();
|
Chris@173
|
150 }
|
Chris@173
|
151
|
Chris@871
|
152 QColor
|
Chris@871
|
153 Overview::getFillWithin() const
|
Chris@871
|
154 {
|
Chris@871
|
155 return Qt::transparent;
|
Chris@871
|
156 }
|
Chris@871
|
157
|
Chris@871
|
158 QColor
|
Chris@871
|
159 Overview::getFillWithout() const
|
Chris@871
|
160 {
|
Chris@871
|
161 QColor c = palette().window().color();
|
Chris@871
|
162 c.setAlpha(100);
|
Chris@871
|
163 return c;
|
Chris@871
|
164 }
|
Chris@871
|
165
|
Chris@173
|
166 void
|
Chris@965
|
167 Overview::setBoxColour(QColor c)
|
Chris@965
|
168 {
|
Chris@965
|
169 m_boxColour = c;
|
Chris@965
|
170 }
|
Chris@965
|
171
|
Chris@965
|
172 void
|
Chris@173
|
173 Overview::paintEvent(QPaintEvent *e)
|
Chris@173
|
174 {
|
Chris@1325
|
175 using namespace std::rel_ops;
|
Chris@1325
|
176
|
Chris@173
|
177 // Recalculate zoom in case the size of the widget has changed.
|
Chris@173
|
178
|
Chris@642
|
179 #ifdef DEBUG_OVERVIEW
|
Chris@682
|
180 cerr << "Overview::paintEvent: width is " << width() << ", centre frame " << m_centreFrame << endl;
|
Chris@642
|
181 #endif
|
Chris@214
|
182
|
Chris@908
|
183 sv_frame_t startFrame = getModelsStartFrame();
|
Chris@908
|
184 sv_frame_t frameCount = getModelsEndFrame() - getModelsStartFrame();
|
Chris@1325
|
185 ZoomLevel zoomLevel { ZoomLevel::FramesPerPixel, int(frameCount / width()) };
|
Chris@1325
|
186 if (zoomLevel.level < 1) zoomLevel.level = 1;
|
Chris@1325
|
187 zoomLevel = getZoomConstraintLevel(zoomLevel, ZoomConstraint::RoundUp);
|
Chris@173
|
188 if (zoomLevel != m_zoomLevel) {
|
Chris@1266
|
189 m_zoomLevel = zoomLevel;
|
Chris@1266
|
190 emit zoomLevelChanged(m_zoomLevel, m_followZoom);
|
Chris@173
|
191 }
|
Chris@253
|
192
|
Chris@908
|
193 sv_frame_t centreFrame = startFrame + m_zoomLevel * (width() / 2);
|
Chris@173
|
194 if (centreFrame > (startFrame + getModelsEndFrame())/2) {
|
Chris@1266
|
195 centreFrame = (startFrame + getModelsEndFrame())/2;
|
Chris@173
|
196 }
|
Chris@173
|
197 if (centreFrame != m_centreFrame) {
|
Chris@642
|
198 #ifdef DEBUG_OVERVIEW
|
Chris@682
|
199 cerr << "Overview::paintEvent: Centre frame changed from "
|
Chris@642
|
200 << m_centreFrame << " to " << centreFrame << " and thus start frame from " << getStartFrame();
|
Chris@642
|
201 #endif
|
Chris@1266
|
202 m_centreFrame = centreFrame;
|
Chris@642
|
203 #ifdef DEBUG_OVERVIEW
|
Chris@682
|
204 cerr << " to " << getStartFrame() << endl;
|
Chris@642
|
205 #endif
|
Chris@1266
|
206 emit centreFrameChanged(m_centreFrame, false, PlaybackIgnore);
|
Chris@173
|
207 }
|
Chris@173
|
208
|
Chris@173
|
209 View::paintEvent(e);
|
Chris@173
|
210
|
Chris@173
|
211 QPainter paint;
|
Chris@173
|
212 paint.begin(this);
|
Chris@871
|
213 paint.setClipRegion(e->region());
|
Chris@871
|
214 paint.setRenderHints(QPainter::Antialiasing);
|
Chris@871
|
215
|
Chris@871
|
216 // We paint a rounded rect for each distinct set of view extents,
|
Chris@871
|
217 // and we colour in the inside and outside of the rect that
|
Chris@871
|
218 // corresponds to the current view. (One small caveat -- we don't
|
Chris@871
|
219 // know which rect that is yet. We'll have to figure it out
|
Chris@871
|
220 // somehow...)
|
Chris@173
|
221
|
Chris@871
|
222 std::set<std::pair<int, int> > extents;
|
Chris@871
|
223 std::vector<QRect> rects;
|
Chris@871
|
224 QRect primary;
|
Chris@173
|
225
|
Chris@173
|
226 int y = 0;
|
Chris@173
|
227
|
Chris@211
|
228 for (ViewSet::iterator i = m_views.begin(); i != m_views.end(); ++i) {
|
Chris@1266
|
229 if (!*i) continue;
|
Chris@173
|
230
|
Chris@1266
|
231 View *w = (View *)*i;
|
Chris@173
|
232
|
Chris@1266
|
233 sv_frame_t f0 = w->getFrameForX(0);
|
Chris@1266
|
234 sv_frame_t f1 = w->getFrameForX(w->width());
|
Chris@173
|
235
|
Chris@339
|
236 if (f0 >= 0) {
|
Chris@908
|
237 sv_frame_t rf0 = w->alignToReference(f0);
|
Chris@339
|
238 f0 = alignFromReference(rf0);
|
Chris@339
|
239 }
|
Chris@339
|
240 if (f1 >= 0) {
|
Chris@908
|
241 sv_frame_t rf1 = w->alignToReference(f1);
|
Chris@339
|
242 f1 = alignFromReference(rf1);
|
Chris@339
|
243 }
|
Chris@339
|
244
|
Chris@1266
|
245 int x0 = getXForFrame(f0);
|
Chris@1266
|
246 int x1 = getXForFrame(f1);
|
Chris@173
|
247
|
Chris@1266
|
248 if (x1 <= x0) x1 = x0 + 1;
|
Chris@871
|
249
|
Chris@871
|
250 std::pair<int, int> extent(x0, x1);
|
Chris@871
|
251
|
Chris@871
|
252 if (extents.find(extent) == extents.end()) {
|
Chris@871
|
253
|
Chris@1266
|
254 y += height() / 10 + 1;
|
Chris@871
|
255 extents.insert(extent);
|
Chris@173
|
256
|
Chris@871
|
257 QRect vr(x0, y, x1 - x0, height() - 2 * y);
|
Chris@871
|
258 rects.push_back(vr);
|
Chris@871
|
259 primary = vr; //!!! for now
|
Chris@871
|
260 }
|
Chris@871
|
261 }
|
Chris@871
|
262
|
Chris@871
|
263 QPainterPath without;
|
matthiasm@935
|
264 without.addRoundedRect(primary, 4, 4);
|
Chris@871
|
265 without.addRect(rect());
|
Chris@871
|
266 paint.setPen(Qt::NoPen);
|
Chris@871
|
267 paint.setBrush(getFillWithout());
|
Chris@871
|
268 paint.drawPath(without);
|
Chris@871
|
269
|
Chris@871
|
270 paint.setBrush(getFillWithin());
|
matthiasm@935
|
271 paint.drawRoundedRect(primary, 4, 4);
|
Chris@871
|
272
|
Chris@871
|
273 foreach (QRect vr, rects) {
|
Chris@871
|
274 paint.setBrush(Qt::NoBrush);
|
Chris@965
|
275 paint.setPen(QPen(m_boxColour, 2));
|
matthiasm@935
|
276 paint.drawRoundedRect(vr, 4, 4);
|
Chris@173
|
277 }
|
Chris@173
|
278
|
Chris@173
|
279 paint.end();
|
Chris@173
|
280 }
|
Chris@173
|
281
|
Chris@173
|
282 void
|
Chris@173
|
283 Overview::mousePressEvent(QMouseEvent *e)
|
Chris@173
|
284 {
|
Chris@173
|
285 m_clickPos = e->pos();
|
Chris@908
|
286 sv_frame_t clickFrame = getFrameForX(m_clickPos.x());
|
Chris@339
|
287 if (clickFrame > 0) m_dragCentreFrame = clickFrame;
|
Chris@339
|
288 else m_dragCentreFrame = 0;
|
Chris@339
|
289 m_clickedInRange = true;
|
Chris@339
|
290
|
Chris@211
|
291 for (ViewSet::iterator i = m_views.begin(); i != m_views.end(); ++i) {
|
Chris@1266
|
292 if (*i && (*i)->getAligningModel() == getAligningModel()) {
|
Chris@339
|
293 m_dragCentreFrame = (*i)->getCentreFrame();
|
Chris@339
|
294 break;
|
Chris@339
|
295 }
|
Chris@173
|
296 }
|
Chris@173
|
297 }
|
Chris@173
|
298
|
Chris@173
|
299 void
|
Chris@173
|
300 Overview::mouseReleaseEvent(QMouseEvent *e)
|
Chris@173
|
301 {
|
Chris@173
|
302 if (m_clickedInRange) {
|
Chris@1266
|
303 mouseMoveEvent(e);
|
Chris@173
|
304 }
|
Chris@173
|
305 m_clickedInRange = false;
|
Chris@173
|
306 }
|
Chris@173
|
307
|
Chris@173
|
308 void
|
Chris@173
|
309 Overview::mouseMoveEvent(QMouseEvent *e)
|
Chris@173
|
310 {
|
Chris@173
|
311 if (!m_clickedInRange) return;
|
Chris@173
|
312
|
Chris@806
|
313 int xoff = int(e->x()) - int(m_clickPos.x());
|
Chris@908
|
314 sv_frame_t frameOff = xoff * m_zoomLevel;
|
Chris@173
|
315
|
Chris@908
|
316 sv_frame_t newCentreFrame = m_dragCentreFrame;
|
Chris@173
|
317 if (frameOff > 0) {
|
Chris@1266
|
318 newCentreFrame += frameOff;
|
Chris@908
|
319 } else if (newCentreFrame >= -frameOff) {
|
Chris@1266
|
320 newCentreFrame += frameOff;
|
Chris@173
|
321 } else {
|
Chris@1266
|
322 newCentreFrame = 0;
|
Chris@173
|
323 }
|
Chris@173
|
324
|
Chris@173
|
325 if (newCentreFrame >= getModelsEndFrame()) {
|
Chris@1266
|
326 newCentreFrame = getModelsEndFrame();
|
Chris@1266
|
327 if (newCentreFrame > 0) --newCentreFrame;
|
Chris@173
|
328 }
|
Chris@173
|
329
|
Chris@173
|
330 if (std::max(m_centreFrame, newCentreFrame) -
|
Chris@1266
|
331 std::min(m_centreFrame, newCentreFrame) > m_zoomLevel) {
|
Chris@908
|
332 sv_frame_t rf = alignToReference(newCentreFrame);
|
Chris@643
|
333 #ifdef DEBUG_OVERVIEW
|
Chris@682
|
334 cerr << "Overview::mouseMoveEvent: x " << e->x() << " and click x " << m_clickPos.x() << " -> frame " << newCentreFrame << " -> rf " << rf << endl;
|
Chris@643
|
335 #endif
|
Chris@817
|
336 if (m_followPlay == PlaybackScrollContinuous ||
|
Chris@817
|
337 m_followPlay == PlaybackScrollPageWithCentre) {
|
Chris@817
|
338 emit centreFrameChanged(rf, true, PlaybackScrollContinuous);
|
Chris@817
|
339 } else {
|
Chris@817
|
340 emit centreFrameChanged(rf, true, PlaybackIgnore);
|
Chris@817
|
341 }
|
Chris@173
|
342 }
|
Chris@173
|
343 }
|
Chris@173
|
344
|
Chris@189
|
345 void
|
Chris@189
|
346 Overview::mouseDoubleClickEvent(QMouseEvent *e)
|
Chris@189
|
347 {
|
Chris@908
|
348 sv_frame_t frame = getFrameForX(e->x());
|
Chris@908
|
349 sv_frame_t rf = 0;
|
Chris@339
|
350 if (frame > 0) rf = alignToReference(frame);
|
Chris@643
|
351 #ifdef DEBUG_OVERVIEW
|
Chris@682
|
352 cerr << "Overview::mouseDoubleClickEvent: frame " << frame << " -> rf " << rf << endl;
|
Chris@643
|
353 #endif
|
Chris@643
|
354 m_clickedInRange = false; // we're not starting a drag with the second click
|
Chris@339
|
355 emit centreFrameChanged(rf, true, PlaybackScrollContinuous);
|
Chris@189
|
356 }
|
Chris@173
|
357
|
Chris@189
|
358 void
|
Chris@189
|
359 Overview::enterEvent(QEvent *)
|
Chris@189
|
360 {
|
Chris@189
|
361 emit contextHelpChanged(tr("Click and drag to navigate; double-click to jump"));
|
Chris@189
|
362 }
|
Chris@189
|
363
|
Chris@189
|
364 void
|
Chris@189
|
365 Overview::leaveEvent(QEvent *)
|
Chris@189
|
366 {
|
Chris@189
|
367 emit contextHelpChanged("");
|
Chris@189
|
368 }
|
Chris@189
|
369
|
Chris@189
|
370
|