]> git.sesse.net Git - kdenlive/blob - src/monitor.cpp
Add subclips (dragging a zone from clip monitor to project tree)
[kdenlive] / src / monitor.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
21 #include "monitor.h"
22 #include "renderer.h"
23 #include "monitormanager.h"
24 #include "smallruler.h"
25 #include "docclipbase.h"
26 #include "kdenlivesettings.h"
27
28 #include <KDebug>
29 #include <KLocale>
30 #include <KFileDialog>
31 #include <KApplication>
32 #include <KMessageBox>
33
34 #include <QMouseEvent>
35 #include <QStylePainter>
36 #include <QMenu>
37 #include <QToolButton>
38 #include <QToolBar>
39 #include <QDesktopWidget>
40 #include <QLabel>
41 #include <QIntValidator>
42
43
44 Monitor::Monitor(QString name, MonitorManager *manager, QString profile, QWidget *parent) :
45         QWidget(parent),
46         render(NULL),
47         m_name(name),
48         m_monitorManager(manager),
49         m_currentClip(NULL),
50         m_ruler(new SmallRuler(m_monitorManager)),
51         m_overlay(NULL),
52         m_isActive(false),
53         m_scale(1),
54         m_length(0),
55         m_dragStarted(false)
56 {
57     m_ui.setupUi(this);
58     QVBoxLayout *layout = new QVBoxLayout;
59     layout->setContentsMargins(0, 0, 0, 0);
60     layout->addWidget(m_ruler);
61     m_ui.ruler_frame->setLayout(layout);
62     setSizePolicy(QSizePolicy::Expanding, QSizePolicy::Expanding);
63     setMinimumHeight(200);
64     QToolBar *toolbar = new QToolBar(name, this);
65     QVBoxLayout *layout2 = new QVBoxLayout;
66     layout2->setContentsMargins(0, 0, 0, 0);
67
68     m_playIcon = KIcon("media-playback-start");
69     m_pauseIcon = KIcon("media-playback-pause");
70
71     if (name != "chapter") {
72         toolbar->addAction(KIcon("kdenlive-zone-start"), i18n("Set zone start"), this, SLOT(slotSetZoneStart()));
73         toolbar->addAction(KIcon("kdenlive-zone-end"), i18n("Set zone end"), this, SLOT(slotSetZoneEnd()));
74     } else m_ruler->setZone(-3, -2);
75
76     toolbar->addAction(KIcon("media-seek-backward"), i18n("Rewind"), this, SLOT(slotRewind()));
77     toolbar->addAction(KIcon("media-skip-backward"), i18n("Rewind 1 frame"), this, SLOT(slotRewindOneFrame()));
78
79     QToolButton *playButton = new QToolButton(toolbar);
80     m_playMenu = new QMenu(i18n("Play..."), this);
81     m_playAction = m_playMenu->addAction(m_playIcon, i18n("Play"));
82     m_playAction->setCheckable(true);
83     connect(m_playAction, SIGNAL(triggered()), this, SLOT(slotPlay()));
84
85     playButton->setMenu(m_playMenu);
86     playButton->setPopupMode(QToolButton::MenuButtonPopup);
87     toolbar->addWidget(playButton);
88
89     toolbar->addAction(KIcon("media-skip-forward"), i18n("Forward 1 frame"), this, SLOT(slotForwardOneFrame()));
90     toolbar->addAction(KIcon("media-seek-forward"), i18n("Forward"), this, SLOT(slotForward()));
91
92     playButton->setDefaultAction(m_playAction);
93
94     if (name != "chapter") {
95         QToolButton *configButton = new QToolButton(toolbar);
96         m_configMenu = new QMenu(i18n("Misc..."), this);
97         configButton->setIcon(KIcon("system-run"));
98         configButton->setMenu(m_configMenu);
99         configButton->setPopupMode(QToolButton::QToolButton::InstantPopup);
100         toolbar->addWidget(configButton);
101
102         if (name == "clip") {
103             m_markerMenu = new QMenu(i18n("Go to marker..."), this);
104             m_markerMenu->setEnabled(false);
105             m_configMenu->addMenu(m_markerMenu);
106             connect(m_markerMenu, SIGNAL(triggered(QAction *)), this, SLOT(slotGoToMarker(QAction *)));
107         }
108         m_configMenu->addAction(KIcon("transform-scale"), i18n("Resize (100%)"), this, SLOT(slotSetSizeOneToOne()));
109         m_configMenu->addAction(KIcon("transform-scale"), i18n("Resize (50%)"), this, SLOT(slotSetSizeOneToTwo()));
110     }
111
112     QWidget *spacer = new QWidget(this);
113     spacer->setSizePolicy(QSizePolicy::MinimumExpanding, QSizePolicy::MinimumExpanding);
114     toolbar->addWidget(spacer);
115     m_timePos = new KRestrictedLine(this);
116     m_timePos->setSizePolicy(QSizePolicy::Maximum, QSizePolicy::MinimumExpanding);
117     m_frametimecode = KdenliveSettings::frametimecode();
118     if (m_frametimecode) {
119         m_timePos->setInputMask(QString());
120         m_timePos->setValidator(new QIntValidator(this));
121     } else m_timePos->setInputMask("99:99:99:99");
122     toolbar->addWidget(m_timePos);
123
124     connect(m_timePos, SIGNAL(editingFinished()), this, SLOT(slotSeek()));
125
126     layout2->addWidget(toolbar);
127     m_ui.button_frame->setLayout(layout2);
128     const int toolHeight = toolbar->height();
129     m_ui.button_frame->setMinimumHeight(toolHeight);
130
131     //m_ruler->setPixelPerMark(3);
132
133     if (profile.isEmpty()) profile = KdenliveSettings::current_profile();
134
135     QVBoxLayout *rendererBox = new QVBoxLayout(m_ui.video_frame);
136     rendererBox->setContentsMargins(0, 0, 0, 0);
137 #ifdef Q_WS_MAC
138     m_glWidget = new VideoGLWidget(m_ui.video_frame);
139     rendererBox->addWidget(m_glWidget);
140     render = new Render(m_name, (int) m_ui.video_frame->winId(), -1, profile, this);
141     m_glWidget->setImageAspectRatio(render->dar());
142     m_glWidget->setBackgroundColor(KdenliveSettings::window_background());
143     m_glWidget->resize(m_ui.video_frame->size());
144     connect(render, SIGNAL(showImageSignal(QImage)), m_glWidget, SLOT(showImage(QImage)));
145     m_monitorRefresh = 0;
146 #else
147     m_monitorRefresh = new MonitorRefresh(m_ui.video_frame);
148     rendererBox->addWidget(m_monitorRefresh);
149     render = new Render(m_name, (int) m_monitorRefresh->winId(), -1, profile, this);
150     m_monitorRefresh->setRenderer(render);
151 #endif
152
153     connect(m_ruler, SIGNAL(seekRenderer(int)), this, SLOT(slotSeek(int)));
154     connect(render, SIGNAL(durationChanged(int)), this, SLOT(adjustRulerSize(int)));
155     connect(render, SIGNAL(rendererPosition(int)), this, SLOT(seekCursor(int)));
156     connect(render, SIGNAL(rendererStopped(int)), this, SLOT(rendererStopped(int)));
157     connect(render, SIGNAL(blockMonitors()), this, SIGNAL(blockMonitors()));
158
159     //render->createVideoXWindow(m_ui.video_frame->winId(), -1);
160
161     if (name != "clip") {
162         connect(render, SIGNAL(rendererPosition(int)), this, SIGNAL(renderPosition(int)));
163         connect(render, SIGNAL(durationChanged(int)), this, SIGNAL(durationChanged(int)));
164         connect(m_ruler, SIGNAL(zoneChanged(QPoint)), this, SIGNAL(zoneUpdated(QPoint)));
165     } else {
166         connect(m_ruler, SIGNAL(zoneChanged(QPoint)), this, SLOT(setClipZone(QPoint)));
167     }
168 #ifndef Q_WS_MAC
169     m_monitorRefresh->show();
170 #endif
171     kDebug() << "/////// BUILDING MONITOR, ID: " << m_ui.video_frame->winId();
172 }
173
174 Monitor::~Monitor()
175 {
176     delete m_ruler;
177     delete m_timePos;
178     delete m_overlay;
179     delete m_monitorRefresh;
180     delete render;
181 }
182
183 QString Monitor::name() const
184 {
185     return m_name;
186 }
187
188 void Monitor::setupMenu(QMenu *goMenu, QAction *playZone, QAction *loopZone, QMenu *markerMenu)
189 {
190     m_contextMenu = new QMenu(this);
191     m_contextMenu->addMenu(m_playMenu);
192     if (goMenu) m_contextMenu->addMenu(goMenu);
193     if (markerMenu) m_contextMenu->addMenu(markerMenu);
194
195     m_playMenu->addAction(playZone);
196     m_playMenu->addAction(loopZone);
197
198     //TODO: add save zone to timeline monitor when fixed
199     if (m_name == "clip") {
200         m_contextMenu->addMenu(m_markerMenu);
201         m_contextMenu->addAction(KIcon("document-save"), i18n("Save zone"), this, SLOT(slotSaveZone()));
202     }
203     QAction *extractFrame = m_configMenu->addAction(KIcon("document-new"), i18n("Extract frame"), this, SLOT(slotExtractCurrentFrame()));
204     m_contextMenu->addAction(extractFrame);
205
206     if (m_name != "clip") {
207 #ifndef Q_WS_MAC
208         QAction *splitView = m_contextMenu->addAction(KIcon("view-split-left-right"), i18n("Split view"), render, SLOT(slotSplitView(bool)));
209         splitView->setCheckable(true);
210         m_configMenu->addAction(splitView);
211 #endif
212     } else {
213         QAction *setThumbFrame = m_contextMenu->addAction(KIcon("document-new"), i18n("Set current image as thumbnail"), this, SLOT(slotSetThumbFrame()));
214         m_configMenu->addAction(setThumbFrame);
215     }
216
217     QAction *showTips = m_contextMenu->addAction(KIcon("help-hint"), i18n("Monitor overlay infos"));
218     showTips->setCheckable(true);
219     connect(showTips, SIGNAL(toggled(bool)), this, SLOT(slotSwitchMonitorInfo(bool)));
220     showTips->setChecked(KdenliveSettings::displayMonitorInfo());
221
222     QAction *dropFrames = m_contextMenu->addAction(KIcon(), i18n("Real time (drop frames)"));
223     dropFrames->setCheckable(true);
224     dropFrames->setChecked(true);
225     connect(dropFrames, SIGNAL(toggled(bool)), this, SLOT(slotSwitchDropFrames(bool)));
226
227     m_configMenu->addAction(showTips);
228     m_configMenu->addAction(dropFrames);
229
230 }
231
232 void Monitor::slotGoToMarker(QAction *action)
233 {
234     int pos = action->data().toInt();
235     slotSeek(pos);
236 }
237
238 void Monitor::slotSetSizeOneToOne()
239 {
240     QRect r = QApplication::desktop()->screenGeometry();
241     const int maxWidth = r.width() - 20;
242     const int maxHeight = r.height() - 20;
243     int width = render->renderWidth();
244     int height = render->renderHeight();
245     kDebug() << "// render info: " << width << "x" << height;
246     while (width >= maxWidth || height >= maxHeight) {
247         width = width * 0.8;
248         height = height * 0.8;
249     }
250     kDebug() << "// MONITOR; set SIZE: " << width << ", " << height;
251     m_ui.video_frame->setFixedSize(width, height);
252     updateGeometry();
253     adjustSize();
254     //m_ui.video_frame->setMinimumSize(0, 0);
255     emit adjustMonitorSize();
256 }
257
258 void Monitor::slotSetSizeOneToTwo()
259 {
260     QRect r = QApplication::desktop()->screenGeometry();
261     const int maxWidth = r.width() - 20;
262     const int maxHeight = r.height() - 20;
263     int width = render->renderWidth() / 2;
264     int height = render->renderHeight() / 2;
265     kDebug() << "// render info: " << width << "x" << height;
266     while (width >= maxWidth || height >= maxHeight) {
267         width = width * 0.8;
268         height = height * 0.8;
269     }
270     kDebug() << "// MONITOR; set SIZE: " << width << ", " << height;
271     m_ui.video_frame->setFixedSize(width, height);
272     updateGeometry();
273     adjustSize();
274     //m_ui.video_frame->setMinimumSize(0, 0);
275     emit adjustMonitorSize();
276 }
277
278 void Monitor::resetSize()
279 {
280     m_ui.video_frame->setMinimumSize(0, 0);
281 }
282
283 DocClipBase *Monitor::activeClip()
284 {
285     return m_currentClip;
286 }
287
288 void Monitor::updateMarkers(DocClipBase *source)
289 {
290     if (source == m_currentClip && source != NULL) {
291         m_markerMenu->clear();
292         QList <CommentedTime> markers = m_currentClip->commentedSnapMarkers();
293         if (!markers.isEmpty()) {
294             QList <int> marks;
295             for (int i = 0; i < markers.count(); i++) {
296                 int pos = (int) markers.at(i).time().frames(m_monitorManager->timecode().fps());
297                 marks.append(pos);
298                 QString position = m_monitorManager->timecode().getTimecode(markers.at(i).time()) + ' ' + markers.at(i).comment();
299                 QAction *go = m_markerMenu->addAction(position);
300                 go->setData(pos);
301             }
302             m_ruler->setMarkers(marks);
303         } else m_ruler->setMarkers(QList <int>());
304         m_markerMenu->setEnabled(!m_markerMenu->isEmpty());
305     }
306 }
307
308 void Monitor::slotSeekToPreviousSnap()
309 {
310     if (m_currentClip) slotSeek(getSnapForPos(true).frames(m_monitorManager->timecode().fps()));
311 }
312
313 void Monitor::slotSeekToNextSnap()
314 {
315     if (m_currentClip) slotSeek(getSnapForPos(false).frames(m_monitorManager->timecode().fps()));
316 }
317
318 GenTime Monitor::position()
319 {
320     return render->seekPosition();
321 }
322
323 GenTime Monitor::getSnapForPos(bool previous)
324 {
325     QList <GenTime> snaps;
326     QList < GenTime > markers = m_currentClip->snapMarkers();
327     for (int i = 0; i < markers.size(); ++i) {
328         GenTime t = markers.at(i);
329         snaps.append(t);
330     }
331     QPoint zone = m_ruler->zone();
332     snaps.append(GenTime(zone.x(), m_monitorManager->timecode().fps()));
333     snaps.append(GenTime(zone.y(), m_monitorManager->timecode().fps()));
334     snaps.append(GenTime());
335     snaps.append(m_currentClip->duration());
336     qSort(snaps);
337
338     const GenTime pos = render->seekPosition();
339     for (int i = 0; i < snaps.size(); ++i) {
340         if (previous && snaps.at(i) >= pos) {
341             if (i == 0) i = 1;
342             return snaps.at(i - 1);
343         } else if (!previous && snaps.at(i) > pos) {
344             return snaps.at(i);
345         }
346     }
347     return GenTime();
348 }
349
350
351
352 void Monitor::slotZoneMoved(int start, int end)
353 {
354     m_ruler->setZone(start, end);
355     checkOverlay();
356     setClipZone(m_ruler->zone());
357 }
358
359 void Monitor::slotSetZoneStart()
360 {
361     m_ruler->setZone(render->seekFramePosition(), -1);
362     emit zoneUpdated(m_ruler->zone());
363     checkOverlay();
364     setClipZone(m_ruler->zone());
365 }
366
367 void Monitor::slotSetZoneEnd()
368 {
369     m_ruler->setZone(-1, render->seekFramePosition());
370     emit zoneUpdated(m_ruler->zone());
371     checkOverlay();
372     setClipZone(m_ruler->zone());
373 }
374
375 // virtual
376 void Monitor::mousePressEvent(QMouseEvent * event)
377 {
378     if (event->button() != Qt::RightButton) {
379         if (m_ui.video_frame->underMouse()) {
380             m_dragStarted = true;
381             m_DragStartPosition = event->pos();
382         }
383     } else m_contextMenu->popup(event->globalPos());
384 }
385
386 // virtual
387 void Monitor::mouseReleaseEvent(QMouseEvent * event)
388 {
389     if (m_dragStarted) {
390         if (m_ui.video_frame->underMouse()) {
391             if (isActive()) slotPlay();
392             else activateMonitor();
393         } else QWidget::mouseReleaseEvent(event);
394         m_dragStarted = false;
395     }
396 }
397
398
399 // virtual
400 void Monitor::mouseMoveEvent(QMouseEvent *event)
401 {
402     // kDebug() << "// DRAG STARTED, MOUSE MOVED: ";
403     if (!m_dragStarted || m_currentClip == NULL) return;
404
405     if ((event->pos() - m_DragStartPosition).manhattanLength()
406             < QApplication::startDragDistance())
407         return;
408
409     {
410         QDrag *drag = new QDrag(this);
411         QMimeData *mimeData = new QMimeData;
412
413         QStringList list;
414         list.append(m_currentClip->getId());
415         QPoint p = m_ruler->zone();
416         list.append(QString::number(p.x()));
417         list.append(QString::number(p.y()));
418         QByteArray data;
419         data.append(list.join(";").toUtf8());
420         mimeData->setData("kdenlive/clip", data);
421         drag->setMimeData(mimeData);
422         QPixmap pix = m_currentClip->thumbnail();
423         drag->setPixmap(pix);
424         drag->setHotSpot(QPoint(0, 50));
425         drag->start(Qt::MoveAction);
426
427         //Qt::DropAction dropAction;
428         //dropAction = drag->start(Qt::CopyAction | Qt::MoveAction);
429
430         //Qt::DropAction dropAction = drag->exec();
431
432     }
433     //event->accept();
434 }
435
436 /*void Monitor::dragMoveEvent(QDragMoveEvent * event) {
437     event->setDropAction(Qt::IgnoreAction);
438     event->setDropAction(Qt::MoveAction);
439     if (event->mimeData()->hasText()) {
440         event->acceptProposedAction();
441     }
442 }
443
444 Qt::DropActions Monitor::supportedDropActions() const {
445     // returns what actions are supported when dropping
446     return Qt::MoveAction;
447 }*/
448
449 QStringList Monitor::mimeTypes() const
450 {
451     QStringList qstrList;
452     // list of accepted mime types for drop
453     qstrList.append("kdenlive/clip");
454     return qstrList;
455 }
456
457
458 // virtual
459 /** Move to other position on mousewheel
460  *
461  * Moves towards end of clip/timeline on mousewheel down/back,
462  * opposite for mousewheel up/forward.
463  *
464  * Ctrl+wheel moves single frame, without Ctrl moves a second.
465  *
466  * See also http://www.kdenlive.org/mantis/view.php?id=265 */
467 void Monitor::wheelEvent(QWheelEvent * event)
468 {
469     if (event->modifiers() == Qt::ControlModifier) {
470         int delta = m_monitorManager->timecode().fps();
471         if (event->delta() > 0) delta = 0 - delta;
472         slotSeek(render->seekFramePosition() - delta);
473     } else {
474         if (event->delta() >= 0) slotForwardOneFrame();
475         else slotRewindOneFrame();
476     }
477 }
478
479 void Monitor::slotSetThumbFrame()
480 {
481     if (m_currentClip == NULL) {
482         return;
483     }
484     m_currentClip->setClipThumbFrame((uint) render->seekFramePosition());
485     emit refreshClipThumbnail(m_currentClip->getId());
486 }
487
488 void Monitor::slotExtractCurrentFrame()
489 {
490     QImage frame = render->extractFrame(render->seekFramePosition());
491     KFileDialog *fs = new KFileDialog(KUrl(), "image/png", this);
492     fs->setOperationMode(KFileDialog::Saving);
493     fs->setMode(KFile::File);
494 #if KDE_IS_VERSION(4,2,0)
495     fs->setConfirmOverwrite(true);
496 #endif
497     fs->setKeepLocation(true);
498     fs->exec();
499     QString path = fs->selectedFile();
500     delete fs;
501     if (!path.isEmpty()) {
502         frame.save(path);
503     }
504 }
505
506 bool Monitor::isActive() const
507 {
508     return m_isActive;
509 }
510
511 void Monitor::activateMonitor()
512 {
513     if (!m_isActive) m_monitorManager->switchMonitors(); //m_monitorManager->activateMonitor(m_name);
514 }
515
516 void Monitor::setTimePos(const QString &pos)
517 {
518     if (m_frametimecode) {
519         int frames = m_monitorManager->timecode().getFrameCount(pos);
520         m_timePos->setText(QString::number(frames));
521     } else m_timePos->setText(pos);
522     slotSeek();
523 }
524
525 void Monitor::slotSeek()
526 {
527     int frames;
528     if (m_frametimecode) frames = m_timePos->text().toInt();
529     else frames = m_monitorManager->timecode().getFrameCount(m_timePos->text());
530     //kDebug() << "// / / SEEK TO: " << frames;
531     slotSeek(frames);
532 }
533
534 void Monitor::slotSeek(int pos)
535 {
536     activateMonitor();
537     if (render == NULL) return;
538     render->seekToFrame(pos);
539     emit renderPosition(render->seekFramePosition());
540 }
541
542 void Monitor::checkOverlay()
543 {
544     if (m_overlay == NULL) return;
545     int pos = render->seekFramePosition();
546     QPoint zone = m_ruler->zone();
547     if (pos == zone.x()) m_overlay->setOverlayText(i18n("In Point"));
548     else if (pos == zone.y()) m_overlay->setOverlayText(i18n("Out Point"));
549     else {
550         if (m_currentClip) {
551             QString markerComment = m_currentClip->markerComment(GenTime(pos, m_monitorManager->timecode().fps()));
552             if (markerComment.isEmpty()) m_overlay->setHidden(true);
553             else m_overlay->setOverlayText(markerComment, false);
554         } else m_overlay->setHidden(true);
555     }
556 }
557
558 void Monitor::slotStart()
559 {
560     activateMonitor();
561     render->play(0);
562     render->seekToFrame(0);
563     emit renderPosition(0);
564 }
565
566 void Monitor::slotEnd()
567 {
568     activateMonitor();
569     render->play(0);
570     render->seekToFrame(render->getLength());
571     emit renderPosition(render->seekFramePosition());
572 }
573
574 void Monitor::slotZoneStart()
575 {
576     activateMonitor();
577     render->play(0);
578     render->seekToFrame(m_ruler->zone().x());
579     emit renderPosition(render->seekFramePosition());
580 }
581
582 void Monitor::slotZoneEnd()
583 {
584     activateMonitor();
585     render->play(0);
586     render->seekToFrame(m_ruler->zone().y());
587     emit renderPosition(render->seekFramePosition());
588 }
589
590 void Monitor::slotRewind(double speed)
591 {
592     activateMonitor();
593     if (speed == 0) {
594         double currentspeed = render->playSpeed();
595         if (currentspeed >= 0) render->play(-2);
596         else render->play(currentspeed * 2);
597     } else render->play(speed);
598     m_playAction->setChecked(true);
599     m_playAction->setIcon(m_pauseIcon);
600 }
601
602 void Monitor::slotForward(double speed)
603 {
604     activateMonitor();
605     if (speed == 0) {
606         double currentspeed = render->playSpeed();
607         if (currentspeed <= 1) render->play(2);
608         else render->play(currentspeed * 2);
609     } else render->play(speed);
610     m_playAction->setChecked(true);
611     m_playAction->setIcon(m_pauseIcon);
612 }
613
614 void Monitor::slotRewindOneFrame(int diff)
615 {
616     activateMonitor();
617     render->play(0);
618     render->seekToFrameDiff(-diff);
619     emit renderPosition(render->seekFramePosition());
620 }
621
622 void Monitor::slotForwardOneFrame(int diff)
623 {
624     activateMonitor();
625     render->play(0);
626     render->seekToFrameDiff(diff);
627     emit renderPosition(render->seekFramePosition());
628 }
629
630 void Monitor::seekCursor(int pos)
631 {
632     activateMonitor();
633     checkOverlay();
634     if (m_frametimecode) m_timePos->setText(QString::number(pos));
635     else m_timePos->setText(m_monitorManager->timecode().getTimecodeFromFrames(pos));
636     m_ruler->slotNewValue(pos);
637 }
638
639 void Monitor::rendererStopped(int pos)
640 {
641     if (m_currentClip != NULL && m_playAction->isChecked()) {
642         // Clip monitor
643         if (pos >= m_length - 1) {
644             slotStart();
645             return;
646         }
647     }
648     m_ruler->slotNewValue(pos);
649     checkOverlay();
650     if (m_frametimecode) m_timePos->setText(QString::number(pos));
651     else m_timePos->setText(m_monitorManager->timecode().getTimecodeFromFrames(pos));
652     m_playAction->setChecked(false);
653     m_playAction->setIcon(m_playIcon);
654 }
655
656 void Monitor::initMonitor()
657 {
658     kDebug() << "/////// INITING MONITOR, ID: " << m_ui.video_frame->winId();
659 }
660
661 // virtual
662 /*void Monitor::resizeEvent(QResizeEvent * event) {
663     QWidget::resizeEvent(event);
664     adjustRulerSize(-1);
665     if (render && m_isActive) render->doRefresh();
666     //
667 }*/
668
669 void Monitor::adjustRulerSize(int length)
670 {
671     if (length > 0) m_length = length;
672     m_ruler->adjustScale(m_length);
673     if (m_currentClip != NULL) {
674         QPoint zone = m_currentClip->zone();
675         m_ruler->setZone(zone.x(), zone.y());
676     }
677 }
678
679 void Monitor::stop()
680 {
681     m_isActive = false;
682     if (render) render->stop();
683     //kDebug()<<"/// MONITOR RENDER STOP";
684 }
685
686 void Monitor::start()
687 {
688     m_isActive = true;
689     if (render) render->start();
690     //kDebug()<<"/// MONITOR RENDER START";
691 }
692
693 void Monitor::refreshMonitor(bool visible)
694 {
695     if (visible && render) {
696         activateMonitor();
697         render->doRefresh(); //askForRefresh();
698     }
699 }
700
701 void Monitor::pause()
702 {
703     if (render == NULL) return;
704     activateMonitor();
705     render->pause();
706     //m_playAction->setChecked(true);
707     //m_playAction->setIcon(m_pauseIcon);
708 }
709
710 void Monitor::slotPlay()
711 {
712     if (render == NULL) return;
713     activateMonitor();
714     if (render->playSpeed() == 0) {
715         m_playAction->setChecked(true);
716         m_playAction->setIcon(m_pauseIcon);
717     } else {
718         m_playAction->setChecked(false);
719         m_playAction->setIcon(m_playIcon);
720     }
721     render->switchPlay();
722 }
723
724 void Monitor::slotPlayZone()
725 {
726     if (render == NULL) return;
727     activateMonitor();
728     QPoint p = m_ruler->zone();
729     render->playZone(GenTime(p.x(), m_monitorManager->timecode().fps()), GenTime(p.y(), m_monitorManager->timecode().fps()));
730     m_playAction->setChecked(true);
731     m_playAction->setIcon(m_pauseIcon);
732 }
733
734 void Monitor::slotLoopZone()
735 {
736     if (render == NULL) return;
737     activateMonitor();
738     QPoint p = m_ruler->zone();
739     render->loopZone(GenTime(p.x(), m_monitorManager->timecode().fps()), GenTime(p.y(), m_monitorManager->timecode().fps()));
740     m_playAction->setChecked(true);
741     m_playAction->setIcon(m_pauseIcon);
742 }
743
744 void Monitor::slotSetXml(DocClipBase *clip, QPoint zone, const int position)
745 {
746     if (render == NULL) return;
747     activateMonitor();
748     if (clip == NULL && m_currentClip != NULL) {
749         m_currentClip = NULL;
750         render->setProducer(NULL, -1);
751         return;
752     }
753     if (clip != m_currentClip) {
754         m_currentClip = clip;
755         updateMarkers(clip);
756         if (render->setProducer(clip->producer(), position) == -1) {
757             // MLT CONSUMER is broken
758             emit blockMonitors();
759         }
760     } else if (position != -1) render->seek(GenTime(position, m_monitorManager->timecode().fps()));
761     if (!zone.isNull()) {
762         m_ruler->setZone(zone.x(), zone.y());
763         render->seek(GenTime(zone.x(), m_monitorManager->timecode().fps()));
764     }
765 }
766
767 void Monitor::slotOpenFile(const QString &file)
768 {
769     if (render == NULL) return;
770     activateMonitor();
771     QDomDocument doc;
772     QDomElement mlt = doc.createElement("mlt");
773     doc.appendChild(mlt);
774     QDomElement prod = doc.createElement("producer");
775     mlt.appendChild(prod);
776     prod.setAttribute("mlt_service", "avformat");
777     prod.setAttribute("resource", file);
778     render->setSceneList(doc, 0);
779 }
780
781 void Monitor::slotSaveZone()
782 {
783     if (render == NULL) return;
784     emit saveZone(render, m_ruler->zone());
785
786     //render->setSceneList(doc, 0);
787 }
788
789
790 void Monitor::resetProfile(const QString profile)
791 {
792     if (render == NULL) return;
793     render->resetProfile(profile);
794 }
795
796 void Monitor::saveSceneList(QString path, QDomElement info)
797 {
798     if (render == NULL) return;
799     render->saveSceneList(path, info);
800 }
801
802 const QString Monitor::sceneList()
803 {
804     if (render == NULL) return QString();
805     return render->sceneList();
806 }
807
808
809 void Monitor::setClipZone(QPoint pos)
810 {
811     if (m_currentClip == NULL) return;
812     m_currentClip->setZone(pos);
813 }
814
815 void Monitor::slotSwitchDropFrames(bool show)
816 {
817     render->setDropFrames(show);
818 }
819
820 void Monitor::slotSwitchMonitorInfo(bool show)
821 {
822     KdenliveSettings::setDisplayMonitorInfo(show);
823     if (show) {
824         if (m_overlay) return;
825 #ifndef Q_WS_MAC
826         m_overlay = new Overlay(m_monitorRefresh);
827         m_overlay->raise();
828         m_overlay->setHidden(true);
829 #else
830         m_overlay = new Overlay(m_glWidget);
831 #endif
832     } else {
833         delete m_overlay;
834         m_overlay = NULL;
835     }
836 }
837
838 void Monitor::updateTimecodeFormat()
839 {
840     m_frametimecode = KdenliveSettings::frametimecode();
841     if (m_frametimecode) {
842         int frames = m_monitorManager->timecode().getFrameCount(m_timePos->text());
843         m_timePos->setValidator(new QIntValidator(this));
844         m_timePos->setInputMask(QString());
845         m_timePos->setText(QString::number(frames));
846     } else {
847         int pos = m_timePos->text().toInt();
848         m_timePos->setValidator(0);
849         m_timePos->setInputMask("99:99:99:99");
850         m_timePos->setText(m_monitorManager->timecode().getTimecodeFromFrames(pos));
851     }
852 }
853
854
855 MonitorRefresh::MonitorRefresh(QWidget* parent) : \
856         QWidget(parent),
857         m_renderer(NULL)
858 {
859     setAttribute(Qt::WA_PaintOnScreen);
860     setAttribute(Qt::WA_OpaquePaintEvent);
861     //setAttribute(Qt::WA_NoSystemBackground);
862 }
863
864 void MonitorRefresh::setRenderer(Render* render)
865 {
866     m_renderer = render;
867 }
868
869 void MonitorRefresh::paintEvent(QPaintEvent * /*event*/)
870 {
871     if (m_renderer) m_renderer->doRefresh();
872 }
873
874
875 Overlay::Overlay(QWidget* parent) :
876         QLabel(parent)
877 {
878     setAttribute(Qt::WA_TransparentForMouseEvents);
879     setAttribute(Qt::WA_OpaquePaintEvent);
880     //setAttribute(Qt::WA_NoSystemBackground);
881     setAutoFillBackground(false);
882 }
883
884 void Overlay::paintEvent(QPaintEvent * /*event*/)
885 {
886     QPainter painter(this);
887     QColor col;
888     painter.setPen(Qt::white);
889     if (m_isZone) col = QColor(200, 0, 0);
890     else col = QColor(0, 0, 200);
891     painter.fillRect(rect(), col);
892     painter.drawText(rect(), Qt::AlignCenter, text());
893 }
894
895
896
897 void Overlay::setOverlayText(const QString &text, bool isZone)
898 {
899     setHidden(true);
900     m_isZone = isZone;
901     setText(' ' + text + ' ');
902     setHidden(false);
903     update();
904 }
905
906 #include "monitor.moc"