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