]> git.sesse.net Git - kdenlive/blob - src/customruler.cpp
Wheel on project timeline ruler to seek, ctrl+wheel seeks 1 second instead of 1 frame
[kdenlive] / src / customruler.cpp
1 /***************************************************************************
2  *   Copyright (C) 2007 by Jean-Baptiste Mardelle (jb@kdenlive.org)        *
3  *                                                                         *
4  *   This program is free software; you can redistribute it and/or modify  *
5  *   it under the terms of the GNU General Public License as published by  *
6  *   the Free Software Foundation; either version 2 of the License, or     *
7  *   (at your option) any later version.                                   *
8  *                                                                         *
9  *   This program is distributed in the hope that it will be useful,       *
10  *   but WITHOUT ANY WARRANTY; without even the implied warranty of        *
11  *   MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the         *
12  *   GNU General Public License for more details.                          *
13  *                                                                         *
14  *   You should have received a copy of the GNU General Public License     *
15  *   along with this program; if not, write to the                         *
16  *   Free Software Foundation, Inc.,                                       *
17  *   51 Franklin Street, Fifth Floor, Boston, MA  02110-1301  USA          *
18  ***************************************************************************/
19
20 #include <QMouseEvent>
21 #include <QStylePainter>
22
23 #include <KDebug>
24 #include <KGlobalSettings>
25
26 #include "customruler.h"
27
28
29 #define INIT_VALUE 0
30 #define INIT_MIN_VALUE 0
31 #define INIT_MAX_VALUE 100
32 #define INIT_TINY_MARK_DISTANCE 1
33 #define INIT_LITTLE_MARK_DISTANCE 5
34 #define INIT_MIDDLE_MARK_DISTANCE (INIT_LITTLE_MARK_DISTANCE * 2)
35 #define INIT_BIG_MARK_DISTANCE (INIT_LITTLE_MARK_DISTANCE * 10)
36 #define INIT_SHOW_TINY_MARK false
37 #define INIT_SHOW_LITTLE_MARK true
38 #define INIT_SHOW_MEDIUM_MARK true
39 #define INIT_SHOW_BIG_MARK true
40 #define INIT_SHOW_END_MARK true
41 #define INIT_SHOW_POINTER true
42 #define INIT_SHOW_END_LABEL true
43
44 #define INIT_PIXEL_PER_MARK (double)10.0 /* distance between 2 base marks in pixel */
45 #define INIT_OFFSET (-20)
46 #define INIT_LENGTH_FIX true
47 #define INIT_END_OFFSET 0
48
49 #define FIX_WIDTH 24 /* widget width in pixel */
50 #define LINE_END (FIX_WIDTH - 3)
51 #define END_MARK_LENGTH (FIX_WIDTH - 8)
52 #define END_MARK_X2 LINE_END
53 #define END_MARK_X1 (END_MARK_X2 - END_MARK_LENGTH)
54 #define BIG_MARK_LENGTH (END_MARK_LENGTH*3/4)
55 #define BIG_MARK_X2 LINE_END
56 #define BIG_MARK_X1 (BIG_MARK_X2 - BIG_MARK_LENGTH)
57 #define MIDDLE_MARK_LENGTH (END_MARK_LENGTH/2)
58 #define MIDDLE_MARK_X2 LINE_END
59 #define MIDDLE_MARK_X1 (MIDDLE_MARK_X2 - MIDDLE_MARK_LENGTH)
60 #define LITTLE_MARK_LENGTH (MIDDLE_MARK_LENGTH/2)
61 #define LITTLE_MARK_X2 LINE_END
62 #define LITTLE_MARK_X1 (LITTLE_MARK_X2 - LITTLE_MARK_LENGTH)
63 #define BASE_MARK_LENGTH (LITTLE_MARK_LENGTH/2)
64 #define BASE_MARK_X2 LINE_END
65 #define BASE_MARK_X1 (BASE_MARK_X2 - 3) //BASE_MARK_LENGTH
66
67 #define LABEL_SIZE 9
68 #define END_LABEL_X 4
69 #define END_LABEL_Y (END_LABEL_X + LABEL_SIZE - 2)
70
71 #include "definitions.h"
72
73 const int CustomRuler::comboScale[] = { 1, 2, 5, 10, 25, 50, 125, 250, 500, 725, 1500, 3000, 6000, 12000};
74
75 CustomRuler::CustomRuler(Timecode tc, CustomTrackView *parent)
76         : KRuler(parent), m_timecode(tc), m_view(parent), m_duration(0) {
77     setFont(KGlobalSettings::toolBarFont());
78     slotNewOffset(0);
79     setRulerMetricStyle(KRuler::Pixel);
80     setLength(1024);
81     setMaximum(1024);
82     setPixelPerMark(3);
83     setLittleMarkDistance(FRAME_SIZE);
84     setMediumMarkDistance(FRAME_SIZE * m_timecode.fps());
85     setBigMarkDistance(FRAME_SIZE * m_timecode.fps() * 60);
86     m_zoneStart = 2 * m_timecode.fps();
87     m_zoneEnd = 10 * m_timecode.fps();
88 }
89
90 // virtual
91 void CustomRuler::mousePressEvent(QMouseEvent * event) {
92     m_view->activateMonitor();
93     int pos = (int)((event->x() + offset()) / pixelPerMark() / FRAME_SIZE);
94     m_moveCursor = RULER_CURSOR;
95     if (event->y() > 10) {
96         if (abs(pos - m_zoneStart) < 4) m_moveCursor = RULER_START;
97         else if (abs(pos - (m_zoneStart + (m_zoneEnd - m_zoneStart) / 2)) < 4) m_moveCursor = RULER_MIDDLE;
98         else if (abs(pos - m_zoneEnd) < 4) m_moveCursor = RULER_END;
99     }
100     if (m_moveCursor == RULER_CURSOR)
101         m_view->setCursorPos(pos);
102 }
103
104 // virtual
105 void CustomRuler::mouseMoveEvent(QMouseEvent * event) {
106     int pos = (int)((event->x() + offset()) / pixelPerMark() / FRAME_SIZE);
107     if (m_moveCursor == RULER_CURSOR) {
108         m_view->setCursorPos(pos);
109         return;
110     } else if (m_moveCursor == RULER_START) m_zoneStart = pos;
111     else if (m_moveCursor == RULER_END) m_zoneEnd = pos;
112     else if (m_moveCursor == RULER_MIDDLE) {
113         int move = pos - (m_zoneStart + (m_zoneEnd - m_zoneStart) / 2);
114         m_zoneStart += move;
115         m_zoneEnd += move;
116     }
117     update();
118 }
119
120
121 // virtual
122 void CustomRuler::wheelEvent(QWheelEvent * e) {
123     int delta = 1;
124     if (e->modifiers() == Qt::ControlModifier) delta = m_timecode.fps();
125     if (e->delta() < 0) delta = 0 - delta;
126     m_view->moveCursorPos(delta);
127 }
128
129 int CustomRuler::inPoint() const {
130     return m_zoneStart;
131 }
132
133 int CustomRuler::outPoint() const {
134     return m_zoneEnd;
135 }
136
137 void CustomRuler::slotMoveRuler(int newPos) {
138     KRuler::slotNewOffset(newPos);
139 }
140
141 void CustomRuler::slotCursorMoved(int oldpos, int newpos) {
142     //TODO: optimize (redraw only around cursor positions
143     update();
144 }
145
146 void CustomRuler::setPixelPerMark(double rate) {
147     int scale = comboScale[(int) rate];
148     KRuler::setPixelPerMark(1.0 / scale);
149     double fend = pixelPerMark() * littleMarkDistance();
150     switch ((int) rate) {
151     case 0:
152         m_textSpacing = fend;
153         break;
154     case 1:
155         m_textSpacing = fend * 5;
156         break;
157     case 2:
158     case 3:
159     case 4:
160         m_textSpacing = fend * m_timecode.fps();
161         break;
162     case 5:
163         m_textSpacing = fend * m_timecode.fps() * 5;
164         break;
165     case 6:
166         m_textSpacing = fend * m_timecode.fps() * 10;
167         break;
168     case 7:
169         m_textSpacing = fend * m_timecode.fps() * 30;
170         break;
171     case 8:
172     case 9:
173     case 10:
174         m_textSpacing = fend * m_timecode.fps() * 60;
175         break;
176     case 11:
177     case 12:
178         m_textSpacing = fend * m_timecode.fps() * 300;
179         break;
180     case 13:
181         m_textSpacing = fend * m_timecode.fps() * 600;
182         break;
183     }
184 }
185
186 void CustomRuler::setDuration(int d) {
187     m_duration = d;
188     update();
189 }
190
191 // virtual
192 void CustomRuler::paintEvent(QPaintEvent *e) {
193     //  debug ("KRuler::drawContents, %s",(horizontal==dir)?"horizontal":"vertical");
194
195     QStylePainter p(this);
196     p.setClipRect(e->rect());
197
198     //p.fillRect(e->rect(), QBrush(QColor(200, 200, 200)));
199     //kDebug()<<"RULER ZONE: "<<m_zoneStart<<", OFF: "<<offset()<<", END: "<<m_zoneEnd<<", FACTOR: "<<pixelPerMark() * FRAME_SIZE;
200     int projectEnd = (int)(m_duration * pixelPerMark() * FRAME_SIZE);
201     p.fillRect(QRect(- offset(), e->rect().y(), projectEnd, e->rect().height()), QBrush(QColor(245, 245, 245)));
202
203
204     int zoneStart = (int)(m_zoneStart * pixelPerMark() * FRAME_SIZE);
205     int zoneEnd = (int)(m_zoneEnd * pixelPerMark() * FRAME_SIZE);
206
207     p.fillRect(QRect(zoneStart - offset(), e->rect().y() + e->rect().height() / 2, zoneEnd - zoneStart, e->rect().height() / 2), QBrush(QColor(133, 255, 143)));
208
209     int value  = m_view->cursorPos() - offset() + 4;
210     int minval = minimum();
211     int maxval = maximum() + offset() - endOffset();
212
213     //ioffsetval = value-offset;
214     //    pixelpm = (int)ppm;
215     //    left  = clip.left(),
216     //    right = clip.right();
217     double f, fend,
218     offsetmin = (double)(minval - offset()),
219                 offsetmax = (double)(maxval - offset()),
220                             fontOffset = (((double)minval) > offsetmin) ? (double)minval : offsetmin;
221     QRect bg = QRect((int)offsetmin, 0, (int)offsetmax, height());
222
223     QPalette palette;
224     //p.fillRect(bg, palette.light());
225     // draw labels
226     p.setPen(palette.dark().color());
227     // draw littlemarklabel
228
229     // draw mediummarklabel
230
231     // draw bigmarklabel
232
233     // draw endlabel
234     /*if (d->showEndL) {
235       if (d->dir == Qt::Horizontal) {
236         p.translate( fontOffset, 0 );
237         p.drawText( END_LABEL_X, END_LABEL_Y, d->endlabel );
238       }*/
239
240     // draw the tiny marks
241     //if (showTinyMarks())
242     /*{
243       fend =   pixelPerMark()*tinyMarkDistance();
244       if (fend > 5) for ( f=offsetmin; f<offsetmax; f+=fend ) {
245           p.drawLine((int)f, BASE_MARK_X1, (int)f, BASE_MARK_X2);
246       }
247     }*/
248
249     for (f = offsetmin; f < offsetmax; f += m_textSpacing) {
250         QString lab = m_timecode.getTimecodeFromFrames((int)((f - offsetmin) / pixelPerMark() / FRAME_SIZE + 0.5));
251         p.drawText((int)f + 2, LABEL_SIZE, lab);
252     }
253
254     if (showLittleMarks()) {
255         // draw the little marks
256         fend = pixelPerMark() * littleMarkDistance();
257         if (fend > 5) for (f = offsetmin; f < offsetmax; f += fend)
258                 p.drawLine((int)f, LITTLE_MARK_X1, (int)f, LITTLE_MARK_X2);
259     }
260     if (showMediumMarks()) {
261         // draw medium marks
262         fend = pixelPerMark() * mediumMarkDistance();
263         if (fend > 5) for (f = offsetmin; f < offsetmax; f += fend)
264                 p.drawLine((int)f, MIDDLE_MARK_X1, (int)f, MIDDLE_MARK_X2);
265     }
266     if (showBigMarks()) {
267         // draw big marks
268         fend = pixelPerMark() * bigMarkDistance();
269         if (fend > 5) for (f = offsetmin; f < offsetmax; f += fend)
270                 p.drawLine((int)f, BIG_MARK_X1, (int)f, BIG_MARK_X2);
271     }
272     /*   if (d->showem) {
273          // draw end marks
274          if (d->dir == Qt::Horizontal) {
275            p.drawLine(minval-d->offset, END_MARK_X1, minval-d->offset, END_MARK_X2);
276            p.drawLine(maxval-d->offset, END_MARK_X1, maxval-d->offset, END_MARK_X2);
277          }
278          else {
279            p.drawLine(END_MARK_X1, minval-d->offset, END_MARK_X2, minval-d->offset);
280            p.drawLine(END_MARK_X1, maxval-d->offset, END_MARK_X2, maxval-d->offset);
281          }
282        }*/
283
284
285     // draw zone cursors
286     int off = offset();
287     if (zoneStart > 0) {
288         QPolygon pa(4);
289         pa.setPoints(4, zoneStart - off + 3, 9, zoneStart - off, 9, zoneStart - off, 18, zoneStart - off + 3, 18);
290         p.drawPolyline(pa);
291     }
292
293     if (zoneEnd > 0) {
294         QRect rec(zoneStart - off + (zoneEnd - zoneStart) / 2 - 4, 9, 8, 9);
295         p.fillRect(rec, QColor(255, 255, 255, 150));
296         p.drawRect(rec);
297
298         QPolygon pa(4);
299         pa.setPoints(4, zoneEnd - off - 3, 9, zoneEnd - off, 9, zoneEnd - off, 18, zoneEnd - off - 3, 18);
300         p.drawPolyline(pa);
301     }
302
303     // draw pointer
304     if (showPointer() && value > 0) {
305         QPolygon pa(3);
306         pa.setPoints(3, value - 6, 7, value + 6, 7, value, 16);
307         p.setBrush(QBrush(Qt::yellow));
308         p.drawPolygon(pa);
309     }
310
311
312
313 }
314
315 #include "customruler.moc"