X-Git-Url: https://git.sesse.net/?a=blobdiff_plain;f=src%2Fdocumentchecker.cpp;h=adfb592e096b6be6b6d9d7fd25b998e57c1b59a5;hb=c7c21fd4cd2147de1670c782df81cc2323035ab5;hp=b003b67875cefe970a52bbb196cdc43ae869cddd;hpb=57aa57be75fd61a1801f715d23cf2b237e3a8562;p=kdenlive diff --git a/src/documentchecker.cpp b/src/documentchecker.cpp index b003b678..adfb592e 100644 --- a/src/documentchecker.cpp +++ b/src/documentchecker.cpp @@ -48,6 +48,7 @@ const int idRole = Qt::UserRole + 2; const int statusRole = Qt::UserRole + 3; const int typeRole = Qt::UserRole + 4; const int typeOriginalResource = Qt::UserRole + 5; +const int resetDurationRole = Qt::UserRole + 6; const int CLIPMISSING = 0; const int CLIPOK = 1; @@ -75,7 +76,6 @@ bool DocumentChecker::hasErrorInClips() QString resource; QDomNodeList documentProducers = m_doc.elementsByTagName("producer"); QList wrongDurationClips; - QList missingClips; QList missingProxies; for (int i = 0; i < m_info.count(); i++) { e = m_info.item(i).toElement(); @@ -84,6 +84,7 @@ bool DocumentChecker::hasErrorInClips() if (clipType != TEXT && clipType != IMAGE) { QString id = e.attribute("id"); int duration = e.attribute("duration").toInt(); + int mltDuration = -1; // Check that the duration is in sync between Kdenlive's info and MLT's playlist for (int j = 0; j < documentProducers.count(); j++) { QDomElement mltProd = documentProducers.at(j).toElement(); @@ -92,12 +93,23 @@ bool DocumentChecker::hasErrorInClips() if (prodId.startsWith("slowmotion")) continue; if (prodId.contains("_")) prodId = prodId.section("_", 0, 0); if (prodId != id) continue; - int mltDuration = mltProd.attribute("out").toInt() + 1; - if (mltDuration != duration && !e.hasAttribute("_mismatch")) { + if (mltDuration > 0 ) { + // We have several MLT producers for the same clip (probably track producers) + int newLength = EffectsList::property(mltProd, "length").toInt(); + if (newLength != mltDuration) { + // we have a different duration for the same clip, that is not safe + e.setAttribute("_resetDuration", 1); + } + } + mltDuration = EffectsList::property(mltProd, "length").toInt(); + if (mltDuration != duration) { // Duration mismatch e.setAttribute("_mismatch", mltDuration); + if (mltDuration == 15000) { + // a length of 15000 might indicate a wrong clip length since it is a default length + e.setAttribute("_resetDuration", 1); + } if (!wrongDurationClips.contains(e)) wrongDurationClips.append(e); - kDebug() << "WRONG DURTAION: "<button(QDialogButtonBox::Ok)->setEnabled(false); - for (int i = 0; i < missingClips.count(); i++) { - e = missingClips.at(i).toElement(); + for (int i = 0; i < m_missingClips.count(); i++) { + e = m_missingClips.at(i).toElement(); QString clipType; int t = e.attribute("type").toInt(); switch (t) { @@ -217,7 +230,7 @@ bool DocumentChecker::hasErrorInClips() item->setToolTip(0, i18n("Missing item")); } - if (missingClips.count() > 0) { + if (m_missingClips.count() > 0) { if (wrongDurationClips.count() > 0) { m_ui.infoLabel->setText(i18n("The project file contains missing clips or files and clip duration mismatch")); } @@ -233,9 +246,9 @@ bool DocumentChecker::hasErrorInClips() m_ui.infoLabel->setText(m_ui.infoLabel->text() + i18n("Missing proxies will be recreated after opening.")); } - m_ui.removeSelected->setEnabled(!missingClips.isEmpty()); - m_ui.recursiveSearch->setEnabled(!missingClips.isEmpty()); - m_ui.usePlaceholders->setEnabled(!missingClips.isEmpty()); + m_ui.removeSelected->setEnabled(!m_missingClips.isEmpty()); + m_ui.recursiveSearch->setEnabled(!m_missingClips.isEmpty() || !missingLumas.isEmpty()); + m_ui.usePlaceholders->setEnabled(!m_missingClips.isEmpty()); m_ui.fixDuration->setEnabled(!wrongDurationClips.isEmpty()); for (int i = 0; i < wrongDurationClips.count(); i++) { @@ -270,27 +283,28 @@ bool DocumentChecker::hasErrorInClips() item->setData(0, hashRole, e.attribute("file_hash")); item->setData(0, sizeRole, e.attribute("_mismatch")); e.removeAttribute("_mismatch"); + item->setData(0, resetDurationRole, (int) e.hasAttribute("_resetDuration")); + e.removeAttribute("_resetDuration"); item->setData(0, statusRole, CLIPWRONGDURATION); item->setData(0, typeRole, t); item->setData(0, idRole, e.attribute("id")); item->setToolTip(0, i18n("Duration mismatch")); } + if (missingProxies.count() > 0) { + QTreeWidgetItem *item = new QTreeWidgetItem(m_ui.treeWidget, QStringList() << i18n("Proxy clip")); + item->setIcon(0, KIcon("dialog-warning")); + item->setText(1, i18n("%1 missing proxy clips, will be recreated on project opening", missingProxies.count())); + item->setData(0, hashRole, e.attribute("file_hash")); + item->setData(0, statusRole, PROXYMISSING); + item->setToolTip(0, i18n("Missing proxy")); + } for (int i = 0; i < missingProxies.count(); i++) { e = missingProxies.at(i).toElement(); QString clipType; - int t = e.attribute("type").toInt(); QString realPath = e.attribute("resource"); QString id = e.attribute("id"); - QTreeWidgetItem *item = new QTreeWidgetItem(m_ui.treeWidget, QStringList() << i18n("Proxy clip")); - item->setIcon(0, KIcon("dialog-close")); - item->setText(1, e.attribute("proxy")); - item->setData(0, hashRole, e.attribute("file_hash")); - item->setData(0, statusRole, PROXYMISSING); - item->setData(0, typeRole, t); - item->setData(0, idRole, id); - item->setToolTip(0, i18n("Missing proxy")); // Replace proxy url with real clip in MLT producers QDomNodeList properties; QDomElement mltProd; @@ -419,6 +433,13 @@ QString DocumentChecker::searchLuma(QString file) const else searchPath.cd("../lumas/NTSC"); QString result = searchPath.path(KUrl::AddTrailingSlash) + KUrl(file).fileName(); + if (QFile::exists(result)) + return result; + // try to find luma in application path + searchPath.clear(); + searchPath = KUrl(QCoreApplication::applicationDirPath()); + searchPath.cd("../share/apps/kdenlive/lumas"); + result = searchPath.path(KUrl::AddTrailingSlash) + KUrl(file).fileName(); if (QFile::exists(result)) return result; return QString(); @@ -577,6 +598,7 @@ void DocumentChecker::acceptDialog() } else if (child->data(0, statusRole).toInt() == LUMAOK) { for (int i = 0; i < trans.count(); i++) { QString luma = getProperty(trans.at(i).toElement(), "luma"); + kDebug() << "luma: " << luma; if (!luma.isEmpty() && luma == child->data(0, idRole).toString()) { setProperty(trans.at(i).toElement(), "luma", child->text(1)); @@ -619,18 +641,38 @@ void DocumentChecker::slotFixDuration() { int ix = 0; QTreeWidgetItem *child = m_ui.treeWidget->topLevelItem(ix); + QDomNodeList documentProducers = m_doc.elementsByTagName("producer"); while (child) { if (child->data(0, statusRole).toInt() == CLIPWRONGDURATION) { - child->setData(0, statusRole, CLIPOK); - child->setIcon(0, KIcon("dialog-ok")); QString id = child->data(0, idRole).toString(); + bool resetDuration = child->data(0, resetDurationRole).toInt(); + for (int i = 0; i < m_info.count(); i++) { QDomElement e = m_info.at(i).toElement(); if (e.attribute("id") == id) { - e.setAttribute("duration", child->data(0, sizeRole).toString()); + if (m_missingClips.contains(e)) { + // we cannot fix duration of missing clips + resetDuration = false; + } + else { + if (resetDuration) e.removeAttribute("duration"); + else e.setAttribute("duration", child->data(0, sizeRole).toString()); + child->setData(0, statusRole, CLIPOK); + child->setIcon(0, KIcon("dialog-ok")); + } break; } } + if (resetDuration) { + // something is wrong in clip durations, so remove them so mlt fetches them again + for (int j = 0; j < documentProducers.count(); j++) { + QDomElement mltProd = documentProducers.at(j).toElement(); + QString prodId = mltProd.attribute("id"); + if (prodId == id || prodId.startsWith(id + "_")) { + EffectsList::removeProperty(mltProd, "length"); + } + } + } } ix++; child = m_ui.treeWidget->topLevelItem(ix); @@ -666,15 +708,32 @@ void DocumentChecker::slotDeleteSelected() if (KMessageBox::warningContinueCancel(m_dialog, i18np("This will remove the selected clip from this project", "This will remove the selected clips from this project", m_ui.treeWidget->selectedItems().count()), i18n("Remove clips")) == KMessageBox::Cancel) return; QStringList deletedIds; + QStringList deletedLumas; QDomNodeList playlists = m_doc.elementsByTagName("playlist"); foreach(QTreeWidgetItem *child, m_ui.treeWidget->selectedItems()) { - if (child->data(0, statusRole).toInt() < 10) { + int id = child->data(0, statusRole).toInt(); + if (id == CLIPMISSING) { deletedIds.append(child->data(0, idRole).toString()); delete child; } + else if (id == LUMAMISSING) { + deletedLumas.append(child->data(0, idRole).toString()); + delete child; + } + } + + if (!deletedLumas.isEmpty()) { + QDomElement e; + QDomNodeList transitions = m_doc.elementsByTagName("transition"); + foreach (QString lumaPath, deletedLumas) { + for (int i = 0; i < transitions.count(); i++) { + e = transitions.item(i).toElement(); + QString resource = EffectsList::property(e, "luma"); + if (resource == lumaPath) EffectsList::removeProperty(e, "luma"); + } + } } - kDebug() << "// Clips to delete: " << deletedIds; if (!deletedIds.isEmpty()) { QDomElement e; @@ -726,7 +785,7 @@ void DocumentChecker::slotDeleteSelected() } } -void DocumentChecker::checkMissingImages(QList &missingClips, QStringList images, QStringList fonts, QString id, QString baseClip) +void DocumentChecker::checkMissingImages(QStringList images, QStringList fonts, QString id, QString baseClip) { QDomDocument doc; foreach(const QString &img, images) { @@ -736,7 +795,7 @@ void DocumentChecker::checkMissingImages(QList &missingClips, QStri e.setAttribute("resource", img); e.setAttribute("id", id); e.setAttribute("name", baseClip); - missingClips.append(e); + m_missingClips.append(e); } } kDebug() << "/ / / CHK FONTS: " << fonts; @@ -749,7 +808,7 @@ void DocumentChecker::checkMissingImages(QList &missingClips, QStri e.setAttribute("resource", fontelement); e.setAttribute("id", id); e.setAttribute("name", baseClip); - missingClips.append(e); + m_missingClips.append(e); } } } @@ -760,7 +819,8 @@ void DocumentChecker::slotCheckButtons() if (m_ui.treeWidget->currentItem()) { QTreeWidgetItem *item = m_ui.treeWidget->currentItem(); int t = item->data(0, typeRole).toInt(); - if (t == TITLE_FONT_ELEMENT || t == TITLE_IMAGE_ELEMENT) { + int s = item->data(0, statusRole).toInt(); + if (t == TITLE_FONT_ELEMENT || t == TITLE_IMAGE_ELEMENT || s == PROXYMISSING) { m_ui.removeSelected->setEnabled(false); } else m_ui.removeSelected->setEnabled(true); }