#include "documentvalidator.h"
#include "definitions.h"
+#include "initeffects.h"
#include <KDebug>
#include <KMessageBox>
#include <QFile>
#include <QColor>
+#include <QString>
+
+#include <mlt++/Mlt.h>
+
+#include "locale.h"
+
DocumentValidator::DocumentValidator(QDomDocument doc):
m_doc(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);
}
/*
*/
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 ||
tracks.count() - 1 < tracksMax ||
tracksinfo.count() < tracksMax) {
+ kDebug() << "//// WARNING, PROJECT IS CORRUPTED, MISSING TRACK";
m_modified = true;
int difference;
+ // use the MLT tracks as reference
+ if (tracks.count() - 1 < tracksMax) {
+ // 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 = kdenliveDoc.elementsByTagName("tracksinfo").at(0);
+ QDomNode tnode = tinfo.firstChild();
+ while (!tnode.isNull()) {
+ tinfo.removeChild(tnode);
+ tnode = tinfo.firstChild();
+ }
+
+ for (int i = 1; i < tracks.count(); i++) {
+ QString hide = tracks.at(i).toElement().attribute("hide");
+ QDomElement newTrack = m_doc.createElement("trackinfo");
+ if (hide == "video") {
+ // audio track;
+ newTrack.setAttribute("type", "audio");
+ newTrack.setAttribute("blind", 1);
+ newTrack.setAttribute("mute", 0);
+ newTrack.setAttribute("lock", 0);
+ } else {
+ newTrack.setAttribute("blind", 0);
+ newTrack.setAttribute("mute", 0);
+ newTrack.setAttribute("lock", 0);
+ }
+ tinfo.appendChild(newTrack);
+ }
+ }
+ }
+
if (playlists.count() - 1 < tracksMax) {
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) {
tracksinfoElm.appendChild(trackinfo);
}
}
- }
+ }
// TODO: check the tracks references
// TODO: check internal mix transitions
+
}
return true;
// No conversion needed
if (version == currentVersion) {
- // TODO: uncomment when currentVersion == 0.84
- //return true;
+ return true;
}
// 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;
}
// <kdenlivedoc />
QDomNode infoXmlNode = m_doc.elementsByTagName("kdenlivedoc").at(0);
QDomElement infoXml = infoXmlNode.toElement();
+ infoXml.setAttribute("upgraded", "1");
if (version <= 0.6) {
QDomElement infoXml_old = infoXmlNode.cloneNode(true).toElement(); // Needed for folders
if (version <= 0.83) {
// Replace point size with pixel size in text titles
if (m_doc.toString().contains("font-size")) {
- KMessageBox::ButtonCode convert;
+ KMessageBox::ButtonCode convert = KMessageBox::Continue;
QDomNodeList kproducerNodes = m_doc.elementsByTagName("kdenlive_producer");
for (int i = 0; i < kproducerNodes.count() && convert != KMessageBox::No; ++i) {
QDomElement kproducer = kproducerNodes.at(i).toElement();
docProperties.setAttribute("zonein", infoXml.attribute("zonein"));
docProperties.setAttribute("zoneout", infoXml.attribute("zoneout"));
docProperties.setAttribute("zoom", infoXml.attribute("zoom"));
+ docProperties.setAttribute("position", infoXml.attribute("position"));
infoXml.appendChild(docProperties);
}
}
+ if (version <= 0.84) {
+ // update the title clips to use the new MLT kdenlivetitle producer
+ QDomNodeList kproducerNodes = m_doc.elementsByTagName("kdenlive_producer");
+ for (int i = 0; i < kproducerNodes.count(); ++i) {
+ QDomElement kproducer = kproducerNodes.at(i).toElement();
+ if (kproducer.attribute("type").toInt() == TEXT) {
+ QString data = kproducer.attribute("xmldata");
+ QString datafile = kproducer.attribute("resource");
+ if (!datafile.endsWith(".kdenlivetitle")) {
+ datafile = QString();
+ kproducer.setAttribute("resource", QString());
+ }
+ QString id = kproducer.attribute("id");
+ QDomNodeList mltproducers = m_doc.elementsByTagName("producer");
+ bool foundData = false;
+ bool foundResource = false;
+ bool foundService = false;
+ for (int j = 0; j < mltproducers.count(); j++) {
+ QDomElement wproducer = mltproducers.at(j).toElement();
+ if (wproducer.attribute("id") == id) {
+ QDomNodeList props = wproducer.childNodes();
+ for (int k = 0; k < props.count(); k++) {
+ if (props.at(k).toElement().attribute("name") == "xmldata") {
+ props.at(k).firstChild().setNodeValue(data);
+ foundData = true;
+ } else if (props.at(k).toElement().attribute("name") == "mlt_service") {
+ props.at(k).firstChild().setNodeValue("kdenlivetitle");
+ foundService = true;
+ } else if (props.at(k).toElement().attribute("name") == "resource") {
+ props.at(k).firstChild().setNodeValue(datafile);
+ foundResource = true;
+ }
+ }
+ if (!foundData) {
+ QDomElement e = m_doc.createElement("property");
+ e.setAttribute("name", "xmldata");
+ QDomText value = m_doc.createTextNode(data);
+ e.appendChild(value);
+ wproducer.appendChild(e);
+ }
+ if (!foundService) {
+ QDomElement e = m_doc.createElement("property");
+ e.setAttribute("name", "mlt_service");
+ QDomText value = m_doc.createTextNode("kdenlivetitle");
+ e.appendChild(value);
+ wproducer.appendChild(e);
+ }
+ if (!foundResource) {
+ QDomElement e = m_doc.createElement("property");
+ e.setAttribute("name", "resource");
+ QDomText value = m_doc.createTextNode(datafile);
+ e.appendChild(value);
+ wproducer.appendChild(e);
+ }
+ break;
+ }
+ }
+ }
+ }
+ }
+ 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);
m_modified = true;
-
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";