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