+void Wizard::slotDetectWebcam()
+{
+#ifdef USE_V4L
+ m_capture.v4l_devices->blockSignals(true);
+ m_capture.v4l_devices->clear();
+
+ // Video 4 Linux device detection
+ for (int i = 0; i < 10; ++i) {
+ QString path = "/dev/video" + QString::number(i);
+ if (QFile::exists(path)) {
+ QStringList deviceInfo = V4lCaptureHandler::getDeviceName(path.toUtf8().constData());
+ if (!deviceInfo.isEmpty()) {
+ m_capture.v4l_devices->addItem(deviceInfo.at(0), path);
+ m_capture.v4l_devices->setItemData(m_capture.v4l_devices->count() - 1, deviceInfo.at(1), Qt::UserRole + 1);
+ }
+ }
+ }
+ if (m_capture.v4l_devices->count() > 0) {
+ m_capture.v4l_status->setText(i18n("Default video4linux device:"));
+ // select default device
+ bool found = false;
+ for (int i = 0; i < m_capture.v4l_devices->count(); ++i) {
+ QString device = m_capture.v4l_devices->itemData(i).toString();
+ if (device == KdenliveSettings::video4vdevice()) {
+ m_capture.v4l_devices->setCurrentIndex(i);
+ found = true;
+ break;
+ }
+ }
+ slotUpdateCaptureParameters();
+ if (!found) m_capture.v4l_devices->setCurrentIndex(0);
+ } else m_capture.v4l_status->setText(i18n("No device found, plug your webcam and refresh."));
+ m_capture.v4l_devices->blockSignals(false);
+#endif /* USE_V4L */
+}
+
+void Wizard::slotUpdateCaptureParameters()
+{
+ QString device = m_capture.v4l_devices->itemData(m_capture.v4l_devices->currentIndex()).toString();
+ if (!device.isEmpty()) KdenliveSettings::setVideo4vdevice(device);
+
+ QString formats = m_capture.v4l_devices->itemData(m_capture.v4l_devices->currentIndex(), Qt::UserRole + 1).toString();
+
+ m_capture.v4l_formats->blockSignals(true);
+ m_capture.v4l_formats->clear();
+
+ QString vl4ProfilePath = KStandardDirs::locateLocal("appdata", "profiles/video4linux");
+ if (QFile::exists(vl4ProfilePath)) {
+ MltVideoProfile profileInfo = ProfilesDialog::getVideoProfile(vl4ProfilePath);
+ m_capture.v4l_formats->addItem(i18n("Current settings (%1x%2, %3/%4fps)", profileInfo.width, profileInfo.height, profileInfo.frame_rate_num, profileInfo.frame_rate_den), QStringList() << QString("unknown") <<QString::number(profileInfo.width)<<QString::number(profileInfo.height)<<QString::number(profileInfo.frame_rate_num)<<QString::number(profileInfo.frame_rate_den));
+ }
+ QStringList pixelformats = formats.split('>', QString::SkipEmptyParts);
+ QString itemSize;
+ QString pixelFormat;
+ QStringList itemRates;
+ for (int i = 0; i < pixelformats.count(); ++i) {
+ QString format = pixelformats.at(i).section(':', 0, 0);
+ QStringList sizes = pixelformats.at(i).split(':', QString::SkipEmptyParts);
+ pixelFormat = sizes.takeFirst();
+ for (int j = 0; j < sizes.count(); j++) {
+ itemSize = sizes.at(j).section('=', 0, 0);
+ itemRates = sizes.at(j).section('=', 1, 1).split(',', QString::SkipEmptyParts);
+ for (int k = 0; k < itemRates.count(); k++) {
+ QString formatDescription = '[' + format + "] " + itemSize + " (" + itemRates.at(k) + ')';
+ if (m_capture.v4l_formats->findText(formatDescription) == -1)
+ m_capture.v4l_formats->addItem(formatDescription, QStringList() << format << itemSize.section('x', 0, 0) << itemSize.section('x', 1, 1) << itemRates.at(k).section('/', 0, 0) << itemRates.at(k).section('/', 1, 1));
+ }
+ }
+ }
+ if (!QFile::exists(vl4ProfilePath)) {
+ if (m_capture.v4l_formats->count() > 9) slotSaveCaptureFormat();
+ else {
+ // No existing profile and no autodetected profiles
+ MltVideoProfile profileInfo;
+ profileInfo.width = 320;
+ profileInfo.height = 200;
+ profileInfo.frame_rate_num = 15;
+ profileInfo.frame_rate_den = 1;
+ profileInfo.display_aspect_num = 4;
+ profileInfo.display_aspect_den = 3;
+ profileInfo.sample_aspect_num = 1;
+ profileInfo.sample_aspect_den = 1;
+ profileInfo.progressive = 1;
+ profileInfo.colorspace = 601;
+ ProfilesDialog::saveProfile(profileInfo, vl4ProfilePath);
+ m_capture.v4l_formats->addItem(i18n("Default settings (%1x%2, %3/%4fps)", profileInfo.width, profileInfo.height, profileInfo.frame_rate_num, profileInfo.frame_rate_den), QStringList() << QString("unknown") <<QString::number(profileInfo.width)<<QString::number(profileInfo.height)<<QString::number(profileInfo.frame_rate_num)<<QString::number(profileInfo.frame_rate_den));
+ }
+ }
+ m_capture.v4l_formats->blockSignals(false);
+}
+
+void Wizard::checkMltComponents()
+{
+ QSize itemSize(20, fontMetrics().height() * 2.5);
+ m_mltCheck.programList->setColumnWidth(0, 30);
+ m_mltCheck.programList->setIconSize(QSize(24, 24));
+
+
+ QTreeWidgetItem *mltitem = new QTreeWidgetItem(m_mltCheck.programList);
+
+ QTreeWidgetItem *meltitem = new QTreeWidgetItem(m_mltCheck.programList, QStringList() << QString() << i18n("Melt") + " (" + KdenliveSettings::rendererpath() + ')');
+ meltitem->setData(1, Qt::UserRole, i18n("Required for rendering (part of MLT package)"));
+ meltitem->setSizeHint(0, itemSize);
+ meltitem->setIcon(0, m_okIcon);
+
+
+ Mlt::Repository *repository = Mlt::Factory::init();
+ if (!repository) {
+ mltitem->setData(1, Qt::UserRole, i18n("Cannot start the MLT video backend!"));
+ mltitem->setIcon(0, m_badIcon);
+ m_systemCheckIsOk = false;
+ button(QWizard::NextButton)->setEnabled(false);
+ }
+ else {
+ int mltVersion = (mltVersionMajor << 16) + (mltVersionMinor << 8) + mltVersionRevision;
+ mltitem->setText(1, i18n("MLT version: %1", mlt_version_get_string()));
+ mltitem->setSizeHint(0, itemSize);
+ if (mlt_version_get_int() < 1792) {
+ mltitem->setData(1, Qt::UserRole, i18n("Your MLT version is unsupported!!!"));
+ mltitem->setIcon(0, m_badIcon);
+ m_systemCheckIsOk = false;
+ button(QWizard::NextButton)->setEnabled(false);
+ }
+ else if (mlt_version_get_int() < mltVersion) {
+ mltitem->setData(1, Qt::UserRole, i18n("Please upgrade to MLT %1.%2.%3", mltVersionMajor, mltVersionMinor, mltVersionRevision));
+ mltitem->setIcon(0, m_badIcon);
+ }
+ else {
+ mltitem->setData(1, Qt::UserRole, i18n("MLT video backend!"));
+ mltitem->setIcon(0, m_okIcon);
+ }
+
+ // Retrieve the list of available transitions.
+ Mlt::Properties *producers = repository->producers();
+ QStringList producersItemList;
+ for (int i = 0; i < producers->count(); ++i)
+ producersItemList << producers->get_name(i);
+ delete producers;
+
+ Mlt::Properties *consumers = repository->consumers();
+ QStringList consumersItemList;
+ for (int i = 0; i < consumers->count(); ++i)
+ consumersItemList << consumers->get_name(i);
+ delete consumers;
+
+ // SDL module
+ QTreeWidgetItem *sdlItem = new QTreeWidgetItem(m_mltCheck.programList, QStringList() << QString() << i18n("SDL module"));
+ sdlItem->setData(1, Qt::UserRole, i18n("Required for Kdenlive"));
+ sdlItem->setSizeHint(0, itemSize);
+
+ if (!consumersItemList.contains("sdl")) {
+ sdlItem->setIcon(0, m_badIcon);
+ m_systemCheckIsOk = false;
+ button(QWizard::NextButton)->setEnabled(false);
+ }
+ else {
+ sdlItem->setIcon(0, m_okIcon);
+ }
+
+ // AVformat module
+ QTreeWidgetItem *avformatItem = new QTreeWidgetItem(m_mltCheck.programList, QStringList() << QString() << i18n("Avformat module (FFmpeg)"));
+ avformatItem->setData(1, Qt::UserRole, i18n("Required to work with various video formats (hdv, mpeg, flash, ...)"));
+ avformatItem->setSizeHint(0, itemSize);
+ Mlt::Consumer *consumer = NULL;
+ Mlt::Profile p;
+ if (consumersItemList.contains("avformat"))
+ consumer = new Mlt::Consumer(p, "avformat");
+ if (consumer == NULL || !consumer->is_valid()) {
+ avformatItem->setIcon(0, m_badIcon);
+ m_mltCheck.tabWidget->setTabEnabled(1, false);
+ }
+ else {
+ avformatItem->setIcon(0, m_okIcon);
+ consumer->set("vcodec", "list");
+ consumer->set("acodec", "list");
+ consumer->set("f", "list");
+ consumer->start();
+ QStringList result;
+ Mlt::Properties vcodecs((mlt_properties) consumer->get_data("vcodec"));
+ for (int i = 0; i < vcodecs.count(); ++i)
+ result << QString(vcodecs.get(i));
+ m_mltCheck.vcodecs_list->addItems(result);
+ KdenliveSettings::setVideocodecs(result);
+ result.clear();
+ Mlt::Properties acodecs((mlt_properties) consumer->get_data("acodec"));
+ for (int i = 0; i < acodecs.count(); ++i)
+ result << QString(acodecs.get(i));
+ m_mltCheck.acodecs_list->addItems(result);
+ KdenliveSettings::setAudiocodecs(result);
+ result.clear();
+ Mlt::Properties formats((mlt_properties) consumer->get_data("f"));
+ for (int i = 0; i < formats.count(); ++i)
+ result << QString(formats.get(i));
+ m_mltCheck.formats_list->addItems(result);
+ KdenliveSettings::setSupportedformats(result);
+ checkMissingCodecs();
+ delete consumer;
+ }
+
+ // DV module
+ QTreeWidgetItem *dvItem = new QTreeWidgetItem(m_mltCheck.programList, QStringList() << QString() << i18n("DV module (libdv)"));
+ dvItem->setData(1, Qt::UserRole, i18n("Required to work with dv files if avformat module is not installed"));
+ dvItem->setSizeHint(0, itemSize);
+ if (!producersItemList.contains("libdv")) {
+ dvItem->setIcon(0, m_badIcon);
+ }
+ else {
+ dvItem->setIcon(0, m_okIcon);
+ }
+
+ // Image module
+ QTreeWidgetItem *imageItem = new QTreeWidgetItem(m_mltCheck.programList, QStringList() << QString() << i18n("QImage module"));
+ imageItem->setData(1, Qt::UserRole, i18n("Required to work with images"));
+ imageItem->setSizeHint(0, itemSize);
+ if (!producersItemList.contains("qimage")) {
+ imageItem->setIcon(0, m_badIcon);
+ imageItem = new QTreeWidgetItem(m_mltCheck.programList, QStringList() << QString() << i18n("Pixbuf module"));
+ imageItem->setData(1, Qt::UserRole, i18n("Required to work with images"));
+ imageItem->setSizeHint(0, itemSize);
+ if (producersItemList.contains("pixbuf")) {
+ imageItem->setIcon(0, m_badIcon);
+ }
+ else {
+ imageItem->setIcon(0, m_okIcon);
+ }
+ }
+ else {
+ imageItem->setIcon(0, m_okIcon);
+ }
+
+ // Titler module
+ QTreeWidgetItem *titleItem = new QTreeWidgetItem(m_mltCheck.programList, QStringList() << QString() << i18n("Title module"));
+ titleItem->setData(1, Qt::UserRole, i18n("Required to work with titles"));
+ titleItem->setSizeHint(0, itemSize);
+ if (!producersItemList.contains("kdenlivetitle")) {
+ KdenliveSettings::setHastitleproducer(false);
+ titleItem->setIcon(0, m_badIcon);
+ } else {
+ titleItem->setIcon(0, m_okIcon);
+ KdenliveSettings::setHastitleproducer(true);
+ }
+ }
+}
+
+void Wizard::checkMissingCodecs()
+{
+ const QStringList acodecsList = KdenliveSettings::audiocodecs();
+ const QStringList vcodecsList = KdenliveSettings::videocodecs();
+ bool replaceVorbisCodec = false;
+ if (acodecsList.contains("libvorbis")) replaceVorbisCodec = true;
+ bool replaceLibfaacCodec = false;
+ if (!acodecsList.contains("aac") && acodecsList.contains("libfaac")) replaceLibfaacCodec = true;
+
+ QString exportFolder = KStandardDirs::locateLocal("appdata", "export/");
+ QDir directory = QDir(exportFolder);
+ QStringList filter;
+ filter << "*.xml";
+ QStringList fileList = directory.entryList(filter, QDir::Files);
+ // We should parse customprofiles.xml in last position, so that user profiles
+ // can also override profiles installed by KNewStuff
+ QStringList requiredACodecs;
+ QStringList requiredVCodecs;
+ foreach(const QString &filename, fileList) {
+ QDomDocument doc;
+ QFile file(exportFolder + filename);
+ doc.setContent(&file, false);
+ file.close();
+ QString std;
+ QString format;
+ QDomNodeList profiles = doc.elementsByTagName("profile");
+ for (int i = 0; i < profiles.count(); ++i) {
+ std = profiles.at(i).toElement().attribute("args");
+ format.clear();
+ if (std.startsWith("acodec=")) format = std.section("acodec=", 1, 1);
+ else if (std.contains(" acodec=")) format = std.section(" acodec=", 1, 1);
+ if (!format.isEmpty()) requiredACodecs << format.section(' ', 0, 0).toLower();
+ format.clear();
+ if (std.startsWith("vcodec=")) format = std.section("vcodec=", 1, 1);
+ else if (std.contains(" vcodec=")) format = std.section(" vcodec=", 1, 1);
+ if (!format.isEmpty()) requiredVCodecs << format.section(' ', 0, 0).toLower();
+ }
+ }
+ requiredACodecs.removeDuplicates();
+ requiredVCodecs.removeDuplicates();
+ if (replaceVorbisCodec) requiredACodecs.replaceInStrings("vorbis", "libvorbis");
+ if (replaceLibfaacCodec) requiredACodecs.replaceInStrings("aac", "libfaac");
+
+ for (int i = 0; i < acodecsList.count(); ++i)
+ requiredACodecs.removeAll(acodecsList.at(i));
+ for (int i = 0; i < vcodecsList.count(); ++i)
+ requiredVCodecs.removeAll(vcodecsList.at(i));
+ if (!requiredACodecs.isEmpty() || !requiredVCodecs.isEmpty()) {
+ QString missing = requiredACodecs.join(",");
+ if (!missing.isEmpty() && !requiredVCodecs.isEmpty()) missing.append(',');
+ missing.append(requiredVCodecs.join(","));
+ missing.prepend(i18n("The following codecs were not found on your system. Check our <a href=''>online manual</a> if you need them: "));
+ // Some codecs required for rendering are not present on this system, warn user
+ show();
+#if KDE_IS_VERSION(4,7,0)
+ KMessageWidget *infoMessage = new KMessageWidget(this);
+ m_startLayout->insertWidget(1, infoMessage);
+ infoMessage->setCloseButtonVisible(false);
+ infoMessage->setWordWrap(true);
+ infoMessage->setMessageType(KMessageWidget::Warning);
+#if KDE_IS_VERSION(4,10,0)
+ connect(infoMessage, SIGNAL(linkActivated(QString)), this, SLOT(slotOpenManual()));
+ infoMessage->setText(missing);
+#else
+ // clickable text in kmessagewidget only available since KDE 4.10
+ // remove link from text
+ missing.remove(QRegExp("<[^>]*>"));
+ infoMessage->setText(missing);
+ QAction *manualAction = new QAction(i18n("Check online manual"), this);
+ connect(manualAction, SIGNAL(triggered()), this, SLOT(slotOpenManual()));
+ infoMessage->addAction(manualAction);
+#endif
+ infoMessage->animatedShow();
+#else
+ m_welcomeLabel->setText(m_welcomeLabel->text() + "<br><hr />" + missing);
+ connect(m_welcomeLabel, SIGNAL(linkActivated(QString)), this, SLOT(slotOpenManual()));
+#endif
+ }
+
+}
+
+void Wizard::slotCheckPrograms()
+{
+ QSize itemSize(20, fontMetrics().height() * 2.5);
+ m_check.programList->setColumnWidth(0, 30);
+ m_check.programList->setIconSize(QSize(24, 24));
+
+ QTreeWidgetItem *item = new QTreeWidgetItem(m_check.programList, QStringList() << QString() << i18n("FFmpeg & ffplay"));
+ item->setData(1, Qt::UserRole, i18n("Required for proxy clips, transcoding and screen capture"));
+ item->setSizeHint(0, itemSize);
+ QString exepath = KStandardDirs::findExe("ffmpeg");
+ QString playpath = KStandardDirs::findExe("ffplay");
+ item->setIcon(0, m_okIcon);
+ if (exepath.isEmpty()) {
+ // Check for libav version
+ exepath = KStandardDirs::findExe("avconv");
+ if (exepath.isEmpty()) item->setIcon(0, m_badIcon);
+ }
+ if (playpath.isEmpty()) {
+ // Check for libav version
+ playpath = KStandardDirs::findExe("avplay");
+ if (playpath.isEmpty()) item->setIcon(0, m_badIcon);
+ }
+ if (!exepath.isEmpty()) KdenliveSettings::setFfmpegpath(exepath);
+ if (!playpath.isEmpty()) KdenliveSettings::setFfplaypath(playpath);
+
+#ifndef Q_WS_MAC
+ item = new QTreeWidgetItem(m_check.programList, QStringList() << QString() << i18n("dvgrab"));
+ item->setData(1, Qt::UserRole, i18n("Required for firewire capture"));
+ item->setSizeHint(0, itemSize);
+ if (KStandardDirs::findExe("dvgrab").isEmpty()) item->setIcon(0, m_badIcon);
+ else item->setIcon(0, m_okIcon);
+#endif
+
+ item = new QTreeWidgetItem(m_check.programList, QStringList() << QString() << i18n("dvdauthor"));
+ item->setData(1, Qt::UserRole, i18n("Required for creation of DVD"));
+ item->setSizeHint(0, itemSize);
+ if (KStandardDirs::findExe("dvdauthor").isEmpty()) item->setIcon(0, m_badIcon);
+ else item->setIcon(0, m_okIcon);
+
+
+ item = new QTreeWidgetItem(m_check.programList, QStringList() << QString() << i18n("genisoimage or mkisofs"));
+ item->setData(1, Qt::UserRole, i18n("Required for creation of DVD ISO images"));
+ item->setSizeHint(0, itemSize);
+ if (KStandardDirs::findExe("genisoimage").isEmpty()) {
+ // no GenIso, check for mkisofs
+ if (!KStandardDirs::findExe("mkisofs").isEmpty()) {
+ item->setIcon(0, m_okIcon);
+ } else item->setIcon(0, m_badIcon);
+ } else item->setIcon(0, m_okIcon);
+
+ item = new QTreeWidgetItem(m_check.programList, QStringList() << QString() << i18n("xine"));
+ item->setData(1, Qt::UserRole, i18n("Required to preview your DVD"));
+ item->setSizeHint(0, itemSize);
+ if (KStandardDirs::findExe("xine").isEmpty()) {
+ if (!KStandardDirs::findExe("vlc").isEmpty()) {
+ item->setText(1, i18n("vlc"));
+ item->setIcon(0, m_okIcon);
+ }
+ else item->setIcon(0, m_badIcon);
+ }
+ else item->setIcon(0, m_okIcon);
+
+ // set up some default applications
+ QString program;
+ if (KdenliveSettings::defaultimageapp().isEmpty()) {
+ program = KStandardDirs::findExe("gimp");
+ if (program.isEmpty()) program = KStandardDirs::findExe("krita");
+ if (!program.isEmpty()) KdenliveSettings::setDefaultimageapp(program);
+ }
+ if (KdenliveSettings::defaultaudioapp().isEmpty()) {
+ program = KStandardDirs::findExe("audacity");
+ if (program.isEmpty()) program = KStandardDirs::findExe("traverso");
+ if (!program.isEmpty()) KdenliveSettings::setDefaultaudioapp(program);
+ }
+ if (KdenliveSettings::defaultplayerapp().isEmpty()) {
+ KService::Ptr offer = KMimeTypeTrader::self()->preferredService("video/mpeg");
+ if (offer)
+ KdenliveSettings::setDefaultplayerapp(KRun::binaryName(offer->exec(), true));
+ }
+}
+
+void Wizard::installExtraMimes(const QString &baseName, const QStringList &globs)
+{