]> git.sesse.net Git - kdenlive/blob - src/recmonitor.cpp
Some update for decklink capture & use mlt_threads for proxies
[kdenlive] / src / recmonitor.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 "recmonitor.h"
22 #include "gentime.h"
23 #include "mltdevicecapture.h"
24 #include "kdenlivesettings.h"
25 #include "managecapturesdialog.h"
26 #include "monitormanager.h"
27 #include "monitor.h"
28 #include "profilesdialog.h"
29
30 #include <KDebug>
31 #include <KLocale>
32 #include <KStandardDirs>
33 #include <KComboBox>
34 #include <KIO/NetAccess>
35 #include <KFileItem>
36 #include <KMessageBox>
37 #include <KApplication>
38
39 #if KDE_IS_VERSION(4,2,0)
40 #include <KDiskFreeSpaceInfo>
41 #endif
42
43 #include <QMouseEvent>
44 #include <QMenu>
45 #include <QToolButton>
46 #include <QFile>
47 #include <QDir>
48
49
50 RecMonitor::RecMonitor(QString name, MonitorManager *manager, QWidget *parent) :
51     AbstractMonitor(parent),
52     m_name(name),
53     m_isCapturing(false),
54     m_didCapture(false),
55     m_isPlaying(false),
56     m_manager(manager),
57     m_captureDevice(NULL),
58     m_analyse(false)
59 {
60     setupUi(this);
61
62     video_frame->setAttribute(Qt::WA_PaintOnScreen);
63     device_selector->setCurrentIndex(KdenliveSettings::defaultcapture());
64     connect(device_selector, SIGNAL(currentIndexChanged(int)), this, SLOT(slotVideoDeviceChanged(int)));
65
66     // Video widget holder
67     QVBoxLayout *l = new QVBoxLayout;
68     l->setContentsMargins(0, 0, 0, 0);
69     l->setSpacing(0);
70     m_videoBox = new VideoPreviewContainer();
71     m_videoBox->setContentsMargins(0, 0, 0, 0);
72     m_videoBox->setSizePolicy(QSizePolicy::Preferred, QSizePolicy::Preferred);
73     l->addWidget(m_videoBox);
74     video_frame->setLayout(l);
75
76     QToolBar *toolbar = new QToolBar(this);
77     QHBoxLayout *layout = new QHBoxLayout;
78     layout->setContentsMargins(0, 0, 0, 0);
79     m_playIcon = KIcon("media-playback-start");
80     m_pauseIcon = KIcon("media-playback-pause");
81
82     m_discAction = toolbar->addAction(KIcon("network-connect"), i18n("Connect"));
83     connect(m_discAction, SIGNAL(triggered()), this, SLOT(slotDisconnect()));
84
85     m_rewAction = toolbar->addAction(KIcon("media-seek-backward"), i18n("Rewind"));
86     connect(m_rewAction, SIGNAL(triggered()), this, SLOT(slotRewind()));
87
88     m_playAction = toolbar->addAction(m_playIcon, i18n("Play"));
89     connect(m_playAction, SIGNAL(triggered()), this, SLOT(slotStartPreview()));
90
91     m_stopAction = toolbar->addAction(KIcon("media-playback-stop"), i18n("Stop"));
92     connect(m_stopAction, SIGNAL(triggered()), this, SLOT(slotStopCapture()));
93     m_stopAction->setEnabled(false);
94     m_fwdAction = toolbar->addAction(KIcon("media-seek-forward"), i18n("Forward"));
95     connect(m_fwdAction, SIGNAL(triggered()), this, SLOT(slotForward()));
96
97     m_recAction = toolbar->addAction(KIcon("media-record"), i18n("Record"));
98     connect(m_recAction, SIGNAL(triggered()), this, SLOT(slotRecord()));
99     m_recAction->setCheckable(true);
100
101     toolbar->addSeparator();
102
103     QAction *configAction = toolbar->addAction(KIcon("configure"), i18n("Configure"));
104     connect(configAction, SIGNAL(triggered()), this, SLOT(slotConfigure()));
105     configAction->setCheckable(false);
106
107     layout->addWidget(toolbar);
108     layout->addWidget(&m_logger);
109     layout->addWidget(&m_dvinfo);
110     m_logger.setMaxCount(10);
111     m_logger.setSizePolicy(QSizePolicy::Expanding, QSizePolicy::Expanding);
112     m_logger.setFrame(false);
113     //m_logger.setInsertPolicy(QComboBox::InsertAtTop);
114
115 #if KDE_IS_VERSION(4,2,0)
116     m_freeSpace = new KCapacityBar(KCapacityBar::DrawTextInline, this);
117     m_freeSpace->setMaximumWidth(150);
118     QFontMetricsF fontMetrics(font());
119     m_freeSpace->setMaximumHeight(fontMetrics.height() * 1.2);
120     slotUpdateFreeSpace();
121     layout->addWidget(m_freeSpace);
122     connect(&m_spaceTimer, SIGNAL(timeout()), this, SLOT(slotUpdateFreeSpace()));
123     m_spaceTimer.setInterval(30000);
124     m_spaceTimer.setSingleShot(false);
125 #endif
126
127     control_frame_firewire->setLayout(layout);
128     m_displayProcess = new QProcess;
129     m_captureProcess = new QProcess;
130
131     connect(m_captureProcess, SIGNAL(stateChanged(QProcess::ProcessState)), this, SLOT(slotProcessStatus(QProcess::ProcessState)));
132     connect(m_captureProcess, SIGNAL(readyReadStandardError()), this, SLOT(slotReadDvgrabInfo()));
133
134     
135     QString videoDriver = KdenliveSettings::videodrivername();
136 #if QT_VERSION >= 0x040600
137     QProcessEnvironment env = QProcessEnvironment::systemEnvironment();
138     env.insert("SDL_WINDOWID", QString::number(video_frame->winId()));
139     if (!videoDriver.isEmpty()) {
140         if (videoDriver == "x11_noaccel") {
141             env.insert("SDL_VIDEO_YUV_HWACCEL", "0");
142             env.insert("SDL_VIDEODRIVER", "x11");
143         } else env.insert("SDL_VIDEODRIVER", videoDriver);
144     }
145     m_displayProcess->setProcessEnvironment(env);
146 #else
147     QStringList env = QProcess::systemEnvironment();
148     env << "SDL_WINDOWID=" + QString::number(video_frame->winId());
149     if (!videoDriver.isEmpty()) {
150         if (videoDriver == "x11_noaccel") {
151             env << "SDL_VIDEO_YUV_HWACCEL=0";
152             env << "SDL_VIDEODRIVER=x11";
153         } else env << "SDL_VIDEODRIVER=" + videoDriver;
154     }
155     m_displayProcess->setEnvironment(env);
156 #endif
157     setenv("SDL_VIDEO_ALLOW_SCREENSAVER", "1", 1);
158
159     kDebug() << "/////// BUILDING MONITOR, ID: " << video_frame->winId();
160     slotVideoDeviceChanged(device_selector->currentIndex());
161     recording_preview->setToolTip(i18n("Capture preview settings"));
162     recording_preview->setCurrentIndex(KdenliveSettings::recording_preview());
163     connect(recording_preview, SIGNAL(currentIndexChanged(int)), this, SLOT(slotChangeRecordingPreview(int)));
164 }
165
166 RecMonitor::~RecMonitor()
167 {
168 #if KDE_IS_VERSION(4,2,0)
169     m_spaceTimer.stop();
170 #endif
171     delete m_captureProcess;
172     delete m_displayProcess;
173     if (m_captureDevice) delete m_captureDevice;
174 }
175
176 const QString RecMonitor::name() const
177 {
178     return m_name;
179 }
180
181 void RecMonitor::stop()
182 {
183     slotStopCapture();
184 }
185
186 void RecMonitor::start()
187 {
188 }
189
190 void RecMonitor::slotConfigure()
191 {
192     emit showConfigDialog(4, device_selector->currentIndex());
193 }
194
195 void RecMonitor::slotUpdateCaptureFolder(const QString currentProjectFolder)
196 {
197     if (KdenliveSettings::capturetoprojectfolder()) m_capturePath = currentProjectFolder;
198     else m_capturePath = KdenliveSettings::capturefolder();
199     if (m_captureProcess) m_captureProcess->setWorkingDirectory(m_capturePath);
200     if (m_captureProcess->state() != QProcess::NotRunning) {
201         if (device_selector->currentIndex() == FIREWIRE)
202             KMessageBox::information(this, i18n("You need to disconnect and reconnect in the capture monitor to apply your changes"), i18n("Capturing"));
203         else KMessageBox::information(this, i18n("You need to stop capture before your changes can be applied"), i18n("Capturing"));
204     } else slotVideoDeviceChanged(device_selector->currentIndex());
205
206 #if KDE_IS_VERSION(4,2,0)
207     // update free space info
208     slotUpdateFreeSpace();
209 #endif
210 }
211
212 void RecMonitor::slotVideoDeviceChanged(int ix)
213 {
214     QString capturefile;
215     QString capturename;
216     recording_preview->setHidden(ix != VIDEO4LINUX && ix != BLACKMAGIC);
217     m_fwdAction->setVisible(ix == FIREWIRE);
218     m_discAction->setVisible(ix == FIREWIRE);
219     m_rewAction->setVisible(ix == FIREWIRE);
220     m_recAction->setEnabled(ix != FIREWIRE);
221     m_logger.setVisible(ix == BLACKMAGIC);
222     if (m_captureDevice) {
223         // MLT capture still running, abort
224         m_captureDevice->stop();
225         delete m_captureDevice;
226         m_captureDevice = NULL;
227         m_manager->clearScopeSource();
228     }
229
230     // The m_videoBox container has to be shown once before the MLT consumer is build, or preview will fail
231     m_videoBox->setHidden(ix != VIDEO4LINUX && ix != BLACKMAGIC);
232     m_videoBox->setHidden(true);
233     switch (ix) {
234     case SCREENGRAB:
235         m_discAction->setEnabled(false);
236         m_rewAction->setEnabled(false);
237         m_fwdAction->setEnabled(false);
238         m_stopAction->setEnabled(false);
239         m_playAction->setEnabled(false);
240         if (KdenliveSettings::rmd_path().isEmpty()) {
241             QString rmdpath = KStandardDirs::findExe("recordmydesktop");
242             if (rmdpath.isEmpty()) video_frame->setPixmap(mergeSideBySide(KIcon("dialog-warning").pixmap(QSize(50, 50)), i18n("Recordmydesktop utility not found,\n please install it for screen grabs")));
243             else KdenliveSettings::setRmd_path(rmdpath);
244         }
245         if (!KdenliveSettings::rmd_path().isEmpty()) video_frame->setPixmap(mergeSideBySide(KIcon("video-display").pixmap(QSize(50, 50)), i18n("Press record button\nto start screen capture\nFiles will be saved in:\n%1", m_capturePath)));
246         //video_frame->setText(i18n("Press record button\nto start screen capture"));
247         break;
248     case VIDEO4LINUX:
249         m_stopAction->setEnabled(false);
250         m_playAction->setEnabled(true);
251         checkDeviceAvailability();
252         break;
253     case BLACKMAGIC:
254         m_stopAction->setEnabled(false);
255         m_playAction->setEnabled(true);
256         capturefile = m_capturePath;
257         if (!capturefile.endsWith("/")) capturefile.append("/");
258         capturename = KdenliveSettings::decklink_filename();
259         capturename.append("xxx.");
260         capturename.append(KdenliveSettings::decklink_extension());
261         capturefile.append(capturename);
262         video_frame->setPixmap(mergeSideBySide(KIcon("camera-photo").pixmap(QSize(50, 50)), i18n("Plug your camcorder and\npress play button\nto start preview.\nFiles will be saved in:\n%1", capturefile)));
263         break;
264     default: // FIREWIRE
265         m_discAction->setEnabled(true);
266         m_stopAction->setEnabled(false);
267         m_playAction->setEnabled(false);
268         m_rewAction->setEnabled(false);
269         m_fwdAction->setEnabled(false);
270
271         // Check that dvgab is available
272         if (KdenliveSettings::dvgrab_path().isEmpty()) {
273             QString dvgrabpath = KStandardDirs::findExe("dvgrab");
274             if (dvgrabpath.isEmpty()) video_frame->setPixmap(mergeSideBySide(KIcon("dialog-warning").pixmap(QSize(50, 50)), i18n("dvgrab utility not found,\n please install it for firewire capture")));
275             else KdenliveSettings::setDvgrab_path(dvgrabpath);
276         } else {
277             // Show capture info
278             capturefile = m_capturePath;
279             if (!capturefile.endsWith("/")) capturefile.append("/");
280             capturename = KdenliveSettings::dvgrabfilename();
281             if (capturename.isEmpty()) capturename = "capture";
282             QString extension;
283             switch (KdenliveSettings::firewireformat()) {
284             case 0:
285                 extension = ".dv";
286                 break;
287             case 1:
288             case 2:
289                 extension = ".avi";
290                 break;
291             case 3:
292                 extension = ".m2t";
293                 break;
294             }
295             capturename.append("xxx" + extension);
296             capturefile.append(capturename);
297             video_frame->setPixmap(mergeSideBySide(KIcon("network-connect").pixmap(QSize(50, 50)), i18n("Plug your camcorder and\npress connect button\nto initialize connection\nFiles will be saved in:\n%1", capturefile)));
298         }
299         break;
300     }
301 }
302
303 void RecMonitor::slotSetInfoMessage(const QString &message)
304 {
305     m_logger.insertItem(0, message);
306 }
307
308 QPixmap RecMonitor::mergeSideBySide(const QPixmap& pix, const QString txt)
309 {
310     QPainter p;
311     QRect r = QApplication::fontMetrics().boundingRect(QRect(0, 0, video_frame->width(), video_frame->height()), Qt::AlignLeft, txt);
312     int strWidth = r.width();
313     int strHeight = r.height();
314     int pixWidth = pix.width();
315     int pixHeight = pix.height();
316     QPixmap res(strWidth + 8 + pixWidth, qMax(strHeight, pixHeight));
317     res.fill(Qt::transparent);
318     p.begin(&res);
319     p.drawPixmap(0, 0, pix);
320     p.setPen(kapp->palette().text().color());
321     p.drawText(QRect(pixWidth + 8, 0, strWidth, strHeight), 0, txt);
322     p.end();
323     return res;
324 }
325
326
327 void RecMonitor::checkDeviceAvailability()
328 {
329     if (!KIO::NetAccess::exists(KUrl(KdenliveSettings::video4vdevice()), KIO::NetAccess::SourceSide , this)) {
330         m_playAction->setEnabled(false);
331         m_recAction->setEnabled(false);
332         video_frame->setPixmap(mergeSideBySide(KIcon("camera-web").pixmap(QSize(50, 50)), i18n("Cannot read from device %1\nPlease check drivers and access rights.", KdenliveSettings::video4vdevice())));
333     } else {
334         video_frame->setPixmap(mergeSideBySide(KIcon("camera-web").pixmap(QSize(50, 50)), i18n("Press play or record button\nto start video capture\nFiles will be saved in:\n%1", m_capturePath)));
335     }
336 }
337
338 void RecMonitor::slotDisconnect()
339 {
340     if (m_captureProcess->state() == QProcess::NotRunning) {
341         m_captureTime = KDateTime::currentLocalDateTime();
342         kDebug() << "CURRENT TIME: " << m_captureTime.toString();
343         m_didCapture = false;
344         slotStartPreview(false);
345         m_discAction->setIcon(KIcon("network-disconnect"));
346         m_discAction->setText(i18n("Disconnect"));
347         m_recAction->setEnabled(true);
348         m_stopAction->setEnabled(true);
349         m_playAction->setEnabled(true);
350         m_rewAction->setEnabled(true);
351         m_fwdAction->setEnabled(true);
352     } else {
353         m_captureProcess->write("q", 1);
354         QTimer::singleShot(1000, m_captureProcess, SLOT(kill()));
355         if (m_didCapture) manageCapturedFiles();
356         m_didCapture = false;
357     }
358 }
359
360 void RecMonitor::slotRewind()
361 {
362     m_captureProcess->write("a", 1);
363 }
364
365 void RecMonitor::slotForward()
366 {
367     m_captureProcess->write("z", 1);
368 }
369
370 void RecMonitor::slotStopCapture()
371 {
372     // stop capture
373     if (!m_isCapturing && !m_isPlaying) return;
374     m_videoBox->setHidden(true);
375     switch (device_selector->currentIndex()) {
376     case FIREWIRE:
377         m_captureProcess->write("\e", 2);
378         m_playAction->setIcon(m_playIcon);
379         m_isPlaying = false;
380         break;
381     case SCREENGRAB:
382         m_captureProcess->write("q\n", 3);
383         m_captureProcess->terminate();
384         video_frame->setText(i18n("Encoding captured video..."));
385         QTimer::singleShot(1000, m_captureProcess, SLOT(kill()));
386         break;
387     case VIDEO4LINUX:
388     case BLACKMAGIC:
389         if (m_captureDevice) {
390             m_captureDevice->stop();
391         }
392         recording_preview->setEnabled(true);
393         m_isCapturing = false;
394         m_isPlaying = false;
395         m_playAction->setEnabled(true);
396         m_stopAction->setEnabled(false);
397         m_recAction->setEnabled(true);
398         slotSetInfoMessage(i18n("Capture stopped"));
399         m_isCapturing = false;
400         m_recAction->setChecked(false);
401         if (autoaddbox->isChecked() && !m_captureFile.isEmpty() && QFile::exists(m_captureFile.path())) {
402             emit addProjectClip(m_captureFile);
403             m_captureFile.clear();
404         }
405         break;
406     default:
407         break;
408     }
409 }
410
411 void RecMonitor::slotStartPreview(bool play)
412 {
413     if (m_captureProcess->state() != QProcess::NotRunning) {
414         if (device_selector->currentIndex() == FIREWIRE) {
415             if (m_isPlaying) {
416                 m_captureProcess->write("k", 1);
417                 //captureProcess->write("\e", 2);
418                 m_playAction->setIcon(m_playIcon);
419                 m_isPlaying = false;
420             } else {
421                 m_captureProcess->write("p", 1);
422                 m_playAction->setIcon(m_pauseIcon);
423                 m_isPlaying = true;
424             }
425         }
426         return;
427     }
428     m_captureArgs.clear();
429     m_displayArgs.clear();
430     m_isPlaying = false;
431     QString capturename = KdenliveSettings::dvgrabfilename();
432     QString path;
433     MltVideoProfile profile;
434     QString producer;
435     QStringList dvargs = KdenliveSettings::dvgrabextra().simplified().split(" ", QString::SkipEmptyParts);
436     int ix = device_selector->currentIndex();
437     m_videoBox->setHidden(ix != VIDEO4LINUX && ix != BLACKMAGIC);
438     switch (ix) {
439     case FIREWIRE:
440         switch (KdenliveSettings::firewireformat()) {
441         case 0:
442             // RAW DV CAPTURE
443             m_captureArgs << "--format" << "raw";
444             m_displayArgs << "-f" << "dv";
445             break;
446         case 1:
447             // DV type 1
448             m_captureArgs << "--format" << "dv1";
449             m_displayArgs << "-f" << "dv";
450             break;
451         case 2:
452             // DV type 2
453             m_captureArgs << "--format" << "dv2";
454             m_displayArgs << "-f" << "dv";
455             break;
456         case 3:
457             // HDV CAPTURE
458             m_captureArgs << "--format" << "hdv";
459             m_displayArgs << "-f" << "mpegts";
460             break;
461         }
462         if (KdenliveSettings::firewireautosplit()) m_captureArgs << "--autosplit";
463         if (KdenliveSettings::firewiretimestamp()) m_captureArgs << "--timestamp";
464         if (!dvargs.isEmpty()) {
465             m_captureArgs << dvargs;
466         }
467         m_captureArgs << "-i";
468         if (capturename.isEmpty()) capturename = "capture";
469         m_captureArgs << capturename << "-";
470
471         m_displayArgs << "-x" << QString::number(video_frame->width()) << "-y" << QString::number(video_frame->height()) << "-";
472
473         m_captureProcess->setStandardOutputProcess(m_displayProcess);
474         m_captureProcess->setWorkingDirectory(m_capturePath);
475         kDebug() << "Capture: Running dvgrab " << m_captureArgs.join(" ");
476
477         m_captureProcess->start(KdenliveSettings::dvgrab_path(), m_captureArgs);
478         if (play) m_captureProcess->write(" ", 1);
479         m_discAction->setEnabled(true);
480         break;
481     case VIDEO4LINUX:
482         path = KStandardDirs::locateLocal("appdata", "profiles/video4linux");
483         m_manager->activateMonitor("record");
484         buildMltDevice(path);
485         profile = ProfilesDialog::getVideoProfile(path);
486         producer = QString("avformat-novalidate:video4linux2:%1?width:%2&height:%3&frame_rate:%4").arg(KdenliveSettings::video4vdevice()).arg(profile.width).arg(profile.height).arg((double) profile.frame_rate_num / profile.frame_rate_den);
487         kDebug()<< "PROD: "<<producer;
488         if (!m_captureDevice->slotStartPreview(producer)) {
489             // v4l capture failed to start
490             video_frame->setText(i18n("Failed to start Video4Linux,\ncheck your parameters..."));
491             m_videoBox->setHidden(true);
492             
493         } else {
494             m_playAction->setEnabled(false);
495             m_stopAction->setEnabled(true);
496             m_isPlaying = true;
497         }
498         
499         break;
500     case BLACKMAGIC:
501         path = KdenliveSettings::current_profile();
502         m_manager->activateMonitor("record");
503         buildMltDevice(path);
504         profile = ProfilesDialog::getVideoProfile(path);
505         producer = QString("decklink:%1").arg(KdenliveSettings::decklink_capturedevice());
506         if (!m_captureDevice->slotStartPreview(producer)) {
507             // v4l capture failed to start
508             video_frame->setText(i18n("Failed to start Decklink,\ncheck your parameters..."));
509             m_videoBox->setHidden(true);
510             
511         } else {
512             m_playAction->setEnabled(false);
513             m_stopAction->setEnabled(true);
514             m_isPlaying = true;
515         }
516         break;
517     default:
518         break;
519     }
520
521     if (device_selector->currentIndex() == FIREWIRE) {
522         kDebug() << "Capture: Running ffplay " << m_displayArgs.join(" ");
523         m_displayProcess->start("ffplay", m_displayArgs);
524         video_frame->setText(i18n("Initialising..."));
525     } else {
526         // do something when starting screen grab
527     }
528 }
529
530 void RecMonitor::slotRecord()
531 {
532     if (m_captureProcess->state() == QProcess::NotRunning && device_selector->currentIndex() == FIREWIRE) {
533         slotStartPreview();
534     }
535     if (m_isCapturing) {
536         // User stopped capture
537         slotStopCapture();
538         return;
539     } else if (device_selector->currentIndex() == FIREWIRE) {
540         m_isCapturing = true;
541         m_didCapture = true;
542         m_captureProcess->write("c\n", 3);
543 #if KDE_IS_VERSION(4,2,0)
544         m_spaceTimer.start();
545 #endif
546         return;
547     }
548     if (m_captureProcess->state() == QProcess::NotRunning) {
549         m_logger.clear();
550         m_recAction->setChecked(true);
551         QString extension = "mpg";
552         if (device_selector->currentIndex() == SCREENGRAB) extension = "ogv"; //KdenliveSettings::screengrabextension();
553         else if (device_selector->currentIndex() == VIDEO4LINUX) extension = KdenliveSettings::v4l_extension();
554         else if (device_selector->currentIndex() == BLACKMAGIC) extension = KdenliveSettings::decklink_extension();
555         QString path = KUrl(m_capturePath).path(KUrl::AddTrailingSlash) + "capture0000." + extension;
556         int i = 1;
557         while (QFile::exists(path)) {
558             QString num = QString::number(i).rightJustified(4, '0', false);
559             path = KUrl(m_capturePath).path(KUrl::AddTrailingSlash) + "capture" + num + '.' + extension;
560             i++;
561         }
562         m_captureFile = KUrl(path);
563
564         m_captureArgs.clear();
565         m_displayArgs.clear();
566         QString args;
567         QString playlist;
568         MltVideoProfile profile;
569         QString capturename = KdenliveSettings::dvgrabfilename();
570         if (capturename.isEmpty()) capturename = "capture";
571
572         switch (device_selector->currentIndex()) {
573         case VIDEO4LINUX:
574             path = KStandardDirs::locateLocal("appdata", "profiles/video4linux");
575             profile = ProfilesDialog::getVideoProfile(path);
576             m_videoBox->setRatio((double) profile.display_aspect_num / profile.display_aspect_den);
577             buildMltDevice(path);
578             playlist = QString("<mlt title=\"capture\"><producer id=\"producer0\" in=\"0\" out=\"99999\"><property name=\"mlt_type\">producer</property><property name=\"length\">100000</property><property name=\"eof\">pause</property><property name=\"resource\">video4linux2:%1?width:%2&amp;height:%3&amp;frame_rate:%4</property><property name=\"mlt_service\">avformat-novalidate</property></producer><playlist id=\"playlist0\"><entry producer=\"producer0\" in=\"0\" out=\"99999\"/></playlist>").arg(KdenliveSettings::video4vdevice()).arg(profile.width).arg(profile.height).arg((double) profile.frame_rate_num / profile.frame_rate_den);
579
580             // Add alsa audio capture
581             if (KdenliveSettings::v4l_captureaudio()) {
582                 playlist.append(QString("<producer id=\"producer1\" in=\"0\" out=\"99999\"><property name=\"mlt_type\">producer</property><property name=\"length\">100000</property><property name=\"eof\">pause</property><property name=\"resource\">alsa:%5</property><property name=\"audio_index\">0</property><property name=\"video_index\">-1</property><property name=\"mlt_service\">avformat</property></producer><playlist id=\"playlist1\"><entry producer=\"producer1\" in=\"0\" out=\"99999\"/></playlist>").arg(KdenliveSettings::v4l_alsadevicename()));
583             }
584             
585
586             playlist.append("<tractor id=\"tractor0\" title=\"video0\" global_feed=\"1\" in=\"0\" out=\"99999\">");
587
588             playlist.append("<track producer=\"playlist0\"/>");            
589
590             // Audio mix
591             if (KdenliveSettings::v4l_captureaudio()) {
592                 playlist.append("<track producer=\"playlist1\"/>");
593                 playlist.append("<transition id=\"transition0\" in=\"0\" out=\"0\"><property name=\"a_track\">0</property><property name=\"b_track\">1</property><property name=\"mlt_type\">transition</property><property name=\"mlt_service\">mix</property></transition>");
594             }
595
596             playlist.append("</tractor></mlt>");
597
598             if (m_captureDevice->slotStartCapture(KdenliveSettings::v4l_parameters(), m_captureFile.path(), playlist, recording_preview->currentIndex())) {
599                 m_videoBox->setHidden(false);
600                 m_isCapturing = true;
601                 m_recAction->setEnabled(false);
602                 m_stopAction->setEnabled(true);
603                 recording_preview->setEnabled(false);
604             }
605             else {
606                 video_frame->setText(i18n("Failed to start Video4Linux,\ncheck your parameters..."));                
607                 m_videoBox->setHidden(true);
608                 m_isCapturing = false;
609             }
610             break;
611             
612         case BLACKMAGIC:
613             path = KdenliveSettings::current_profile();
614             profile = ProfilesDialog::getVideoProfile(path);
615             m_videoBox->setRatio((double) profile.display_aspect_num / profile.display_aspect_den);
616             buildMltDevice(path);
617                
618             playlist = QString("<producer id=\"producer0\" in=\"0\" out=\"99999\"><property name=\"mlt_type\">producer</property><property name=\"length\">100000</property><property name=\"eof\">pause</property><property name=\"resource\">%1</property><property name=\"mlt_service\">decklink</property></producer>").arg(KdenliveSettings::decklink_capturedevice());
619
620             if (m_captureDevice->slotStartCapture(KdenliveSettings::decklink_parameters(), m_captureFile.path(), QString("decklink:%1").arg(KdenliveSettings::decklink_capturedevice()), recording_preview->currentIndex(), false)) {
621                 m_videoBox->setHidden(false);
622                 m_isCapturing = true;
623                 slotSetInfoMessage(i18n("Capturing to %1", m_captureFile.fileName()));
624                 m_recAction->setEnabled(false);
625                 m_stopAction->setEnabled(true);
626                 recording_preview->setEnabled(false);
627             }
628             else {
629                 video_frame->setText(i18n("Failed to start Decklink,\ncheck your parameters..."));
630                 slotSetInfoMessage(i18n("Failed to start capture"));
631                 m_videoBox->setHidden(true);
632                 m_isCapturing = false;
633             }
634             break;
635             
636         case SCREENGRAB:
637             switch (KdenliveSettings::rmd_capture_type()) {
638             case 0:
639                 // Full screen capture, nothing special to do
640                 break;
641             default:
642                 // Region capture
643                 m_captureArgs << "--width" << QString::number(KdenliveSettings::rmd_width()) << "--height" << QString::number(KdenliveSettings::rmd_height());
644                 if (!KdenliveSettings::rmd_follow_mouse()) {
645                     m_captureArgs << "-x" << QString::number(KdenliveSettings::rmd_offsetx()) << "-y" << QString::number(KdenliveSettings::rmd_offsety());
646                 } else {
647                     m_captureArgs << "--follow-mouse";
648                     if (KdenliveSettings::rmd_hide_frame()) m_captureArgs << "--no-frame";
649                 }
650                 break;
651             }
652             if (KdenliveSettings::rmd_hide_mouse()) m_captureArgs << "--no-cursor";
653             m_isCapturing = true;
654             if (KdenliveSettings::rmd_capture_audio()) {
655                 m_captureArgs << "--freq" << KdenliveSettings::rmd_freq();
656                 m_captureArgs << "--channels" << QString::number(KdenliveSettings::rmd_audio_channels());
657                 if (KdenliveSettings::rmd_use_jack()) {
658                     m_captureArgs << "--use-jack";
659                     QStringList ports = KdenliveSettings::rmd_jackports().split(" ", QString::SkipEmptyParts);
660                     for (int i = 0; i < ports.count(); ++i) {
661                         m_captureArgs << ports.at(i);
662                     }
663                     if (KdenliveSettings::rmd_jack_buffer() > 0.0)
664                         m_captureArgs << "--ring-buffer-size" << QString::number(KdenliveSettings::rmd_jack_buffer());
665                 } else {
666                     if (!KdenliveSettings::rmd_alsadevicename().isEmpty())
667                         m_captureArgs << "--device" << KdenliveSettings::rmd_alsadevicename();
668                     if (KdenliveSettings::rmd_alsa_buffer() > 0)
669                         m_captureArgs << "--buffer-size" << QString::number(KdenliveSettings::rmd_alsa_buffer());
670                 }
671             } else m_captureArgs << "--no-sound";
672
673             if (KdenliveSettings::rmd_fullshots()) m_captureArgs << "--full-shots";
674             m_captureArgs << "--v_bitrate" << QString::number(KdenliveSettings::rmd_bitrate());
675             m_captureArgs << "--v_quality" << QString::number(KdenliveSettings::rmd_quality());
676             m_captureArgs << "--workdir" << KdenliveSettings::currenttmpfolder();
677             m_captureArgs << "--fps" << QString::number(KdenliveSettings::rmd_fps()) << "-o" << m_captureFile.path();
678             m_captureProcess->start(KdenliveSettings::rmd_path(), m_captureArgs);
679             kDebug() << "// RecordMyDesktop params: " << m_captureArgs;
680             break;
681         default:
682             break;
683         }
684
685
686         if (device_selector->currentIndex() == FIREWIRE) {
687             m_isCapturing = true;
688             kDebug() << "Capture: Running ffplay " << m_displayArgs.join(" ");
689             m_displayProcess->start("ffplay", m_displayArgs);
690             video_frame->setText(i18n("Initialising..."));
691         }
692     } else {
693         // stop capture
694         m_displayProcess->kill();
695         //captureProcess->kill();
696         QTimer::singleShot(1000, this, SLOT(slotRecord()));
697     }
698 }
699
700 /*
701 void RecMonitor::slotStartGrab(const QRect &rect) {
702     rgnGrab->deleteLater();
703     QApplication::restoreOverrideCursor();
704     show();
705     if (rect.isNull()) return;
706     int width = rect.width();
707     int height = rect.height();
708     if (width % 2 != 0) width--;
709     if (height % 2 != 0) height--;
710     QString args = KdenliveSettings::screengrabcapture().replace("%size", QString::number(width) + "x" + QString::number(height)).replace("%offset", "+" + QString::number(rect.x()) + "," + QString::number(rect.y()));
711     if (KdenliveSettings::screengrabenableaudio()) {
712         // also capture audio
713         if (KdenliveSettings::useosscapture()) m_captureArgs << KdenliveSettings::screengrabosscapture().simplified().split(' ');
714         else m_captureArgs << KdenliveSettings::screengrabalsacapture2().simplified().split(' ');
715     }
716     m_captureArgs << args.simplified().split(' ') << KdenliveSettings::screengrabencoding().simplified().split(' ') << m_captureFile.path();
717     m_isCapturing = true;
718     video_frame->setText(i18n("Capturing..."));
719     if (KdenliveSettings::screengrabenableaudio() && !KdenliveSettings::useosscapture()) {
720         QStringList alsaArgs = KdenliveSettings::screengrabalsacapture().simplified().split(' ');
721         alsaProcess->setStandardOutputProcess(captureProcess);
722         kDebug() << "Capture: Running arecord " << alsaArgs.join(" ");
723         alsaProcess->start("arecord", alsaArgs);
724     }
725     kDebug() << "Capture: Running ffmpeg " << m_captureArgs.join(" ");
726     captureProcess->start("ffmpeg", m_captureArgs);
727 }*/
728
729 void RecMonitor::slotProcessStatus(QProcess::ProcessState status)
730 {
731     if (status == QProcess::NotRunning) {
732         m_displayProcess->kill();
733         if (m_isCapturing && device_selector->currentIndex() != FIREWIRE)
734             if (autoaddbox->isChecked() && !m_captureFile.isEmpty() && QFile::exists(m_captureFile.path())) {
735                 emit addProjectClip(m_captureFile);
736                 m_captureFile.clear();
737             }
738         if (device_selector->currentIndex() == FIREWIRE) {
739             m_discAction->setIcon(KIcon("network-connect"));
740             m_discAction->setText(i18n("Connect"));
741             m_playAction->setEnabled(false);
742             m_rewAction->setEnabled(false);
743             m_fwdAction->setEnabled(false);
744             m_recAction->setEnabled(false);
745         }
746         m_isPlaying = false;
747         m_playAction->setIcon(m_playIcon);
748         m_recAction->setChecked(false);
749         m_stopAction->setEnabled(false);
750         device_selector->setEnabled(true);
751         if (m_captureProcess && m_captureProcess->exitStatus() == QProcess::CrashExit) {
752             video_frame->setText(i18n("Capture crashed, please check your parameters"));
753         } else {
754             if (device_selector->currentIndex() != SCREENGRAB) {
755                 video_frame->setText(i18n("Not connected"));
756             } else {
757                 if (m_captureProcess->exitCode() != 0) {
758                     video_frame->setText(i18n("Capture crashed, please check your parameters\nRecordMyDesktop exit code: %1", QString::number(m_captureProcess->exitCode())));
759                 } else {
760                     video_frame->setPixmap(mergeSideBySide(KIcon("video-display").pixmap(QSize(50, 50)), i18n("Press record button\nto start screen capture\nFiles will be saved in:\n%1", m_capturePath)));
761                 }
762             }
763         }
764         m_isCapturing = false;
765
766 #if KDE_IS_VERSION(4,2,0)
767         m_spaceTimer.stop();
768         // update free space info
769         slotUpdateFreeSpace();
770 #endif
771
772     } else {
773         if (device_selector->currentIndex() != SCREENGRAB) m_stopAction->setEnabled(true);
774         device_selector->setEnabled(false);
775     }
776 }
777
778 void RecMonitor::manageCapturedFiles()
779 {
780     QString extension;
781     switch (KdenliveSettings::firewireformat()) {
782     case 0:
783         extension = ".dv";
784         break;
785     case 1:
786     case 2:
787         extension = ".avi";
788         break;
789     case 3:
790         extension = ".m2t";
791         break;
792     }
793     QDir dir(m_capturePath);
794     QStringList filters;
795     QString capturename = KdenliveSettings::dvgrabfilename();
796     if (capturename.isEmpty()) capturename = "capture";
797     filters << capturename + "*" + extension;
798     const QStringList result = dir.entryList(filters, QDir::Files, QDir::Time);
799     KUrl::List capturedFiles;
800     foreach(const QString & name, result) {
801         KUrl url = KUrl(dir.filePath(name));
802         if (KIO::NetAccess::exists(url, KIO::NetAccess::SourceSide, this)) {
803             KFileItem file(KFileItem::Unknown, KFileItem::Unknown, url, true);
804             if (file.time(KFileItem::ModificationTime) > m_captureTime) capturedFiles.append(url);
805         }
806     }
807     kDebug() << "Found : " << capturedFiles.count() << " new capture files";
808     kDebug() << capturedFiles;
809
810     if (capturedFiles.count() > 0) {
811         ManageCapturesDialog *d = new ManageCapturesDialog(capturedFiles, this);
812         if (d->exec() == QDialog::Accepted) {
813             capturedFiles = d->importFiles();
814             foreach(const KUrl & url, capturedFiles) {
815                 emit addProjectClip(url);
816             }
817         }
818         delete d;
819     }
820 }
821
822 // virtual
823 void RecMonitor::mousePressEvent(QMouseEvent * /*event*/)
824 {
825 #if KDE_IS_VERSION(4,2,0)
826     if (m_freeSpace->underMouse()) slotUpdateFreeSpace();
827 #endif
828 }
829
830 void RecMonitor::slotUpdateFreeSpace()
831 {
832 #if KDE_IS_VERSION(4,2,0)
833     KDiskFreeSpaceInfo info = KDiskFreeSpaceInfo::freeSpaceInfo(m_capturePath);
834     if (info.isValid()) {
835         m_freeSpace->setValue(100 * info.used() / info.size());
836         m_freeSpace->setText(i18n("Free space: %1", KIO::convertSize(info.available())));
837         m_freeSpace->update();
838     }
839 #endif
840 }
841
842 void RecMonitor::refreshRecMonitor(bool visible)
843 {
844     if (visible) {
845         //if (!m_isActive) m_monitorManager->activateRecMonitor(m_name);
846
847     }
848 }
849
850 void RecMonitor::slotPlay()
851 {
852
853     //if (!m_isActive) m_monitorManager->activateRecMonitor(m_name);
854
855 }
856
857 void RecMonitor::slotReadDvgrabInfo()
858 {
859     QString data = m_captureProcess->readAllStandardError().simplified();
860     data = data.section('"', 2, 2).simplified();
861     m_dvinfo.setText(data.left(11));
862     m_dvinfo.updateGeometry();
863 }
864
865 AbstractRender *RecMonitor::abstractRender()
866 {
867     return m_captureDevice;
868 }
869
870
871 void RecMonitor::analyseFrames(bool analyse)
872 {
873     m_analyse = analyse;
874     if (m_captureDevice) m_captureDevice->sendFrameForAnalysis = analyse;
875 }
876
877 void RecMonitor::slotDroppedFrames(int dropped)
878 {
879     slotSetInfoMessage(i18n("%1 dropped frames", dropped));
880 }
881
882 void RecMonitor::buildMltDevice(const QString &path)
883 {
884     if (m_captureDevice == NULL) {
885         m_captureDevice = new MltDeviceCapture(path, m_videoBox, this);
886         connect(m_captureDevice, SIGNAL(droppedFrames(int)), this, SLOT(slotDroppedFrames(int)));
887         m_captureDevice->sendFrameForAnalysis = m_analyse;
888         m_manager->updateScopeSource();
889     }
890 }
891
892 void RecMonitor::slotChangeRecordingPreview(int ix)
893 {
894     KdenliveSettings::setRecording_preview(ix);
895 }
896
897 #include "recmonitor.moc"
898