X-Git-Url: https://git.sesse.net/?a=blobdiff_plain;f=src%2Fdocumentvalidator.cpp;h=185d597e36934db512640e7c1b97a17ca1dd992d;hb=b3b1b1c1d6e229744ebd10093305c69fbf7ae9b0;hp=251021a554c84b51ffea6939ca8580c2d784c867;hpb=b61080cba46daa3b44dcc7992c68f91534b3ab27;p=kdenlive diff --git a/src/documentvalidator.cpp b/src/documentvalidator.cpp index 251021a5..185d597e 100644 --- a/src/documentvalidator.cpp +++ b/src/documentvalidator.cpp @@ -20,6 +20,7 @@ #include "documentvalidator.h" #include "definitions.h" +#include "initeffects.h" #include #include @@ -28,6 +29,12 @@ #include #include +#include + +#include + +#include "locale.h" + DocumentValidator::DocumentValidator(QDomDocument doc): m_doc(doc), @@ -36,36 +43,57 @@ DocumentValidator::DocumentValidator(QDomDocument doc): bool DocumentValidator::validate(const double currentVersion) { + QDomElement mlt = m_doc.firstChildElement("mlt"); + // At least the root element must be there + if (mlt.isNull()) + return false; + + QDomElement kdenliveDoc = mlt.firstChildElement("kdenlivedoc"); // Check if we're validating a Kdenlive project - if (!isProject()) + if (kdenliveDoc.isNull()) return false; + // Previous MLT / Kdenlive versions used C locale by default + QLocale documentLocale = QLocale::c(); + + if (mlt.hasAttribute("LC_NUMERIC")) { + // Set locale for the document + // WARNING: what should be done in case the locale does not exist on the system? + setlocale(LC_NUMERIC, mlt.attribute("LC_NUMERIC").toUtf8().constData()); + documentLocale = QLocale(mlt.attribute("LC_NUMERIC")); + } + + documentLocale.setNumberOptions(QLocale::OmitGroupSeparator); + + if (documentLocale != QLocale()) { + QLocale::setDefault(documentLocale); + // locale conversion might need to be redone + initEffects::parseEffectFiles(); + } + + // TODO: remove after string freeze + if (0) + KMessageBox::sorry(kapp->activeWindow(), i18n("The document you are opening uses a different locale (%1) than your system. You can only open and render it, no editing is supported unless you change your system's locale.", mlt.attribute("LC_NUMERIC")), i18n("Read only project")); + // Upgrade the document to the latest version - QDomNode kdenlivedocNode = m_doc.elementsByTagName("kdenlivedoc").at(0); - QDomElement kdenlivedocElm = kdenlivedocNode.toElement(); - if (!upgrade(kdenlivedocElm.attribute("version").toDouble(), currentVersion)) + if (!upgrade(documentLocale.toDouble(kdenliveDoc.attribute("version")), currentVersion)) return false; /* * Check the syntax (this will be replaced by XSD validation with Qt 4.6) * and correct some errors */ - QDomNode mltNode = m_doc.elementsByTagName("mlt").at(0); - QDomElement mltElm = mltNode.toElement(); - if (mltElm.isNull()) // At least the root element must be there - return false; - else { + { // Return (or create) the tractor - QDomNode tractorNode = m_doc.elementsByTagName("tractor").at(0); - QDomElement tractorElm = tractorNode.toElement(); - if (tractorElm.isNull()) { + QDomElement tractor = mlt.firstChildElement("tractor"); + if (tractor.isNull()) { m_modified = true; - tractorElm = m_doc.createElement("tractor"); - tractorElm.setAttribute("global_feed", "1"); - tractorElm.setAttribute("in", "0"); - tractorElm.setAttribute("out", "-1"); - tractorElm.setAttribute("id", "maintractor"); - mltElm.appendChild(tractorElm); + tractor = m_doc.createElement("tractor"); + tractor.setAttribute("global_feed", "1"); + tractor.setAttribute("in", "0"); + tractor.setAttribute("out", "-1"); + tractor.setAttribute("id", "maintractor"); + mlt.appendChild(tractor); } /* @@ -74,9 +102,9 @@ bool DocumentValidator::validate(const double currentVersion) */ QDomNodeList playlists = m_doc.elementsByTagName("playlist"); int tracksMax = playlists.count() - 1; // Remove the black track - QDomNodeList tracks = m_doc.elementsByTagName("track"); + QDomNodeList tracks = tractor.elementsByTagName("track"); tracksMax = qMax(tracks.count() - 1, tracksMax); - QDomNodeList tracksinfo = m_doc.elementsByTagName("trackinfo"); + QDomNodeList tracksinfo = kdenliveDoc.elementsByTagName("trackinfo"); tracksMax = qMax(tracksinfo.count(), tracksMax); tracksMax = qMax(1, tracksMax); // Force existance of one track if (playlists.count() - 1 < tracksMax || @@ -90,7 +118,7 @@ bool DocumentValidator::validate(const double currentVersion) // Looks like one MLT track is missing, remove the extra Kdenlive track if there is one. if (tracksinfo.count() != tracks.count() - 1) { // The Kdenlive tracks are not ok, clear and rebuild them - QDomNode tinfo = m_doc.elementsByTagName("tracksinfo").at(0); + QDomNode tinfo = kdenliveDoc.elementsByTagName("tracksinfo").at(0); QDomNode tnode = tinfo.firstChild(); while (!tnode.isNull()) { tinfo.removeChild(tnode); @@ -120,22 +148,21 @@ bool DocumentValidator::validate(const double currentVersion) difference = tracksMax - (playlists.count() - 1); for (int i = 0; i < difference; ++i) { QDomElement playlist = m_doc.createElement("playlist"); - mltElm.appendChild(playlist); + mlt.appendChild(playlist); } } if (tracks.count() - 1 < tracksMax) { difference = tracksMax - (tracks.count() - 1); for (int i = 0; i < difference; ++i) { QDomElement track = m_doc.createElement("track"); - tractorElm.appendChild(track); + tractor.appendChild(track); } } if (tracksinfo.count() < tracksMax) { - QDomNode tracksinfoNode = m_doc.elementsByTagName("tracksinfo").at(0); - QDomElement tracksinfoElm = tracksinfoNode.toElement(); + QDomElement tracksinfoElm = kdenliveDoc.firstChildElement("tracksinfo"); if (tracksinfoElm.isNull()) { tracksinfoElm = m_doc.createElement("tracksinfo"); - kdenlivedocElm.appendChild(tracksinfoElm); + kdenliveDoc.appendChild(tracksinfoElm); } difference = tracksMax - tracksinfo.count(); for (int i = 0; i < difference; ++i) { @@ -145,10 +172,11 @@ bool DocumentValidator::validate(const double currentVersion) tracksinfoElm.appendChild(trackinfo); } } - } + } // TODO: check the tracks references // TODO: check internal mix transitions + } return true; @@ -166,7 +194,7 @@ bool DocumentValidator::upgrade(double version, const double currentVersion) // The document is too new if (version > currentVersion) { kDebug() << "Unable to open document with version " << version; - KMessageBox::sorry(kapp->activeWindow(), i18n("This project type is unsupported (version %1) and can't be loaded.\nPlease consider upgrading you Kdenlive version.", version), i18n("Unable to open project")); + KMessageBox::sorry(kapp->activeWindow(), i18n("This project type is unsupported (version %1) and can't be loaded.\nPlease consider upgrading your Kdenlive version.", version), i18n("Unable to open project")); return false; } @@ -814,7 +842,84 @@ bool DocumentValidator::upgrade(double version, const double currentVersion) } } } + if (version <= 0.85) { + // update the LADSPA effects to use the new ladspa.id format instead of external xml file + QDomNodeList effectNodes = m_doc.elementsByTagName("filter"); + for (int i = 0; i < effectNodes.count(); ++i) { + QDomElement effect = effectNodes.at(i).toElement(); + if (EffectsList::property(effect, "mlt_service") == "ladspa") { + // Needs to be converted + QStringList info = getInfoFromEffectName(EffectsList::property(effect, "kdenlive_id")); + if (info.isEmpty()) continue; + // info contains the correct ladspa.id from kdenlive effect name, and a list of parameter's old and new names + EffectsList::setProperty(effect, "kdenlive_id", info.at(0)); + EffectsList::setProperty(effect, "tag", info.at(0)); + EffectsList::setProperty(effect, "mlt_service", info.at(0)); + EffectsList::removeProperty(effect, "src"); + for (int j = 1; j < info.size(); j++) { + QString value = EffectsList::property(effect, info.at(j).section('=', 0, 0)); + if (!value.isEmpty()) { + // update parameter name + EffectsList::renameProperty(effect, info.at(j).section('=', 0, 0), info.at(j).section('=', 1, 1)); + } + } + } + } + } + + if (version <= 0.86) { + // Make sure we don't have avformat-novalidate producers, since it caused crashes + QDomNodeList producers = m_doc.elementsByTagName("producer"); + int max = producers.count(); + for (int i = 0; i < max; i++) { + QDomElement prod = producers.at(i).toElement(); + if (EffectsList::property(prod, "mlt_service") == "avformat-novalidate") + EffectsList::setProperty(prod, "mlt_service", "avformat"); + } + + // There was a mistake in Geometry transitions where the last keyframe was created one frame after the end of transition, so fix it and move last keyframe to real end of transition + + // Get profile info (width / height) + int profileWidth; + int profileHeight; + QDomElement profile = m_doc.firstChildElement("profile"); + if (profile.isNull()) profile = infoXml.firstChildElement("profileinfo"); + if (profile.isNull()) { + // could not find profile info, set PAL + profileWidth = 720; + profileHeight = 576; + } + else { + profileWidth = profile.attribute("width").toInt(); + profileHeight = profile.attribute("height").toInt(); + } + QDomNodeList transitions = m_doc.elementsByTagName("transition"); + max = transitions.count(); + int out; + for (int i = 0; i < max; i++) { + QDomElement trans = transitions.at(i).toElement(); + out = trans.attribute("out").toInt() - trans.attribute("in").toInt(); + QString geom = EffectsList::property(trans, "geometry"); + Mlt::Geometry *g = new Mlt::Geometry(geom.toUtf8().data(), out, profileWidth, profileHeight); + Mlt::GeometryItem item; + if (g->next_key(&item, out) == 0) { + // We have a keyframe just after last frame, try to move it to last frame + if (item.frame() == out + 1) { + item.frame(out); + g->insert(item); + g->remove(out + 1); + EffectsList::setProperty(trans, "geometry", g->serialise()); + } + } + delete g; + } + } + if (version <= 0.87) { + if (!m_doc.firstChildElement("mlt").hasAttribute("LC_NUMERIC")) { + m_doc.firstChildElement("mlt").setAttribute("LC_NUMERIC", "C"); + } + } // The document has been converted: mark it as modified infoXml.setAttribute("version", currentVersion); @@ -822,6 +927,84 @@ bool DocumentValidator::upgrade(double version, const double currentVersion) return true; } +QStringList DocumentValidator::getInfoFromEffectName(const QString oldName) +{ + QStringList info; + // Returns a list to convert old Kdenlive ladspa effects + if (oldName == "pitch_shift") { + info << "ladspa.1433"; + info << "pitch=0"; + } + else if (oldName == "vinyl") { + info << "ladspa.1905"; + info << "year=0"; + info << "rpm=1"; + info << "warping=2"; + info << "crackle=3"; + info << "wear=4"; + } + else if (oldName == "room_reverb") { + info << "ladspa.1216"; + info << "room=0"; + info << "delay=1"; + info << "damp=2"; + } + else if (oldName == "reverb") { + info << "ladspa.1423"; + info << "room=0"; + info << "damp=1"; + } + else if (oldName == "rate_scale") { + info << "ladspa.1417"; + info << "rate=0"; + } + else if (oldName == "pitch_scale") { + info << "ladspa.1193"; + info << "coef=0"; + } + else if (oldName == "phaser") { + info << "ladspa.1217"; + info << "rate=0"; + info << "depth=1"; + info << "feedback=2"; + info << "spread=3"; + } + else if (oldName == "limiter") { + info << "ladspa.1913"; + info << "gain=0"; + info << "limit=1"; + info << "release=2"; + } + else if (oldName == "equalizer_15") { + info << "ladspa.1197"; + info << "1=0"; + info << "2=1"; + info << "3=2"; + info << "4=3"; + info << "5=4"; + info << "6=5"; + info << "7=6"; + info << "8=7"; + info << "9=8"; + info << "10=9"; + info << "11=10"; + info << "12=11"; + info << "13=12"; + info << "14=13"; + info << "15=14"; + } + else if (oldName == "equalizer") { + info << "ladspa.1901"; + info << "logain=0"; + info << "midgain=1"; + info << "higain=2"; + } + else if (oldName == "declipper") { + info << "ladspa.1195"; + } + return info; +} + QString DocumentValidator::colorToString(const QColor& c) { QString ret = "%1,%2,%3,%4";