]> git.sesse.net Git - kdenlive/blob - src/monitor.cpp
Fix updating of monitor timecode, solving offset between timeline and monitor timecode
[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 "smallruler.h"
23 #include "docclipbase.h"
24 #include "abstractclipitem.h"
25 #include "monitorscene.h"
26 #include "monitoreditwidget.h"
27 #include "kdenlivesettings.h"
28
29 #include <KDebug>
30 #include <KLocale>
31 #include <KFileDialog>
32 #include <KApplication>
33 #include <KMessageBox>
34
35 #include <QMouseEvent>
36 #include <QStylePainter>
37 #include <QMenu>
38 #include <QToolButton>
39 #include <QToolBar>
40 #include <QDesktopWidget>
41 #include <QLabel>
42 #include <QIntValidator>
43 #include <QVBoxLayout>
44
45
46 Monitor::Monitor(QString name, MonitorManager *manager, QString profile, QWidget *parent) :
47     AbstractMonitor(parent),
48     render(NULL),
49     m_name(name),
50     m_monitorManager(manager),
51     m_currentClip(NULL),
52     m_ruler(new SmallRuler(m_monitorManager)),
53     m_overlay(NULL),
54     m_scale(1),
55     m_length(0),
56     m_dragStarted(false),
57     m_monitorRefresh(NULL),
58     m_effectWidget(NULL),
59     m_selectedClip(NULL),
60     m_loopClipTransition(true),
61     m_editMarker(NULL)
62 {
63     QVBoxLayout *layout = new QVBoxLayout;
64     layout->setContentsMargins(0, 0, 0, 0);
65     layout->setSpacing(0);
66
67     // Video widget holder
68     m_videoBox = new VideoContainer(this);
69     m_videoBox->setContentsMargins(0, 0, 0, 0);
70     m_videoBox->setSizePolicy(QSizePolicy::Preferred, QSizePolicy::Preferred);
71     layout->addWidget(m_videoBox, 10);
72     layout->addStretch();
73
74     // Get base size for icons
75     int s = style()->pixelMetric(QStyle::PM_SmallIconSize);
76
77     // Monitor ruler
78     layout->addWidget(m_ruler);
79     // Tool bar buttons
80     m_toolbar = new QToolBar(name, this);
81     m_toolbar->setIconSize(QSize(s, s));
82
83     m_playIcon = KIcon("media-playback-start");
84     m_pauseIcon = KIcon("media-playback-pause");
85
86     if (name != "chapter") {
87         m_toolbar->addAction(KIcon("kdenlive-zone-start"), i18n("Set zone start"), this, SLOT(slotSetZoneStart()));
88         m_toolbar->addAction(KIcon("kdenlive-zone-end"), i18n("Set zone end"), this, SLOT(slotSetZoneEnd()));
89     } else {
90         m_ruler->setZone(-3, -2);
91     }
92
93     m_toolbar->addAction(KIcon("media-seek-backward"), i18n("Rewind"), this, SLOT(slotRewind()));
94     //m_toolbar->addAction(KIcon("media-skip-backward"), i18n("Rewind 1 frame"), this, SLOT(slotRewindOneFrame()));
95
96     QToolButton *playButton = new QToolButton(m_toolbar);
97     m_playMenu = new QMenu(i18n("Play..."), this);
98     m_playAction = m_playMenu->addAction(m_playIcon, i18n("Play"));
99     //m_playAction->setCheckable(true);
100     connect(m_playAction, SIGNAL(triggered()), this, SLOT(slotPlay()));
101
102     playButton->setMenu(m_playMenu);
103     playButton->setPopupMode(QToolButton::MenuButtonPopup);
104     m_toolbar->addWidget(playButton);
105
106     //m_toolbar->addAction(KIcon("media-skip-forward"), i18n("Forward 1 frame"), this, SLOT(slotForwardOneFrame()));
107     m_toolbar->addAction(KIcon("media-seek-forward"), i18n("Forward"), this, SLOT(slotForward()));
108
109     playButton->setDefaultAction(m_playAction);
110
111     if (name != "chapter") {
112         QToolButton *configButton = new QToolButton(m_toolbar);
113         m_configMenu = new QMenu(i18n("Misc..."), this);
114         configButton->setIcon(KIcon("system-run"));
115         configButton->setMenu(m_configMenu);
116         configButton->setPopupMode(QToolButton::QToolButton::InstantPopup);
117         m_toolbar->addWidget(configButton);
118
119         if (name == "clip") {
120             m_markerMenu = new QMenu(i18n("Go to marker..."), this);
121             m_markerMenu->setEnabled(false);
122             m_configMenu->addMenu(m_markerMenu);
123             connect(m_markerMenu, SIGNAL(triggered(QAction *)), this, SLOT(slotGoToMarker(QAction *)));
124         }
125         m_configMenu->addAction(KIcon("transform-scale"), i18n("Resize (100%)"), this, SLOT(slotSetSizeOneToOne()));
126         m_configMenu->addAction(KIcon("transform-scale"), i18n("Resize (50%)"), this, SLOT(slotSetSizeOneToTwo()));
127     }
128
129     // Create Volume slider popup
130     m_volumePopup = new QFrame(this, Qt::Popup);
131     QVBoxLayout *poplayout = new QVBoxLayout;
132     poplayout->setContentsMargins(0, 0, 0, 0);
133     m_audioSlider = new QSlider(Qt::Vertical);
134     m_audioSlider->setRange(0, 100);
135     poplayout->addWidget(m_audioSlider);
136     m_volumePopup->setLayout(poplayout);
137     KIcon icon;
138     if (KdenliveSettings::volume() == 0) icon = KIcon("audio-volume-muted");
139     else icon = KIcon("audio-volume-medium");
140
141     m_volumeWidget = m_toolbar->widgetForAction(m_toolbar->addAction(icon, i18n("Audio volume"), this, SLOT(slotShowVolume())));
142
143     // we need to show / hide the popup once so that it's geometry can be calculated in slotShowVolume
144     m_volumePopup->show();
145     m_volumePopup->hide();
146
147     setSizePolicy(QSizePolicy::Expanding, QSizePolicy::Expanding);
148     setLayout(layout);
149     setMinimumHeight(200);
150
151     if (profile.isEmpty()) profile = KdenliveSettings::current_profile();
152
153     bool monitorCreated = false;
154 #ifdef Q_WS_MAC
155     createOpenGlWidget(m_videoBox, profile);
156     monitorCreated = true;
157     //m_glWidget->setFixedSize(width, height);
158 #elif defined (USE_OPEN_GL)
159     if (KdenliveSettings::openglmonitors()) {
160         monitorCreated = createOpenGlWidget(m_videoBox, profile);
161     }
162 #endif
163     QVBoxLayout *lay = new QVBoxLayout;
164     lay->setContentsMargins(0, 0, 0, 0);
165     if (!monitorCreated) {
166         m_monitorRefresh = new MonitorRefresh;
167         lay->addWidget(m_monitorRefresh);
168         m_videoBox->setLayout(lay);
169         render = new Render(m_name, (int) m_monitorRefresh->winId(), profile, this);
170         m_monitorRefresh->setRenderer(render);
171     }
172 #if defined (USE_OPEN_GL)
173     else if (m_glWidget) {
174         lay->addWidget(m_glWidget);
175         m_videoBox->setLayout(lay);
176     }
177 #endif
178
179     connect(m_audioSlider, SIGNAL(valueChanged(int)), this, SLOT(slotSetVolume(int)));
180     connect(m_ruler, SIGNAL(seekRenderer(int)), this, SLOT(slotSeek(int)));
181     connect(render, SIGNAL(durationChanged(int)), this, SLOT(adjustRulerSize(int)));
182     connect(render, SIGNAL(rendererStopped(int)), this, SLOT(rendererStopped(int)));
183
184     if (name != "clip") {
185         connect(render, SIGNAL(rendererPosition(int)), this, SIGNAL(renderPosition(int)));
186         connect(render, SIGNAL(durationChanged(int)), this, SIGNAL(durationChanged(int)));
187         connect(m_ruler, SIGNAL(zoneChanged(QPoint)), this, SIGNAL(zoneUpdated(QPoint)));
188     } else {
189         connect(m_ruler, SIGNAL(zoneChanged(QPoint)), this, SLOT(setClipZone(QPoint)));
190     }
191
192     if (m_monitorRefresh) m_monitorRefresh->show();
193
194     if (name == "project") {
195         m_effectWidget = new MonitorEditWidget(render, m_videoBox);
196         m_toolbar->addAction(m_effectWidget->getVisibilityAction());
197         lay->addWidget(m_effectWidget);
198         m_effectWidget->hide();
199     }
200
201     QWidget *spacer = new QWidget(this);
202     spacer->setSizePolicy(QSizePolicy::MinimumExpanding, QSizePolicy::Preferred);
203     m_toolbar->addWidget(spacer);
204     m_timePos = new TimecodeDisplay(m_monitorManager->timecode(), this);
205     m_toolbar->addWidget(m_timePos);
206     connect(m_timePos, SIGNAL(editingFinished()), this, SLOT(slotSeek()));
207     m_toolbar->setMaximumHeight(s * 1.5);
208     layout->addWidget(m_toolbar);
209 }
210
211 Monitor::~Monitor()
212 {
213     delete m_ruler;
214     delete m_timePos;
215     delete m_overlay;
216     if (m_effectWidget)
217         delete m_effectWidget;
218     delete m_monitorRefresh;
219     delete render;
220 }
221
222 QWidget *Monitor::container()
223 {
224     return m_videoBox;
225 }
226
227 const QString Monitor::name() const
228 {
229     return m_name;
230 }
231
232 #if defined(Q_WS_MAC) || defined(USE_OPEN_GL)
233 bool Monitor::createOpenGlWidget(QWidget *parent, const QString profile)
234 {
235     render = new Render(m_name, 0, profile, this);
236     m_glWidget = new VideoGLWidget(parent);
237     if (m_glWidget == NULL) {
238         // Creation failed, we are in trouble...
239         return false;
240     }
241     m_glWidget->setImageAspectRatio(render->dar());
242     m_glWidget->setBackgroundColor(KdenliveSettings::window_background());
243     connect(render, SIGNAL(showImageSignal(QImage)), m_glWidget, SLOT(showImage(QImage)));
244     return true;
245 }
246 #endif
247
248 void Monitor::setupMenu(QMenu *goMenu, QAction *playZone, QAction *loopZone, QMenu *markerMenu, QAction *loopClip)
249 {
250     m_contextMenu = new QMenu(this);
251     m_contextMenu->addMenu(m_playMenu);
252     if (goMenu)
253         m_contextMenu->addMenu(goMenu);
254     if (markerMenu) {
255         m_contextMenu->addMenu(markerMenu);
256         QList <QAction *>list = markerMenu->actions();
257         for (int i = 0; i < list.count(); i++) {
258             if (list.at(i)->data().toString() == "edit_marker") {
259                 m_editMarker = list.at(i);
260                 break;
261             }
262         }
263     }
264
265     m_playMenu->addAction(playZone);
266     m_playMenu->addAction(loopZone);
267     if (loopClip) {
268         m_loopClipAction = loopClip;
269         m_playMenu->addAction(loopClip);
270     }
271
272     //TODO: add save zone to timeline monitor when fixed
273     if (m_name == "clip") {
274         m_contextMenu->addMenu(m_markerMenu);
275         m_contextMenu->addAction(KIcon("document-save"), i18n("Save zone"), this, SLOT(slotSaveZone()));
276     }
277     QAction *extractFrame = m_configMenu->addAction(KIcon("document-new"), i18n("Extract frame"), this, SLOT(slotExtractCurrentFrame()));
278     m_contextMenu->addAction(extractFrame);
279
280     if (m_name != "clip") {
281         QAction *splitView = m_contextMenu->addAction(KIcon("view-split-left-right"), i18n("Split view"), render, SLOT(slotSplitView(bool)));
282         splitView->setCheckable(true);
283         m_configMenu->addAction(splitView);
284     } else {
285         QAction *setThumbFrame = m_contextMenu->addAction(KIcon("document-new"), i18n("Set current image as thumbnail"), this, SLOT(slotSetThumbFrame()));
286         m_configMenu->addAction(setThumbFrame);
287     }
288
289     QAction *showTips = m_contextMenu->addAction(KIcon("help-hint"), i18n("Monitor overlay infos"));
290     showTips->setCheckable(true);
291     connect(showTips, SIGNAL(toggled(bool)), this, SLOT(slotSwitchMonitorInfo(bool)));
292     showTips->setChecked(KdenliveSettings::displayMonitorInfo());
293
294     QAction *dropFrames = m_contextMenu->addAction(KIcon(), i18n("Real time (drop frames)"));
295     dropFrames->setCheckable(true);
296     dropFrames->setChecked(true);
297     connect(dropFrames, SIGNAL(toggled(bool)), this, SLOT(slotSwitchDropFrames(bool)));
298
299     m_configMenu->addAction(showTips);
300     m_configMenu->addAction(dropFrames);
301
302 }
303
304 void Monitor::slotGoToMarker(QAction *action)
305 {
306     int pos = action->data().toInt();
307     slotSeek(pos);
308 }
309
310 void Monitor::slotSetSizeOneToOne()
311 {
312     QRect r = QApplication::desktop()->screenGeometry();
313     const int maxWidth = r.width() - 20;
314     const int maxHeight = r.height() - 20;
315     int width = render->renderWidth();
316     int height = render->renderHeight();
317     kDebug() << "// render info: " << width << "x" << height;
318     while (width >= maxWidth || height >= maxHeight) {
319         width = width * 0.8;
320         height = height * 0.8;
321     }
322     kDebug() << "// MONITOR; set SIZE: " << width << ", " << height;
323     m_videoBox->setFixedSize(width, height);
324     updateGeometry();
325     adjustSize();
326     //m_ui.video_frame->setMinimumSize(0, 0);
327     emit adjustMonitorSize();
328 }
329
330 void Monitor::slotSetSizeOneToTwo()
331 {
332     QRect r = QApplication::desktop()->screenGeometry();
333     const int maxWidth = r.width() - 20;
334     const int maxHeight = r.height() - 20;
335     int width = render->renderWidth() / 2;
336     int height = render->renderHeight() / 2;
337     kDebug() << "// render info: " << width << "x" << height;
338     while (width >= maxWidth || height >= maxHeight) {
339         width = width * 0.8;
340         height = height * 0.8;
341     }
342     kDebug() << "// MONITOR; set SIZE: " << width << ", " << height;
343     m_videoBox->setFixedSize(width, height);
344     updateGeometry();
345     adjustSize();
346     //m_ui.video_frame->setMinimumSize(0, 0);
347     emit adjustMonitorSize();
348 }
349
350 void Monitor::resetSize()
351 {
352     m_videoBox->setMinimumSize(0, 0);
353 }
354
355 DocClipBase *Monitor::activeClip()
356 {
357     return m_currentClip;
358 }
359
360 void Monitor::updateMarkers(DocClipBase *source)
361 {
362     if (source == m_currentClip && source != NULL) {
363         m_markerMenu->clear();
364         QList <CommentedTime> markers = m_currentClip->commentedSnapMarkers();
365         if (!markers.isEmpty()) {
366             QList <int> marks;
367             for (int i = 0; i < markers.count(); i++) {
368                 int pos = (int) markers.at(i).time().frames(m_monitorManager->timecode().fps());
369                 marks.append(pos);
370                 QString position = m_monitorManager->timecode().getTimecode(markers.at(i).time()) + ' ' + markers.at(i).comment();
371                 QAction *go = m_markerMenu->addAction(position);
372                 go->setData(pos);
373             }
374             m_ruler->setMarkers(marks);
375         } else m_ruler->setMarkers(QList <int>());
376         m_markerMenu->setEnabled(!m_markerMenu->isEmpty());
377     }
378 }
379
380 void Monitor::slotSeekToPreviousSnap()
381 {
382     if (m_currentClip) slotSeek(getSnapForPos(true).frames(m_monitorManager->timecode().fps()));
383 }
384
385 void Monitor::slotSeekToNextSnap()
386 {
387     if (m_currentClip) slotSeek(getSnapForPos(false).frames(m_monitorManager->timecode().fps()));
388 }
389
390 GenTime Monitor::position()
391 {
392     return render->seekPosition();
393 }
394
395 GenTime Monitor::getSnapForPos(bool previous)
396 {
397     QList <GenTime> snaps;
398     QList < GenTime > markers = m_currentClip->snapMarkers();
399     for (int i = 0; i < markers.size(); ++i) {
400         GenTime t = markers.at(i);
401         snaps.append(t);
402     }
403     QPoint zone = m_ruler->zone();
404     snaps.append(GenTime(zone.x(), m_monitorManager->timecode().fps()));
405     snaps.append(GenTime(zone.y(), m_monitorManager->timecode().fps()));
406     snaps.append(GenTime());
407     snaps.append(m_currentClip->duration());
408     qSort(snaps);
409
410     const GenTime pos = render->seekPosition();
411     for (int i = 0; i < snaps.size(); ++i) {
412         if (previous && snaps.at(i) >= pos) {
413             if (i == 0) i = 1;
414             return snaps.at(i - 1);
415         } else if (!previous && snaps.at(i) > pos) {
416             return snaps.at(i);
417         }
418     }
419     return GenTime();
420 }
421
422 void Monitor::slotZoneMoved(int start, int end)
423 {
424     m_ruler->setZone(start, end);
425     checkOverlay();
426     setClipZone(m_ruler->zone());
427 }
428
429 void Monitor::slotSetZoneStart()
430 {
431     m_ruler->setZone(m_ruler->position(), -1);
432     emit zoneUpdated(m_ruler->zone());
433     checkOverlay();
434     setClipZone(m_ruler->zone());
435 }
436
437 void Monitor::slotSetZoneEnd()
438 {
439     m_ruler->setZone(-1, m_ruler->position());
440     emit zoneUpdated(m_ruler->zone());
441     checkOverlay();
442     setClipZone(m_ruler->zone());
443 }
444
445 // virtual
446 void Monitor::mousePressEvent(QMouseEvent * event)
447 {
448     if (event->button() != Qt::RightButton) {
449         if (m_videoBox->underMouse() && (!m_overlay || !m_overlay->underMouse())) {
450             m_dragStarted = true;
451             m_DragStartPosition = event->pos();
452         }
453     } else if (!m_effectWidget || !m_effectWidget->isVisible()) {
454         m_contextMenu->popup(event->globalPos());
455     }
456 }
457
458 void Monitor::resizeEvent(QResizeEvent *event)
459 {
460     Q_UNUSED(event);
461     if (render && isVisible() && isActive()) render->doRefresh();
462 }
463
464
465 void Monitor::slotSwitchFullScreen()
466 {
467     m_videoBox->switchFullScreen();
468 }
469
470 // virtual
471 void Monitor::mouseReleaseEvent(QMouseEvent * event)
472 {
473     if (m_dragStarted) {
474         if (m_videoBox->underMouse() && (!m_effectWidget || !m_effectWidget->isVisible())) {
475             if (isActive()) slotPlay();
476             else activateMonitor();
477         } else QWidget::mouseReleaseEvent(event);
478         m_dragStarted = false;
479     }
480 }
481
482 // virtual
483 void Monitor::mouseMoveEvent(QMouseEvent *event)
484 {
485     // kDebug() << "// DRAG STARTED, MOUSE MOVED: ";
486     if (!m_dragStarted || m_currentClip == NULL) return;
487
488     if ((event->pos() - m_DragStartPosition).manhattanLength()
489             < QApplication::startDragDistance())
490         return;
491
492     {
493         QDrag *drag = new QDrag(this);
494         QMimeData *mimeData = new QMimeData;
495
496         QStringList list;
497         list.append(m_currentClip->getId());
498         QPoint p = m_ruler->zone();
499         list.append(QString::number(p.x()));
500         list.append(QString::number(p.y()));
501         QByteArray data;
502         data.append(list.join(";").toUtf8());
503         mimeData->setData("kdenlive/clip", data);
504         drag->setMimeData(mimeData);
505         QPixmap pix = m_currentClip->thumbnail();
506         drag->setPixmap(pix);
507         drag->setHotSpot(QPoint(0, 50));
508         drag->start(Qt::MoveAction);
509
510         //Qt::DropAction dropAction;
511         //dropAction = drag->start(Qt::CopyAction | Qt::MoveAction);
512
513         //Qt::DropAction dropAction = drag->exec();
514
515     }
516     //event->accept();
517 }
518
519 /*void Monitor::dragMoveEvent(QDragMoveEvent * event) {
520     event->setDropAction(Qt::IgnoreAction);
521     event->setDropAction(Qt::MoveAction);
522     if (event->mimeData()->hasText()) {
523         event->acceptProposedAction();
524     }
525 }
526
527 Qt::DropActions Monitor::supportedDropActions() const {
528     // returns what actions are supported when dropping
529     return Qt::MoveAction;
530 }*/
531
532 QStringList Monitor::mimeTypes() const
533 {
534     QStringList qstrList;
535     // list of accepted mime types for drop
536     qstrList.append("kdenlive/clip");
537     return qstrList;
538 }
539
540 // virtual
541 void Monitor::wheelEvent(QWheelEvent * event)
542 {
543     slotMouseSeek(event->delta(), event->modifiers() == Qt::ControlModifier);
544     event->accept();
545 }
546
547 void Monitor::slotMouseSeek(int eventDelta, bool fast)
548 {
549     if (fast) {
550         int delta = m_monitorManager->timecode().fps();
551         if (eventDelta > 0) delta = 0 - delta;
552         slotSeek(m_ruler->position() - delta);
553     } else {
554         if (eventDelta >= 0) slotForwardOneFrame();
555         else slotRewindOneFrame();
556     }
557 }
558
559 void Monitor::slotSetThumbFrame()
560 {
561     if (m_currentClip == NULL) {
562         return;
563     }
564     m_currentClip->setClipThumbFrame((uint) render->seekFramePosition());
565     emit refreshClipThumbnail(m_currentClip->getId(), true);
566 }
567
568 void Monitor::slotExtractCurrentFrame()
569 {
570     QImage frame;
571     // check if we are using a proxy
572     if (m_currentClip && !m_currentClip->getProperty("proxy").isEmpty() && m_currentClip->getProperty("proxy") != "-") {
573         // using proxy, use original clip url to get frame
574         frame = render->extractFrame(render->seekFramePosition(), m_currentClip->fileURL().path());
575     }
576     else frame = render->extractFrame(render->seekFramePosition());
577     KFileDialog *fs = new KFileDialog(KUrl(), "image/png", this);
578     fs->setOperationMode(KFileDialog::Saving);
579     fs->setMode(KFile::File);
580 #if KDE_IS_VERSION(4,2,0)
581     fs->setConfirmOverwrite(true);
582 #endif
583     fs->setKeepLocation(true);
584     fs->exec();
585     QString path = fs->selectedFile();
586     delete fs;
587     if (!path.isEmpty()) {
588         frame.save(path);
589     }
590 }
591
592 bool Monitor::isActive() const
593 {
594     return m_monitorManager->isActive(m_name);
595 }
596
597 bool Monitor::activateMonitor()
598 {
599     return m_monitorManager->activateMonitor(m_name);
600 }
601
602 void Monitor::setTimePos(const QString &pos)
603 {
604     m_timePos->setValue(pos);
605     slotSeek();
606 }
607
608 void Monitor::slotSeek()
609 {
610     slotSeek(m_timePos->getValue());
611 }
612
613 void Monitor::slotSeek(int pos)
614 {
615     if (render == NULL) return;
616     activateMonitor();
617     render->seekToFrame(pos);
618 }
619
620 void Monitor::checkOverlay()
621 {
622     if (m_overlay == NULL) return;
623     int pos = m_ruler->position();
624     QPoint zone = m_ruler->zone();
625     if (pos == zone.x())
626         m_overlay->setOverlayText(i18n("In Point"));
627     else if (pos == zone.y())
628         m_overlay->setOverlayText(i18n("Out Point"));
629     else {
630         if (m_currentClip) {
631             QString markerComment = m_currentClip->markerComment(GenTime(pos, m_monitorManager->timecode().fps()));
632             if (markerComment.isEmpty())
633                 m_overlay->setHidden(true);
634             else
635                 m_overlay->setOverlayText(markerComment, false);
636         } else m_overlay->setHidden(true);
637     }
638 }
639
640 void Monitor::slotStart()
641 {
642     activateMonitor();
643     render->play(0);
644     render->seekToFrame(0);
645 }
646
647 void Monitor::slotEnd()
648 {
649     activateMonitor();
650     render->play(0);
651     render->seekToFrame(render->getLength());
652 }
653
654 void Monitor::slotZoneStart()
655 {
656     activateMonitor();
657     render->play(0);
658     render->seekToFrame(m_ruler->zone().x());
659 }
660
661 void Monitor::slotZoneEnd()
662 {
663     activateMonitor();
664     render->play(0);
665     render->seekToFrame(m_ruler->zone().y());
666 }
667
668 void Monitor::slotRewind(double speed)
669 {
670     activateMonitor();
671     if (speed == 0) {
672         double currentspeed = render->playSpeed();
673         if (currentspeed >= 0) render->play(-2);
674         else render->play(currentspeed * 2);
675     } else render->play(speed);
676     //m_playAction->setChecked(true);
677     m_playAction->setIcon(m_pauseIcon);
678 }
679
680 void Monitor::slotForward(double speed)
681 {
682     activateMonitor();
683     if (speed == 0) {
684         double currentspeed = render->playSpeed();
685         if (currentspeed <= 1) render->play(2);
686         else render->play(currentspeed * 2);
687     } else render->play(speed);
688     //m_playAction->setChecked(true);
689     m_playAction->setIcon(m_pauseIcon);
690 }
691
692 void Monitor::slotRewindOneFrame(int diff)
693 {
694     activateMonitor();
695     render->play(0);
696     render->seekToFrameDiff(-diff);
697 }
698
699 void Monitor::slotForwardOneFrame(int diff)
700 {
701     activateMonitor();
702     render->play(0);
703     render->seekToFrameDiff(diff);
704 }
705
706 void Monitor::seekCursor(int pos)
707 {
708     //activateMonitor();
709     if (m_ruler->slotNewValue(pos)) {
710         checkOverlay();
711         m_timePos->setValue(pos);
712     }
713 }
714
715 void Monitor::rendererStopped(int pos)
716 {
717     if (m_ruler->slotNewValue(pos)) {
718         checkOverlay();
719         m_timePos->setValue(pos);
720     }
721     m_playAction->setIcon(m_playIcon);
722 }
723
724 void Monitor::adjustRulerSize(int length)
725 {
726     if (length > 0) m_length = length;
727     m_ruler->adjustScale(m_length);
728     if (m_currentClip != NULL) {
729         QPoint zone = m_currentClip->zone();
730         m_ruler->setZone(zone.x(), zone.y());
731     }
732 }
733
734 void Monitor::stop()
735 {
736     disconnect(render, SIGNAL(rendererPosition(int)), this, SLOT(seekCursor(int)));
737     if (render) render->stop();
738 }
739
740 void Monitor::start()
741 {
742     if (!isVisible()) return;
743     if (render) render->start();
744     connect(render, SIGNAL(rendererPosition(int)), this, SLOT(seekCursor(int)));
745 }
746
747 void Monitor::refreshMonitor(bool visible)
748 {
749     if (visible && render) {
750         if (!activateMonitor()) {
751             // the monitor was already active, simply refreshClipThumbnail
752             render->doRefresh();
753         }
754     }
755 }
756
757 void Monitor::refreshMonitor()
758 {
759     if (isActive()) {
760         render->doRefresh();
761     }
762 }
763
764 void Monitor::pause()
765 {
766     if (render == NULL) return;
767     activateMonitor();
768     render->pause();
769     //m_playAction->setChecked(true);
770     m_playAction->setIcon(m_playIcon);
771 }
772
773 void Monitor::slotPlay()
774 {
775     if (render == NULL) return;
776     activateMonitor();
777     if (render->playSpeed() == 0.0) {
778         m_playAction->setIcon(m_pauseIcon);
779         render->switchPlay(true);
780     } else {
781         m_playAction->setIcon(m_playIcon);
782         render->switchPlay(false);
783     }
784 }
785
786 void Monitor::slotPlayZone()
787 {
788     if (render == NULL) return;
789     activateMonitor();
790     QPoint p = m_ruler->zone();
791     render->playZone(GenTime(p.x(), m_monitorManager->timecode().fps()), GenTime(p.y(), m_monitorManager->timecode().fps()));
792     //m_playAction->setChecked(true);
793     m_playAction->setIcon(m_pauseIcon);
794 }
795
796 void Monitor::slotLoopZone()
797 {
798     if (render == NULL) return;
799     activateMonitor();
800     QPoint p = m_ruler->zone();
801     render->loopZone(GenTime(p.x(), m_monitorManager->timecode().fps()), GenTime(p.y(), m_monitorManager->timecode().fps()));
802     //m_playAction->setChecked(true);
803     m_playAction->setIcon(m_pauseIcon);
804 }
805
806 void Monitor::slotLoopClip()
807 {
808     if (render == NULL || m_selectedClip == NULL)
809         return;
810     activateMonitor();
811     render->loopZone(m_selectedClip->startPos(), m_selectedClip->endPos());
812     //m_playAction->setChecked(true);
813     m_playAction->setIcon(m_pauseIcon);
814 }
815
816 void Monitor::updateClipProducer(Mlt::Producer *prod)
817 {
818     if (render == NULL) return;
819    render->setProducer(prod, render->seekFramePosition());
820 }
821
822 void Monitor::slotSetClipProducer(DocClipBase *clip, QPoint zone, int position)
823 {
824     if (render == NULL) return;
825     if (clip == NULL && m_currentClip != NULL) {
826         kDebug()<<"// SETTING NULL CLIP MONITOR";
827         m_currentClip = NULL;
828         m_length = -1;
829         render->setProducer(NULL, -1);
830         return;
831     }
832     
833     if (clip != m_currentClip) {
834         m_currentClip = clip;
835         if (m_currentClip) activateMonitor();
836         updateMarkers(clip);
837         Mlt::Producer *prod = NULL;
838         if (clip) prod = clip->getCloneProducer();
839         if (render->setProducer(prod, position) == -1) {
840             // MLT CONSUMER is broken
841             kDebug(QtWarningMsg) << "ERROR, Cannot start monitor";
842         }
843     } else {
844         if (m_currentClip) {
845             activateMonitor();
846             if (position == -1) position = render->seekFramePosition();
847             render->seek(position);
848         }
849     }
850     if (!zone.isNull()) {
851         m_ruler->setZone(zone.x(), zone.y());
852         render->seek(zone.x());
853     }
854 }
855
856 void Monitor::slotOpenFile(const QString &file)
857 {
858     if (render == NULL) return;
859     activateMonitor();
860     QDomDocument doc;
861     QDomElement mlt = doc.createElement("mlt");
862     doc.appendChild(mlt);
863     QDomElement prod = doc.createElement("producer");
864     mlt.appendChild(prod);
865     prod.setAttribute("mlt_service", "avformat");
866     prod.setAttribute("resource", file);
867     render->setSceneList(doc, 0);
868 }
869
870 void Monitor::slotSaveZone()
871 {
872     if (render == NULL) return;
873     emit saveZone(render, m_ruler->zone(), m_currentClip);
874
875     //render->setSceneList(doc, 0);
876 }
877
878 void Monitor::resetProfile(const QString &profile)
879 {
880     m_timePos->updateTimeCode(m_monitorManager->timecode());
881     if (render == NULL) return;
882     if (!render->hasProfile(profile)) {
883         activateMonitor();
884         render->resetProfile(profile);
885     }
886     if (m_effectWidget)
887         m_effectWidget->resetProfile(render);
888 }
889
890 void Monitor::saveSceneList(QString path, QDomElement info)
891 {
892     if (render == NULL) return;
893     render->saveSceneList(path, info);
894 }
895
896 const QString Monitor::sceneList()
897 {
898     if (render == NULL) return QString();
899     return render->sceneList();
900 }
901
902 void Monitor::setClipZone(QPoint pos)
903 {
904     if (m_currentClip == NULL) return;
905     m_currentClip->setZone(pos);
906 }
907
908 void Monitor::slotSwitchDropFrames(bool show)
909 {
910     render->setDropFrames(show);
911 }
912
913 void Monitor::slotSwitchMonitorInfo(bool show)
914 {
915     KdenliveSettings::setDisplayMonitorInfo(show);
916     if (show) {
917         if (m_overlay) return;
918         if (m_monitorRefresh == NULL) {
919             // Using OpenGL display
920 #if defined(Q_WS_MAC) || defined(USE_OPEN_GL)
921             if (m_glWidget->layout()) delete m_glWidget->layout();
922             m_overlay = new Overlay();
923             connect(m_overlay, SIGNAL(editMarker()), this, SLOT(slotEditMarker()));
924             QVBoxLayout *layout = new QVBoxLayout;
925             layout->addStretch(10);
926             layout->addWidget(m_overlay);
927             m_glWidget->setLayout(layout);
928 #endif
929         } else {
930             if (m_monitorRefresh->layout()) delete m_monitorRefresh->layout();
931             m_overlay = new Overlay();
932             connect(m_overlay, SIGNAL(editMarker()), this, SLOT(slotEditMarker()));
933             QVBoxLayout *layout = new QVBoxLayout;
934             layout->addStretch(10);
935             layout->addWidget(m_overlay);
936             m_monitorRefresh->setLayout(layout);
937             m_overlay->raise();
938             m_overlay->setHidden(true);
939         }
940         checkOverlay();
941     } else {
942         delete m_overlay;
943         m_overlay = NULL;
944     }
945 }
946
947 void Monitor::slotEditMarker()
948 {
949     if (m_editMarker) m_editMarker->trigger();
950 }
951
952 void Monitor::updateTimecodeFormat()
953 {
954     m_timePos->slotUpdateTimeCodeFormat();
955 }
956
957 QStringList Monitor::getZoneInfo() const
958 {
959     QStringList result;
960     if (m_currentClip == NULL) return result;
961     result << m_currentClip->getId();
962     QPoint zone = m_ruler->zone();
963     result << QString::number(zone.x()) << QString::number(zone.y());
964     return result;
965 }
966
967 void Monitor::slotSetSelectedClip(AbstractClipItem* item)
968 {
969     if (item) {
970         m_loopClipAction->setEnabled(true);
971         m_selectedClip = item;
972     } else {
973         m_loopClipAction->setEnabled(false);
974     }
975 }
976
977 void Monitor::slotSetSelectedClip(ClipItem* item)
978 {
979     if (item || (!item && !m_loopClipTransition)) {
980         m_loopClipTransition = false;
981         slotSetSelectedClip((AbstractClipItem*)item);
982     }
983 }
984
985 void Monitor::slotSetSelectedClip(Transition* item)
986 {
987     if (item || (!item && m_loopClipTransition)) {
988         m_loopClipTransition = true;
989         slotSetSelectedClip((AbstractClipItem*)item);
990     }
991 }
992
993
994 void Monitor::slotEffectScene(bool show)
995 {
996     if (m_name == "project") {
997         if (m_monitorRefresh) {
998             m_monitorRefresh->setVisible(!show);
999         } else {
1000 #if defined(Q_WS_MAC) || defined(USE_OPEN_GL)
1001             m_glWidget->setVisible(!show);
1002 #endif
1003         }
1004         m_effectWidget->setVisible(show);
1005         m_effectWidget->getVisibilityAction()->setChecked(show);
1006         emit requestFrameForAnalysis(show);
1007         if (show) {
1008             m_effectWidget->getScene()->slotZoomFit();
1009         }
1010         render->doRefresh();
1011     }
1012 }
1013
1014 MonitorEditWidget* Monitor::getEffectEdit()
1015 {
1016     return m_effectWidget;
1017 }
1018
1019 bool Monitor::effectSceneDisplayed()
1020 {
1021     return m_effectWidget->isVisible();
1022 }
1023
1024 void Monitor::slotSetVolume(int volume)
1025 {
1026     KdenliveSettings::setVolume(volume);
1027     KIcon icon;
1028     if (volume == 0) icon = KIcon("audio-volume-muted");
1029     else icon = KIcon("audio-volume-medium");
1030     static_cast <QToolButton *>(m_volumeWidget)->setIcon(icon);
1031     render->slotSetVolume(volume);
1032 }
1033
1034 void Monitor::slotShowVolume()
1035 {
1036     m_volumePopup->move(mapToGlobal(m_toolbar->geometry().topLeft()) + QPoint(mapToParent(m_volumeWidget->geometry().bottomLeft()).x(), -m_volumePopup->height()));
1037     int vol = render->volume();
1038     // Disable widget if we cannot get the volume
1039     m_volumePopup->setEnabled(vol != -1);
1040     m_audioSlider->blockSignals(true);
1041     m_audioSlider->setValue(vol);
1042     m_audioSlider->blockSignals(false);
1043     m_volumePopup->show();
1044 }
1045
1046 AbstractRender *Monitor::abstractRender()
1047 {
1048     return render;
1049 }
1050
1051 MonitorRefresh::MonitorRefresh(QWidget* parent) :
1052     QWidget(parent)
1053     , m_renderer(NULL)
1054 {
1055     // MonitorRefresh is used as container for the SDL display (it's window id is passed to SDL)
1056     setAttribute(Qt::WA_PaintOnScreen);
1057     setAttribute(Qt::WA_OpaquePaintEvent);
1058     setSizePolicy(QSizePolicy::MinimumExpanding, QSizePolicy::MinimumExpanding);
1059     //setAttribute(Qt::WA_NoSystemBackground);
1060 }
1061
1062 void MonitorRefresh::setRenderer(Render* render)
1063 {
1064     m_renderer = render;
1065 }
1066
1067
1068 Overlay::Overlay(QWidget* parent) :
1069     QLabel(parent)
1070 {
1071     //setAttribute(Qt::WA_TransparentForMouseEvents);
1072     setAutoFillBackground(true);
1073     setBackgroundRole(QPalette::Base);
1074     setSizePolicy(QSizePolicy::Preferred, QSizePolicy::Preferred);
1075     setCursor(Qt::PointingHandCursor);
1076 }
1077
1078 // virtual
1079 void Overlay::mouseReleaseEvent ( QMouseEvent * event )
1080 {
1081     event->accept();
1082 }
1083
1084 // virtual
1085 void Overlay::mousePressEvent( QMouseEvent * event )
1086 {
1087     event->accept();
1088 }
1089
1090 // virtual
1091 void Overlay::mouseDoubleClickEvent ( QMouseEvent * event )
1092 {
1093     emit editMarker();
1094     event->accept();
1095 }
1096
1097 void Overlay::setOverlayText(const QString &text, bool isZone)
1098 {
1099     setHidden(true);
1100     m_isZone = isZone;
1101     QPalette p;
1102     p.setColor(QPalette::Text, Qt::white);
1103     if (m_isZone) p.setColor(QPalette::Base, QColor(200, 0, 0));
1104     else p.setColor(QPalette::Base, QColor(0, 0, 200));
1105     setPalette(p);
1106     setText(' ' + text + ' ');
1107     setHidden(false);
1108     update();
1109 }
1110
1111 VideoContainer::VideoContainer(Monitor* parent) :
1112     QFrame()
1113     , m_monitor(parent)
1114 {
1115     setFrameShape(QFrame::NoFrame);
1116     setFocusPolicy(Qt::ClickFocus);
1117     setSizePolicy(QSizePolicy::MinimumExpanding, QSizePolicy::MinimumExpanding);
1118 }
1119
1120 // virtual
1121 void VideoContainer::mousePressEvent(QMouseEvent * event)
1122 {
1123     if (m_monitor->underMouse()) event->setAccepted(false);
1124 }
1125
1126 // virtual
1127 void VideoContainer::mouseReleaseEvent(QMouseEvent * event)
1128 {
1129     if (m_monitor->underMouse()) event->setAccepted(false);
1130     else {
1131         if (m_monitor->isActive()) {
1132             m_monitor->slotPlay();
1133             event->accept();
1134         }
1135     }
1136 }
1137
1138 // virtual
1139 void VideoContainer::mouseMoveEvent(QMouseEvent *event)
1140 {
1141     if (m_monitor->underMouse()) event->setAccepted(false);
1142 }
1143
1144 // virtual
1145 void VideoContainer::keyPressEvent(QKeyEvent *event)
1146 {
1147     // Exit fullscreen with Esc key
1148     if (event->key() == Qt::Key_Escape && isFullScreen()) {
1149         switchFullScreen();
1150         event->setAccepted(true);
1151     } else event->setAccepted(false);
1152 }
1153
1154 // virtual
1155 void VideoContainer::wheelEvent(QWheelEvent * event)
1156 {
1157     if (m_monitor->underMouse()) event->setAccepted(false);
1158     else {
1159         m_monitor->slotMouseSeek(event->delta(), event->modifiers() == Qt::ControlModifier);
1160         event->accept();
1161     }
1162 }
1163
1164 void VideoContainer::mouseDoubleClickEvent(QMouseEvent * event)
1165 {
1166     Q_UNUSED(event)
1167
1168     if (!KdenliveSettings::openglmonitors())
1169         switchFullScreen();
1170 }
1171
1172 void VideoContainer::switchFullScreen()
1173 {
1174     // TODO: disable screensaver?
1175     Qt::WindowFlags flags = windowFlags();
1176     if (!isFullScreen()) {
1177         // Check if we ahave a multiple monitor setup
1178         setUpdatesEnabled(false);
1179 #if QT_VERSION >= 0x040600
1180         int monitors = QApplication::desktop()->screenCount();
1181 #else
1182         int monitors = QApplication::desktop()->numScreens();
1183 #endif
1184         if (monitors > 1) {
1185             QRect screenres;
1186             // Move monitor widget to the second screen (one screen for Kdenlive, the other one for the Monitor widget
1187             int currentScreen = QApplication::desktop()->screenNumber(this);
1188             if (currentScreen < monitors - 1)
1189                 screenres = QApplication::desktop()->screenGeometry(currentScreen + 1);
1190             else
1191                 screenres = QApplication::desktop()->screenGeometry(currentScreen - 1);
1192             move(QPoint(screenres.x(), screenres.y()));
1193             resize(screenres.width(), screenres.height());
1194         }
1195
1196         m_baseFlags = flags & (Qt::Window | Qt::SubWindow);
1197         flags |= Qt::Window;
1198         flags ^= Qt::SubWindow;
1199         setWindowFlags(flags);
1200 #ifdef Q_WS_X11
1201         // This works around a bug with Compiz
1202         // as the window must be visible before we can set the state
1203         show();
1204         raise();
1205         setWindowState(windowState() | Qt::WindowFullScreen);   // set
1206 #else
1207         setWindowState(windowState() | Qt::WindowFullScreen);   // set
1208         setUpdatesEnabled(true);
1209         show();
1210 #endif
1211     } else {
1212         setUpdatesEnabled(false);
1213         flags ^= (Qt::Window | Qt::SubWindow); //clear the flags...
1214         flags |= m_baseFlags; //then we reset the flags (window and subwindow)
1215         setWindowFlags(flags);
1216         setWindowState(windowState()  ^ Qt::WindowFullScreen);   // reset
1217         setUpdatesEnabled(true);
1218         show();
1219     }
1220     m_monitor->pause();
1221 }
1222
1223
1224 #include "monitor.moc"