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