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