]> git.sesse.net Git - kdenlive/blob - src/customruler.cpp
Improved timeline ruler
[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 int CustomRuler::inPoint() const {
121     return m_zoneStart;
122 }
123
124 int CustomRuler::outPoint() const {
125     return m_zoneEnd;
126 }
127
128 void CustomRuler::slotMoveRuler(int newPos) {
129     KRuler::slotNewOffset(newPos);
130 }
131
132 void CustomRuler::slotCursorMoved(int oldpos, int newpos) {
133     //TODO: optimize (redraw only around cursor positions
134     update();
135 }
136
137 void CustomRuler::setPixelPerMark(double rate) {
138     int scale = comboScale[(int) rate];
139     KRuler::setPixelPerMark(1.0 / scale);
140     double fend = pixelPerMark() * littleMarkDistance();
141     switch ((int) rate) {
142     case 0:
143         m_textSpacing = fend;
144         break;
145     case 1:
146         m_textSpacing = fend * 5;
147         break;
148     case 2:
149     case 3:
150     case 4:
151         m_textSpacing = fend * m_timecode.fps();
152         break;
153     case 5:
154         m_textSpacing = fend * m_timecode.fps() * 5;
155         break;
156     case 6:
157         m_textSpacing = fend * m_timecode.fps() * 10;
158         break;
159     case 7:
160         m_textSpacing = fend * m_timecode.fps() * 30;
161         break;
162     case 8:
163     case 9:
164     case 10:
165         m_textSpacing = fend * m_timecode.fps() * 60;
166         break;
167     case 11:
168     case 12:
169         m_textSpacing = fend * m_timecode.fps() * 300;
170         break;
171     case 13:
172         m_textSpacing = fend * m_timecode.fps() * 600;
173         break;
174     }
175 }
176
177 void CustomRuler::setDuration(int d) {
178     m_duration = d;
179     update();
180 }
181
182 // virtual
183 void CustomRuler::paintEvent(QPaintEvent *e) {
184     //  debug ("KRuler::drawContents, %s",(horizontal==dir)?"horizontal":"vertical");
185
186     QStylePainter p(this);
187     p.setClipRect(e->rect());
188
189     //p.fillRect(e->rect(), QBrush(QColor(200, 200, 200)));
190     //kDebug()<<"RULER ZONE: "<<m_zoneStart<<", OFF: "<<offset()<<", END: "<<m_zoneEnd<<", FACTOR: "<<pixelPerMark() * FRAME_SIZE;
191     int projectEnd = (int)(m_duration * pixelPerMark() * FRAME_SIZE);
192     p.fillRect(QRect(- offset(), e->rect().y(), projectEnd, e->rect().height()), QBrush(QColor(245, 245, 245)));
193
194
195     int zoneStart = (int)(m_zoneStart * pixelPerMark() * FRAME_SIZE);
196     int zoneEnd = (int)(m_zoneEnd * pixelPerMark() * FRAME_SIZE);
197
198     p.fillRect(QRect(zoneStart - offset(), e->rect().y() + e->rect().height() / 2, zoneEnd - zoneStart, e->rect().height() / 2), QBrush(QColor(133, 255, 143)));
199
200     int value  = m_view->cursorPos() - offset() + 4;
201     int minval = minimum();
202     int maxval = maximum() + offset() - endOffset();
203
204     //ioffsetval = value-offset;
205     //    pixelpm = (int)ppm;
206     //    left  = clip.left(),
207     //    right = clip.right();
208     double f, fend,
209     offsetmin = (double)(minval - offset()),
210                 offsetmax = (double)(maxval - offset()),
211                             fontOffset = (((double)minval) > offsetmin) ? (double)minval : offsetmin;
212     QRect bg = QRect((int)offsetmin, 0, (int)offsetmax, height());
213
214     QPalette palette;
215     //p.fillRect(bg, palette.light());
216     // draw labels
217     p.setPen(palette.dark().color());
218     // draw littlemarklabel
219
220     // draw mediummarklabel
221
222     // draw bigmarklabel
223
224     // draw endlabel
225     /*if (d->showEndL) {
226       if (d->dir == Qt::Horizontal) {
227         p.translate( fontOffset, 0 );
228         p.drawText( END_LABEL_X, END_LABEL_Y, d->endlabel );
229       }*/
230
231     // draw the tiny marks
232     //if (showTinyMarks())
233     /*{
234       fend =   pixelPerMark()*tinyMarkDistance();
235       if (fend > 5) for ( f=offsetmin; f<offsetmax; f+=fend ) {
236           p.drawLine((int)f, BASE_MARK_X1, (int)f, BASE_MARK_X2);
237       }
238     }*/
239
240     for (f = offsetmin; f < offsetmax; f += m_textSpacing) {
241         QString lab = m_timecode.getTimecodeFromFrames((int)((f - offsetmin) / pixelPerMark() / FRAME_SIZE + 0.5));
242         p.drawText((int)f + 2, LABEL_SIZE, lab);
243     }
244
245     if (showLittleMarks()) {
246         // draw the little marks
247         fend = pixelPerMark() * littleMarkDistance();
248         if (fend > 5) for (f = offsetmin; f < offsetmax; f += fend)
249                 p.drawLine((int)f, LITTLE_MARK_X1, (int)f, LITTLE_MARK_X2);
250     }
251     if (showMediumMarks()) {
252         // draw medium marks
253         fend = pixelPerMark() * mediumMarkDistance();
254         if (fend > 5) for (f = offsetmin; f < offsetmax; f += fend)
255                 p.drawLine((int)f, MIDDLE_MARK_X1, (int)f, MIDDLE_MARK_X2);
256     }
257     if (showBigMarks()) {
258         // draw big marks
259         fend = pixelPerMark() * bigMarkDistance();
260         if (fend > 5) for (f = offsetmin; f < offsetmax; f += fend)
261                 p.drawLine((int)f, BIG_MARK_X1, (int)f, BIG_MARK_X2);
262     }
263     /*   if (d->showem) {
264          // draw end marks
265          if (d->dir == Qt::Horizontal) {
266            p.drawLine(minval-d->offset, END_MARK_X1, minval-d->offset, END_MARK_X2);
267            p.drawLine(maxval-d->offset, END_MARK_X1, maxval-d->offset, END_MARK_X2);
268          }
269          else {
270            p.drawLine(END_MARK_X1, minval-d->offset, END_MARK_X2, minval-d->offset);
271            p.drawLine(END_MARK_X1, maxval-d->offset, END_MARK_X2, maxval-d->offset);
272          }
273        }*/
274
275
276     // draw zone cursors
277     int off = offset();
278     if (zoneStart > 0) {
279         QPolygon pa(4);
280         pa.setPoints(4, zoneStart - off + 3, 9, zoneStart - off, 9, zoneStart - off, 18, zoneStart - off + 3, 18);
281         p.drawPolyline(pa);
282     }
283
284     if (zoneEnd > 0) {
285         QRect rec(zoneStart - off + (zoneEnd - zoneStart) / 2 - 4, 9, 8, 9);
286         p.fillRect(rec, QColor(255, 255, 255, 150));
287         p.drawRect(rec);
288
289         QPolygon pa(4);
290         pa.setPoints(4, zoneEnd - off - 3, 9, zoneEnd - off, 9, zoneEnd - off, 18, zoneEnd - off - 3, 18);
291         p.drawPolyline(pa);
292     }
293
294     // draw pointer
295     if (showPointer() && value > 0) {
296         QPolygon pa(3);
297         pa.setPoints(3, value - 6, 7, value + 6, 7, value, 16);
298         p.setBrush(QBrush(Qt::yellow));
299         p.drawPolygon(pa);
300     }
301
302
303
304 }
305
306 #include "customruler.moc"