X-Git-Url: https://git.sesse.net/?a=blobdiff_plain;f=src%2Finiteffects.cpp;h=a10765836a5ef856d8561cc90cba71205b260455;hb=6a33521fe12af44ab9abe8e09c6e007f4093b2da;hp=5d6ee274ce571e0ac03e4a0478f6d5ef69efaaa0;hpb=c5720a8894dd5a0dbd06f3330299dc669454fa06;p=kdenlive diff --git a/src/initeffects.cpp b/src/initeffects.cpp index 5d6ee274..a1076583 100644 --- a/src/initeffects.cpp +++ b/src/initeffects.cpp @@ -17,34 +17,36 @@ * * ***************************************************************************/ - -#include -#include -#include -#include - -#include -#include -#include - #include "initeffects.h" #include "kdenlivesettings.h" #include "effectslist.h" #include "effectstackedit.h" #include "mainwindow.h" -initEffectsThumbnailer::initEffectsThumbnailer() { +#include +#include +#include + +#include +#include +#include +#include +initEffectsThumbnailer::initEffectsThumbnailer() : + QThread() +{ } -void initEffectsThumbnailer::prepareThumbnailsCall(const QStringList& list) { +void initEffectsThumbnailer::prepareThumbnailsCall(const QStringList& list) +{ m_list = list; start(); kDebug() << "done"; } -void initEffectsThumbnailer::run() { - foreach(const QString &entry, m_list) { +void initEffectsThumbnailer::run() +{ + foreach(const QString & entry, m_list) { kDebug() << entry; if (!entry.isEmpty() && (entry.endsWith(".png") || entry.endsWith(".pgm"))) { if (!EffectStackEdit::iconCache.contains(entry)) { @@ -59,42 +61,37 @@ void initEffectsThumbnailer::run() { initEffectsThumbnailer initEffects::thumbnailer; -initEffects::initEffects() { - -} - -initEffects::~initEffects() { -} - - // static -void initEffects::refreshLumas() { - - // Check for Kdenlive installed luma files - QStringList imagenamelist; - QStringList imagefiles; +void initEffects::refreshLumas() +{ + // Check for Kdenlive installed luma files, add empty string at start for no luma + QStringList imagenamelist = QStringList() << i18n("None"); + QStringList imagefiles = QStringList() << QString(); QStringList filters; filters << "*.pgm" << "*.png"; QStringList customLumas = KGlobal::dirs()->findDirs("appdata", "lumas"); - foreach(const QString &folder, customLumas) { + foreach(const QString & folder, customLumas) { QStringList filesnames = QDir(folder).entryList(filters, QDir::Files); - foreach(const QString &fname, filesnames) { + foreach(const QString & fname, filesnames) { imagenamelist.append(fname); - imagefiles.append(folder + '/' + fname); + imagefiles.append(KUrl(folder).path(KUrl::AddTrailingSlash) + fname); } } // Check for MLT lumas - QString folder = mlt_environment("MLT_DATA"); - folder.append("/lumas/").append(mlt_environment("MLT_NORMALISATION")); - QDir lumafolder(folder); + KUrl folder(mlt_environment("MLT_DATA")); + folder.addPath("lumas"); + folder.addPath(mlt_environment("MLT_NORMALISATION")); + QDir lumafolder(folder.path()); QStringList filesnames = lumafolder.entryList(filters, QDir::Files); - foreach(const QString &fname, filesnames) { + foreach(const QString & fname, filesnames) { imagenamelist.append(fname); - imagefiles.append(folder + '/' + fname); + KUrl path(folder); + path.addPath(fname); + imagefiles.append(path.toLocalFile()); } - QDomElement lumaTransition = MainWindow::transitions.getEffectByTag("luma", QString()); + QDomElement lumaTransition = MainWindow::transitions.getEffectByTag("luma", "luma"); QDomNodeList params = lumaTransition.elementsByTagName("parameter"); for (int i = 0; i < params.count(); i++) { QDomElement e = params.item(i).toElement(); @@ -105,144 +102,223 @@ void initEffects::refreshLumas() { } } + QDomElement compositeTransition = MainWindow::transitions.getEffectByTag("composite", "composite"); + params = compositeTransition.elementsByTagName("parameter"); + for (int i = 0; i < params.count(); i++) { + QDomElement e = params.item(i).toElement(); + if (e.attribute("tag") == "luma") { + e.setAttribute("paramlistdisplay", imagenamelist.join(",")); + e.setAttribute("paramlist", imagefiles.join(",")); + break; + } + } +} +// static +QDomDocument initEffects::getUsedCustomEffects(QMap effectids) +{ + QMapIterator i(effectids); + int ix; + QDomDocument doc; + QDomElement list = doc.createElement("customeffects"); + doc.appendChild(list); + while (i.hasNext()) { + i.next(); + ix = MainWindow::customEffects.hasEffect(i.value(), i.key()); + if (ix > -1) { + QDomElement e = MainWindow::customEffects.at(ix); + list.appendChild(doc.importNode(e, true)); + } + } + return doc; } //static -Mlt::Repository *initEffects::parseEffectFiles() { +Mlt::Repository *initEffects::parseEffectFiles() +{ QStringList::Iterator more; QStringList::Iterator it; QStringList fileList; QString itemName; - - // Build effects. Retrieve the list of MLT's available effects first. Mlt::Repository *repository = Mlt::Factory::init(); if (!repository) { - kDebug() << "Repository did not finish init " ; + kDebug() << "Repository didn't finish initialisation" ; return NULL; } + + // Retrieve the list of MLT's available effects. Mlt::Properties *filters = repository->filters(); QStringList filtersList; - - // Check for blacklisted effects - QString blacklist = KStandardDirs::locate("appdata", "blacklisted_effects.txt"); - - QFile file(blacklist); - QStringList blackListed; - - if (file.open(QIODevice::ReadOnly)) { - QTextStream in(&file); - while (!in.atEnd()) { - QString black = in.readLine().simplified(); - if (!black.isEmpty() && !black.startsWith('#')) blackListed.append(black); - } - } - file.close(); - - // Check for blacklisted transitions - blacklist = KStandardDirs::locate("appdata", "blacklisted_transitions.txt"); - - QFile file2(blacklist); - QStringList blackListedtransitions; - - if (file2.open(QIODevice::ReadOnly)) { - QTextStream in(&file2); - while (!in.atEnd()) { - QString black = in.readLine().simplified(); - if (!black.isEmpty() && !black.startsWith('#')) blackListedtransitions.append(black); - } - } - file2.close(); - - for (int i = 0 ; i < filters->count() ; i++) { + for (int i = 0; i < filters->count(); ++i) filtersList << filters->get_name(i); - } - - // Build effects. check producers first. + delete filters; + // Retrieve the list of available producers. Mlt::Properties *producers = repository->producers(); QStringList producersList; - for (int i = 0 ; i < producers->count() ; i++) { + for (int i = 0; i < producers->count(); ++i) producersList << producers->get_name(i); - } - delete filters; + KdenliveSettings::setProducerslist(producersList); delete producers; + // Retrieve the list of available transitions. Mlt::Properties *transitions = repository->transitions(); QStringList transitionsItemList; - for (int i = 0 ; i < transitions->count() ; i++) { + for (int i = 0; i < transitions->count(); ++i) transitionsItemList << transitions->get_name(i); - } delete transitions; - foreach(const QString &trans, blackListedtransitions) { - if (transitionsItemList.contains(trans)) transitionsItemList.removeAll(trans); + // Remove blacklisted transitions from the list. + QFile file(KStandardDirs::locate("appdata", "blacklisted_transitions.txt")); + if (file.open(QIODevice::ReadOnly)) { + QTextStream in(&file); + while (!in.atEnd()) { + QString black = in.readLine().simplified(); + if (!black.isEmpty() && !black.startsWith('#') && + transitionsItemList.contains(black)) + transitionsItemList.removeAll(black); + } + file.close(); } + + // Fill transitions list. fillTransitionsList(repository, &MainWindow::transitions, transitionsItemList); + // Set the directories to look into for ladspa plugins. KGlobal::dirs()->addResourceType("ladspa_plugin", 0, "lib/ladspa"); KGlobal::dirs()->addResourceDir("ladspa_plugin", "/usr/lib/ladspa"); KGlobal::dirs()->addResourceDir("ladspa_plugin", "/usr/local/lib/ladspa"); KGlobal::dirs()->addResourceDir("ladspa_plugin", "/opt/lib/ladspa"); KGlobal::dirs()->addResourceDir("ladspa_plugin", "/opt/local/lib/ladspa"); + KGlobal::dirs()->addResourceDir("ladspa_plugin", "/usr/lib64/ladspa"); + KGlobal::dirs()->addResourceDir("ladspa_plugin", "/usr/local/lib64/ladspa"); - kDebug() << "// INIT EFFECT SEARCH" << endl; + // Set the directories to look into for effects. + QStringList direc = KGlobal::dirs()->findDirs("appdata", "effects"); - QStringList direc = KGlobal::dirs()->findDirs("data", "kdenlive/effects"); - - QDir directory; - for (more = direc.begin() ; more != direc.end() ; ++more) { - directory = QDir(*more); + // Iterate through effects directories to parse all XML files. + for (more = direc.begin(); more != direc.end(); ++more) { + QDir directory(*more); QStringList filter; filter << "*.xml"; fileList = directory.entryList(filter, QDir::Files); - for (it = fileList.begin() ; it != fileList.end() ; ++it) { + for (it = fileList.begin(); it != fileList.end(); ++it) { itemName = KUrl(*more + *it).path(); - parseEffectFile(&MainWindow::customEffects, &MainWindow::audioEffects, &MainWindow::videoEffects, itemName, filtersList, producersList); - // kDebug()<<"// FOUND EFFECT FILE: "< effectsMap; + for (int i = 0; i < MainWindow::transitions.count(); ++i) { + effectInfo = MainWindow::transitions.at(i); + effectsMap.insert(effectInfo.elementsByTagName("name").item(0).toElement().text().toLower().toUtf8().data(), effectInfo); + } + MainWindow::transitions.clearList(); + foreach(const QDomElement & effect, effectsMap) + MainWindow::transitions.append(effect); + effectsMap.clear(); + for (int i = 0; i < MainWindow::customEffects.count(); ++i) { + effectInfo = MainWindow::customEffects.at(i); + effectsMap.insert(effectInfo.elementsByTagName("name").item(0).toElement().text().toLower().toUtf8().data(), effectInfo); + } + MainWindow::customEffects.clearList(); + foreach(const QDomElement & effect, effectsMap) + MainWindow::customEffects.append(effect); + effectsMap.clear(); + for (int i = 0; i < MainWindow::audioEffects.count(); ++i) { + effectInfo = MainWindow::audioEffects.at(i); + effectsMap.insert(effectInfo.elementsByTagName("name").item(0).toElement().text().toLower().toUtf8().data(), effectInfo); + } + MainWindow::audioEffects.clearList(); + foreach(const QDomElement & effect, effectsMap) + MainWindow::audioEffects.append(effect); + effectsMap.clear(); + for (int i = 0; i < MainWindow::videoEffects.count(); ++i) { + effectInfo = MainWindow::videoEffects.at(i); + effectsMap.insert(effectInfo.elementsByTagName("name").item(0).toElement().text().toLower().toUtf8().data(), effectInfo); + } + // Add remaining filters to the list of video effects. + foreach(const QString & filtername, filtersList) { QDomDocument doc = createDescriptionFromMlt(repository, "filters", filtername); if (!doc.isNull()) - MainWindow::videoEffects.append(doc.documentElement()); + effectsMap.insert(doc.documentElement().elementsByTagName("name").item(0).toElement().text().toLower().toUtf8().data(), doc.documentElement()); } + MainWindow::videoEffects.clearList(); + foreach(const QDomElement & effect, effectsMap) + MainWindow::videoEffects.append(effect); + return repository; } // static -void initEffects::parseCustomEffectsFile() { - MainWindow::customEffects.clear(); +void initEffects::parseCustomEffectsFile() +{ + MainWindow::customEffects.clearList(); + /* + * Why a QMap? See parseEffectFiles(). It's probably useless here, but we + * cannot be sure about it. + */ + QMap effectsMap; QString path = KStandardDirs::locateLocal("appdata", "effects/", true); QDir directory = QDir(path); QStringList filter; filter << "*.xml"; const QStringList fileList = directory.entryList(filter, QDir::Files); - QString itemName; - foreach(const QString filename, fileList) { - itemName = KUrl(path + filename).path(); - QDomDocument doc; + /* + * We need to declare these variables outside the foreach, or the QMap will + * refer to non existing variables (QMap::insert() takes references as + * parameters). + */ + QDomDocument doc; + QDomNodeList effects; + QDomElement e; + foreach(const QString & filename, fileList) { + QString itemName = KUrl(path + filename).path(); QFile file(itemName); doc.setContent(&file, false); file.close(); - QDomNodeList effects = doc.elementsByTagName("effect"); + effects = doc.elementsByTagName("effect"); if (effects.count() != 1) { - kDebug() << "More than one effect in file " << itemName << ", NOT SUPPORTED YET"; + kDebug() << "More than one effect in file " << itemName << ", not supported yet"; } else { - QDomElement e = effects.item(0).toElement(); - MainWindow::customEffects.append(e); + e = effects.item(0).toElement(); + effectsMap.insert(e.elementsByTagName("name").item(0).toElement().text().toLower().toUtf8().data(), e); } } + foreach(const QDomElement & effect, effectsMap) + MainWindow::customEffects.append(effect); } // static -void initEffects::parseEffectFile(EffectsList *customEffectList, EffectsList *audioEffectList, EffectsList *videoEffectList, QString name, QStringList filtersList, QStringList producersList) { +void initEffects::parseEffectFile(EffectsList *customEffectList, EffectsList *audioEffectList, EffectsList *videoEffectList, QString name, QStringList filtersList, QStringList producersList) +{ QDomDocument doc; QFile file(name); doc.setContent(&file, false); @@ -251,32 +327,32 @@ void initEffects::parseEffectFile(EffectsList *customEffectList, EffectsList *au QDomNodeList effects = doc.elementsByTagName("effect"); if (effects.count() == 0) { - kDebug() << "// EFFECT FILET: " << name << " IS BROKEN" << endl; + kDebug() << "Effect broken: " << name; return; } - QString groupName; - if (doc.elementsByTagName("effectgroup").item(0).toElement().tagName() == "effectgroup") { - groupName = documentElement.attribute("name", QString::null); - } - int i = 0; + /*QString groupName; + if (doc.elementsByTagName("effectgroup").item(0).toElement().tagName() == "effectgroup") + groupName = documentElement.attribute("name", QString());*/ - while (!effects.item(i).isNull()) { + for (int i = 0; !effects.item(i).isNull(); ++i) { documentElement = effects.item(i).toElement(); - QString tag = documentElement.attribute("tag", QString::null); + QString tag = documentElement.attribute("tag", QString()); bool ladspaOk = true; if (tag == "ladspa") { - QString library = documentElement.attribute("library", QString::null); + QString library = documentElement.attribute("library", QString()); if (KStandardDirs::locate("ladspa_plugin", library).isEmpty()) ladspaOk = false; } - // Parse effect file + // Parse effect information. if ((filtersList.contains(tag) || producersList.contains(tag)) && ladspaOk) { - bool isAudioEffect = false; - QString type = documentElement.attribute("type", QString::null); - if (type == "audio") audioEffectList->append(documentElement); - else if (type == "custom") customEffectList->append(documentElement); - else videoEffectList->append(documentElement); + QString type = documentElement.attribute("type", QString()); + if (type == "audio") + audioEffectList->append(documentElement); + else if (type == "custom") + customEffectList->append(documentElement); + else + videoEffectList->append(documentElement); } /* @@ -297,10 +373,10 @@ void initEffects::parseEffectFile(EffectsList *customEffectList, EffectsList *au QDomNode propsnode = documentElement.elementsByTagName("properties").item(0); if (!propsnode.isNull()) { QDomElement propselement = propsnode.toElement(); - id = propselement.attribute("id", QString::null); - effectTag = propselement.attribute("tag", QString::null); - if (propselement.attribute("type", QString::null) == "audio") type = AUDIOEFFECT; - else if (propselement.attribute("type", QString::null) == "custom") type = CUSTOMEFFECT; + id = propselement.attribute("id", QString()); + effectTag = propselement.attribute("tag", QString()); + if (propselement.attribute("type", QString()) == "audio") type = AUDIOEFFECT; + else if (propselement.attribute("type", QString()) == "custom") type = CUSTOMEFFECT; else type = VIDEOEFFECT; } @@ -348,12 +424,12 @@ void initEffects::parseEffectFile(EffectsList *customEffectList, EffectsList *au } effectList->append(effect); }*/ - i++; } } //static -char* initEffects::ladspaEffectString(int ladspaId, QStringList params) { +const char* initEffects::ladspaEffectString(int ladspaId, QStringList params) +{ if (ladspaId == 1433) //Pitch return ladspaPitchEffectString(params); else if (ladspaId == 1216) //Room Reverb @@ -364,13 +440,14 @@ char* initEffects::ladspaEffectString(int ladspaId, QStringList params) { return ladspaEqualizerEffectString(params); else { kDebug() << "++++++++++ ASKING FOR UNKNOWN LADSPA EFFECT: " << ladspaId << endl; - return (char *) ""; + return ""; } } //static -void initEffects::ladspaEffectFile(const QString & fname, int ladspaId, QStringList params) { - char *filterString; +void initEffects::ladspaEffectFile(const QString & fname, int ladspaId, QStringList params) +{ + const char *filterString; switch (ladspaId) { case 1433: //Pitch filterString = ladspaPitchEffectString(params); @@ -414,60 +491,71 @@ void initEffects::ladspaEffectFile(const QString & fname, int ladspaId, QStringL stream << filterString; f.close(); } else kDebug() << "++++++++++ ERROR CANNOT WRITE TO: " << KdenliveSettings::currenttmpfolder() + fname << endl; - delete filterString; + delete [] filterString; } const QString jackString = "248000"; -char* initEffects::ladspaDeclipEffectString(QStringList) { +const char* initEffects::ladspaDeclipEffectString(QStringList) +{ return qstrdup(QString(jackString + "1195truefalsetrue1.0000001.000000true").toUtf8()); } /* -char* initEffects::ladspaVocoderEffectString(QStringList params) +const char* initEffects::ladspaVocoderEffectString(QStringList params) { return qstrdup( QString(jackString + "1441truefalsetrue1.0000001.000000truetrue0.0000000.000000true%1%1true%1%1true%1%1true%1%1true%2%2true%2%2true%2%2true%2%2true%3%3true%3%3true%3%3true%3%3true%4%4true%4%4true%4%4true%4%4").arg(params[0]).arg(params[1]).arg(params[2]).arg(params[3])); }*/ -char* initEffects::ladspaVinylEffectString(QStringList params) { +const char* initEffects::ladspaVinylEffectString(QStringList params) +{ return qstrdup(QString(jackString + "1905truefalsetrue1.0000001.000000%1%2%3%4%5").arg(params[0]).arg(params[1]).arg(params[2]).arg(params[3]).arg(params[4]).toUtf8()); } -char* initEffects::ladspaPitchEffectString(QStringList params) { +const char* initEffects::ladspaPitchEffectString(QStringList params) +{ return qstrdup(QString(jackString + "1433truefalsetrue1.01.0truetrue%1%1true4.0000004.000000").arg(params[0]).toUtf8()); } -char* initEffects::ladspaRoomReverbEffectString(QStringList params) { +const char* initEffects::ladspaRoomReverbEffectString(QStringList params) +{ return qstrdup(QString(jackString + "1216truefalsetrue1.0000001.000000truetrue%1%1true%2%2true%3%3true0.7500000.750000true-70.000000-70.000000true0.0000000.000000true-17.500000-17.500000").arg(params[0]).arg(params[1]).arg(params[2]).toUtf8()); } -char* initEffects::ladspaReverbEffectString(QStringList params) { +const char* initEffects::ladspaReverbEffectString(QStringList params) +{ return qstrdup(QString(jackString + "1423true falsetrue 1.0000001.000000 truetrue%1 %1true%2%2true0.2500000.250000").arg(params[0]).arg(params[1]).toUtf8()); } -char* initEffects::ladspaEqualizerEffectString(QStringList params) { +const char* initEffects::ladspaEqualizerEffectString(QStringList params) +{ return qstrdup(QString(jackString + "1901true falsetrue 1.0000001.000000%1%2 %3").arg(params[0]).arg(params[1]).arg(params[2]).toUtf8()); } -char* initEffects::ladspaLimiterEffectString(QStringList params) { +const char* initEffects::ladspaLimiterEffectString(QStringList params) +{ return qstrdup(QString(jackString + "1913truefalsetrue1.0000001.000000%1%2%3").arg(params[0]).arg(params[1]).arg(params[2]).toUtf8()); } -char* initEffects::ladspaPitchShifterEffectString(QStringList params) { +const char* initEffects::ladspaPitchShifterEffectString(QStringList params) +{ return qstrdup(QString(jackString + "1193truefalsetrue1.0000001.000000truetrue%1%1").arg(params[0]).toUtf8()); } -char* initEffects::ladspaRateScalerEffectString(QStringList params) { +const char* initEffects::ladspaRateScalerEffectString(QStringList params) +{ return qstrdup(QString(jackString + "1417truefalsetrue1.0000001.000000truetrue%1%1").arg(params[0]).toUtf8()); } -char* initEffects::ladspaPhaserEffectString(QStringList params) { +const char* initEffects::ladspaPhaserEffectString(QStringList params) +{ return qstrdup(QString(jackString + "1217truefalsetrue1.0000001.000000truetrue%1%1true%2%2true%3%3true%4%4").arg(params[0]).arg(params[1]).arg(params[2]).arg(params[3]).toUtf8()); } -QDomDocument initEffects::createDescriptionFromMlt(Mlt::Repository* repository, const QString& type, const QString& filtername) { +QDomDocument initEffects::createDescriptionFromMlt(Mlt::Repository* repository, const QString& /*type*/, const QString& filtername) +{ QDomDocument ret; Mlt::Properties *metadata = repository->metadata(filter_type, filtername.toAscii().data()); @@ -492,7 +580,7 @@ QDomDocument initEffects::createDescriptionFromMlt(Mlt::Repository* repository, eff.appendChild(desc); Mlt::Properties param_props((mlt_properties) metadata->get_data("parameters")); - for (int j = 0; param_props.is_valid() && j < param_props.count();j++) { + for (int j = 0; param_props.is_valid() && j < param_props.count(); j++) { QDomElement params = ret.createElement("parameter"); Mlt::Properties paramdesc((mlt_properties) param_props.get_data(param_props.get_name(j))); @@ -506,8 +594,8 @@ QDomDocument initEffects::createDescriptionFromMlt(Mlt::Repository* repository, if (QString(paramdesc.get("type")) == "float") { params.setAttribute("type", "constant"); params.setAttribute("factor", "1000"); - if (paramdesc.get("maximum")) params.setAttribute("max", QString(paramdesc.get("maximum")).toFloat()*1000.0); - if (paramdesc.get("minimum")) params.setAttribute("min", QString(paramdesc.get("minimum")).toFloat()*1000.0); + if (paramdesc.get("maximum")) params.setAttribute("max", QString(paramdesc.get("maximum")).toFloat() * 1000.0); + if (paramdesc.get("minimum")) params.setAttribute("min", QString(paramdesc.get("minimum")).toFloat() * 1000.0); } if (QString(paramdesc.get("type")) == "boolean") params.setAttribute("type", "bool"); @@ -536,6 +624,8 @@ QDomDocument initEffects::createDescriptionFromMlt(Mlt::Repository* repository, ret.appendChild(eff); } } + delete metadata; + metadata = 0; /* QString outstr; QTextStream str(&outstr); ret.save(str, 2); @@ -543,39 +633,70 @@ QDomDocument initEffects::createDescriptionFromMlt(Mlt::Repository* repository, return ret; } -void initEffects::fillTransitionsList(Mlt::Repository * repository, EffectsList* transitions, QStringList names) { - // remove transitions that are not implemented +void initEffects::fillTransitionsList(Mlt::Repository *repository, EffectsList *transitions, QStringList names) +{ + // Remove transitions that are not implemented. int pos = names.indexOf("mix"); - if (pos != -1) names.takeAt(pos); - pos = names.indexOf("region"); - if (pos != -1) names.takeAt(pos); - foreach(const QString &name, names) { + if (pos != -1) + names.takeAt(pos); + + QStringList imagenamelist = QStringList() << i18n("None"); + QStringList imagefiles = QStringList() << QString(); + QStringList filters; + filters << "*.pgm" << "*.png"; + QStringList customLumas = KGlobal::dirs()->findDirs("appdata", "lumas"); + foreach(QString folder, customLumas) { + if (!folder.endsWith('/')) + folder.append('/'); + QStringList filesnames = QDir(folder).entryList(filters, QDir::Files); + foreach(const QString & fname, filesnames) { + imagenamelist.append(fname); + imagefiles.append(folder + fname); + } + } + + // Check for MLT luma files. + KUrl folder(mlt_environment("MLT_DATA")); + folder.addPath("lumas"); + folder.addPath(mlt_environment("MLT_NORMALISATION")); + QDir lumafolder(folder.path()); + QStringList filesnames = lumafolder.entryList(filters, QDir::Files); + foreach(const QString & fname, filesnames) { + imagenamelist.append(fname); + KUrl path(folder); + path.addPath(fname); + imagefiles.append(path.toLocalFile()); + } + + foreach(const QString & name, names) { QDomDocument ret; QDomElement ktrans = ret.createElement("ktransition"); ret.appendChild(ktrans); - ktrans.setAttribute("tag", name); - QDomElement tname = ret.createElement("name"); + QDomElement tname = ret.createElement("name"); QDomElement desc = ret.createElement("description"); - - QList paramList; - Mlt::Properties *metadata = repository->metadata(transition_type, name.toAscii().data()); - //kDebug() << filtername; + ktrans.appendChild(tname); + ktrans.appendChild(desc); + Mlt::Properties *metadata = repository->metadata(transition_type, name.toUtf8().data()); if (metadata && metadata->is_valid()) { - + // If possible, set name and description. + if (metadata->get("title") && metadata->get("identifier")) + tname.appendChild(ret.createTextNode(metadata->get("title"))); desc.appendChild(ret.createTextNode(metadata->get("description"))); Mlt::Properties param_props((mlt_properties) metadata->get_data("parameters")); - for (int j = 0; param_props.is_valid() && j < param_props.count();j++) { + for (int i = 0; param_props.is_valid() && i < param_props.count(); ++i) { QDomElement params = ret.createElement("parameter"); - Mlt::Properties paramdesc((mlt_properties) param_props.get_data(param_props.get_name(j))); + Mlt::Properties paramdesc((mlt_properties) param_props.get_data(param_props.get_name(i))); params.setAttribute("name", paramdesc.get("identifier")); - if (paramdesc.get("maximum")) params.setAttribute("max", paramdesc.get("maximum")); - if (paramdesc.get("minimum")) params.setAttribute("min", paramdesc.get("minimum")); + if (paramdesc.get("maximum")) + params.setAttribute("max", paramdesc.get("maximum")); + if (paramdesc.get("minimum")) + params.setAttribute("min", paramdesc.get("minimum")); if (QString(paramdesc.get("type")) == "integer") { params.setAttribute("type", "constant"); params.setAttribute("factor", "100"); @@ -585,137 +706,112 @@ void initEffects::fillTransitionsList(Mlt::Repository * repository, EffectsList* if (!QString(paramdesc.get("format")).isEmpty()) { params.setAttribute("type", "complex"); params.setAttribute("format", paramdesc.get("format")); - } - if (paramdesc.get("default")) params.setAttribute("default", paramdesc.get("default")); - if (paramdesc.get("value")) { + if (paramdesc.get("default")) + params.setAttribute("default", paramdesc.get("default")); + if (paramdesc.get("value")) params.setAttribute("value", paramdesc.get("value")); - } else { + else params.setAttribute("value", paramdesc.get("default")); - } - QDomElement pname = ret.createElement("name"); pname.appendChild(ret.createTextNode(paramdesc.get("title"))); params.appendChild(pname); - ktrans.appendChild(params); } - - ret.appendChild(ktrans); - if (metadata->get("title") && metadata->get("identifier")) { - ktrans.setAttribute("tag", name); - QDomElement tname = ret.createElement("name"); - tname.appendChild(ret.createTextNode(metadata->get("title"))); - ktrans.appendChild(tname); - } - - //kDebug() << ret.toString(); + delete metadata; + metadata = 0; } else { - if (name == "luma") { - - tname.appendChild(ret.createTextNode("Luma")); - desc.appendChild(ret.createTextNode("Applies a luma transition between the current and next frames")); - - // Check for Kdenlive installed luma files - QStringList imagenamelist; - QStringList imagefiles; - QStringList filters; - filters << "*.pgm" << "*.png"; - - QStringList customLumas = KGlobal::dirs()->findDirs("appdata", "lumas"); - foreach(const QString &folder, customLumas) { - QStringList filesnames = QDir(folder).entryList(filters, QDir::Files); - foreach(const QString &fname, filesnames) { - imagenamelist.append(fname); - imagefiles.append(folder + '/' + fname); - } - } + /* + * Check for Kdenlive installed luma files, add empty string at + * start for no luma file. + */ - // Check for MLT lumas - QString folder = mlt_environment("MLT_DATA"); - folder.append("/lumas/").append(mlt_environment("MLT_NORMALISATION")); - QDir lumafolder(folder); - QStringList filesnames = lumafolder.entryList(filters, QDir::Files); - foreach(const QString &fname, filesnames) { - imagenamelist.append(fname); - imagefiles.append(folder + '/' + fname); - } - - paramList.append(quickParameterFill(ret, "Softness", "softness", "double", "0", "0", "100", "", "", "100")); - paramList.append(quickParameterFill(ret, "Invert", "invert", "bool", "0", "0", "1")); - paramList.append(quickParameterFill(ret, "ImageFile", "resource", "list", "", "", "", imagefiles.join(","), imagenamelist.join(","))); - paramList.append(quickParameterFill(ret, "Reverse Transition", "reverse", "bool", "0", "0", "1")); + // Implement default transitions. + QList paramList; + if (name == "luma") { + ktrans.setAttribute("id", name); + tname.appendChild(ret.createTextNode(i18n("Wipe"))); + desc.appendChild(ret.createTextNode(i18n("Applies a stationary transition between the current and next frames."))); + + paramList.append(quickParameterFill(ret, i18n("Softness"), "softness", "double", "0", "0", "100", "", "", "100")); + paramList.append(quickParameterFill(ret, i18nc("@property: means that the image is inverted", "Invert"), "invert", "bool", "0", "0", "1")); + paramList.append(quickParameterFill(ret, i18n("Image File"), "resource", "list", "", "", "", imagefiles.join(","), imagenamelist.join(","))); + paramList.append(quickParameterFill(ret, i18n("Reverse Transition"), "reverse", "bool", "0", "0", "1")); //thumbnailer.prepareThumbnailsCall(imagelist); - } else if (name == "composite") { - desc.appendChild(ret.createTextNode("A key-framable alpha-channel compositor for two frames.")); - paramList.append(quickParameterFill(ret, "Geometry", "geometry", "geometry", "0%,0%:100%x100%:100", "-500;-500;-500;-500;0", "500;500;500;500;100")); - paramList.append(quickParameterFill(ret, "Distort", "distort", "bool", "1", "1", "1")); - tname.appendChild(ret.createTextNode("Composite")); - QDomDocument ret1; - QDomElement ktrans1 = ret1.createElement("ktransition"); - ret1.appendChild(ktrans1); - ktrans1.setAttribute("tag", name); - QDomElement tname1 = ret.createElement("name"); - tname1.appendChild(ret1.createTextNode("PIP")); - - } else if (name == "mix") { - tname.appendChild(ret.createTextNode("Mix")); + ktrans.setAttribute("id", name); + tname.appendChild(ret.createTextNode(i18n("Composite"))); + desc.appendChild(ret.createTextNode(i18n("A key-framable alpha-channel compositor for two frames."))); + paramList.append(quickParameterFill(ret, i18n("Geometry"), "geometry", "geometry", "0%,0%:100%x100%:100", "-500;-500;-500;-500;0", "500;500;500;500;100")); + paramList.append(quickParameterFill(ret, i18n("Alpha Channel Operation"), "operator", "list", "over", "", "", "over,and,or,xor", "over,and,or,xor")); + paramList.append(quickParameterFill(ret, i18n("Align"), "aligned", "bool", "1", "0", "1")); + paramList.append(quickParameterFill(ret, i18n("Fill"), "fill", "bool", "1", "0", "1")); + paramList.append(quickParameterFill(ret, i18n("Distort"), "distort", "bool", "0", "0", "1")); + paramList.append(quickParameterFill(ret, i18n("Wipe File"), "luma", "list", "", "", "", imagefiles.join(","), imagenamelist.join(","))); + paramList.append(quickParameterFill(ret, i18n("Wipe Softness"), "softness", "double", "0", "0", "100", "", "", "100")); + paramList.append(quickParameterFill(ret, i18n("Wipe Invert"), "luma_invert", "bool", "0", "0", "1")); + paramList.append(quickParameterFill(ret, i18n("Force Progressive Rendering"), "progressive", "bool", "1", "0", "1")); + paramList.append(quickParameterFill(ret, i18n("Force Deinterlace Overlay"), "deinterlace", "bool", "0", "0", "1")); } else if (name == "affine") { - tname.appendChild(ret.createTextNode("Affine")); - paramList.append(quickParameterFill(ret, "Rotate Y", "rotate_y", "double", "0", "0", "360")); - paramList.append(quickParameterFill(ret, "Rotate X", "rotate_x", "double", "0", "0", "360")); - paramList.append(quickParameterFill(ret, "Rotate Z", "rotate_z", "double", "0", "0", "360")); - paramList.append(quickParameterFill(ret, "Fix Rotate Y", "fix_rotate_y", "double", "0", "0", "360")); - paramList.append(quickParameterFill(ret, "Fix Rotate X", "fix_rotate_x", "double", "0", "0", "360")); - paramList.append(quickParameterFill(ret, "Fix Rotate Z", "fix_rotate_z", "double", "0", "0", "360")); - paramList.append(quickParameterFill(ret, "Shear Y", "shear_y", "double", "0", "0", "360")); - paramList.append(quickParameterFill(ret, "Shear X", "shear_x", "double", "0", "0", "360")); - paramList.append(quickParameterFill(ret, "Shear Z", "shear_z", "double", "0", "0", "360")); - paramList.append(quickParameterFill(ret, "Fix Shear Y", "fix_shear_y", "double", "0", "0", "360")); - paramList.append(quickParameterFill(ret, "Fix Shear X", "fix_shear_x", "double", "0", "0", "360")); - paramList.append(quickParameterFill(ret, "Fix Shear Z", "fix_shear_z", "double", "0", "0", "360")); - paramList.append(quickParameterFill(ret, "Mirror", "mirror_off", "bool", "0", "0", "1")); - paramList.append(quickParameterFill(ret, "Repeat", "repeat_off", "bool", "0", "0", "1")); - paramList.append(quickParameterFill(ret, "Geometry", "geometry", "geometry", "0;0;100;100;100", "0;0;100;100;100", "0;0;100;100;100")); - tname.appendChild(ret.createTextNode("Composite")); + tname.appendChild(ret.createTextNode(i18n("Affine"))); + paramList.append(quickParameterFill(ret, i18n("Rotate Y"), "rotate_y", "double", "0", "0", "360")); + paramList.append(quickParameterFill(ret, i18n("Rotate X"), "rotate_x", "double", "0", "0", "360")); + paramList.append(quickParameterFill(ret, i18n("Rotate Z"), "rotate_z", "double", "0", "0", "360")); + paramList.append(quickParameterFill(ret, i18n("Fix Rotate Y"), "fix_rotate_y", "double", "0", "0", "360")); + paramList.append(quickParameterFill(ret, i18n("Fix Rotate X"), "fix_rotate_x", "double", "0", "0", "360")); + paramList.append(quickParameterFill(ret, i18n("Fix Rotate Z"), "fix_rotate_z", "double", "0", "0", "360")); + paramList.append(quickParameterFill(ret, i18n("Shear Y"), "shear_y", "double", "0", "0", "360")); + paramList.append(quickParameterFill(ret, i18n("Shear X"), "shear_x", "double", "0", "0", "360")); + paramList.append(quickParameterFill(ret, i18n("Shear Z"), "shear_z", "double", "0", "0", "360")); + paramList.append(quickParameterFill(ret, i18n("Fix Shear Y"), "fix_shear_y", "double", "0", "0", "360")); + paramList.append(quickParameterFill(ret, i18n("Fix Shear X"), "fix_shear_x", "double", "0", "0", "360")); + paramList.append(quickParameterFill(ret, i18n("Fix Shear Z"), "fix_shear_z", "double", "0", "0", "360")); + paramList.append(quickParameterFill(ret, i18n("Geometry"), "geometry", "geometry", "0,0,100%,100%,100%", "0,0,100%,100%,100%", "0,0,100%,100%,100%", "", "", "", "", "", "false")); + } else if (name == "mix") { + tname.appendChild(ret.createTextNode(i18n("Mix"))); } else if (name == "region") { - tname.appendChild(ret.createTextNode("Region")); + ktrans.setAttribute("id", name); + tname.appendChild(ret.createTextNode(i18n("Region"))); + desc.appendChild(ret.createTextNode(i18n("Use alpha channel of another clip to create a transition."))); + paramList.append(quickParameterFill(ret, i18n("Transparency clip"), "resource", "url", "", "", "", "", "", "")); + paramList.append(quickParameterFill(ret, i18n("Geometry"), "composite.geometry", "geometry", "0%,0%:100%x100%:100", "-500;-500;-500;-500;0", "500;500;500;500;100")); + paramList.append(quickParameterFill(ret, i18n("Alpha Channel Operation"), "composite.operator", "list", "over", "", "", "over,and,or,xor", "over,and,or,xor")); + paramList.append(quickParameterFill(ret, i18n("Align"), "composite.aligned", "bool", "1", "0", "1")); + paramList.append(quickParameterFill(ret, i18n("Fill"), "composite.fill", "bool", "1", "0", "1")); + paramList.append(quickParameterFill(ret, i18n("Distort"), "composite.distort", "bool", "0", "0", "1")); + paramList.append(quickParameterFill(ret, i18n("Wipe File"), "composite.luma", "list", "", "", "", imagefiles.join(","), imagenamelist.join(","))); + paramList.append(quickParameterFill(ret, i18n("Wipe Softness"), "composite.softness", "double", "0", "0", "100", "", "", "100")); + paramList.append(quickParameterFill(ret, i18n("Wipe Invert"), "composite.luma_invert", "bool", "0", "0", "1")); + paramList.append(quickParameterFill(ret, i18n("Force Progressive Rendering"), "composite.progressive", "bool", "1", "0", "1")); + paramList.append(quickParameterFill(ret, i18n("Force Deinterlace Overlay"), "composite.deinterlace", "bool", "0", "0", "1")); } - - - } - - ktrans.appendChild(tname); - ktrans.appendChild(desc); - - foreach(const QDomElement &e, paramList) { + foreach(const QDomElement & e, paramList) ktrans.appendChild(e); } - + // Add the transition to the global list. transitions->append(ret.documentElement()); - //kDebug() << "//// //// TRANSITON XML"; - kDebug() << ret.toString(); - /* - - - */ + //kDebug() << ret.toString(); } - QString wipetrans = "WipeSlide image from one side to anotherDirection Align"; + // Add some virtual transitions. + QString slidetrans = "" + i18n("Slide") + "" + i18n("Slide image from one side to another.") + "" + i18n("Direction") + " " + i18n("Align") + "" + i18n("Force Progressive Rendering") + "" + i18n("Force Deinterlace Overlay") + "" + i18nc("@property: means that the image is inverted", "Invert") + ""; QDomDocument ret; - ret.setContent(wipetrans); + ret.setContent(slidetrans); transitions->append(ret.documentElement()); - QString alphatrans = "Alpha transparencyMake alpha channel transparentDirectionRescaleAlign"; - QDomDocument ret2; - ret2.setContent(alphatrans); - transitions->append(ret2.documentElement()); + QString dissolve = "" + i18n("Dissolve") + "" + i18n("Fade out one video while fading in the other video.") + "" + i18n("Reverse") + ""; + ret.setContent(dissolve); + transitions->append(ret.documentElement()); + + /*QString alphatrans = "" + i18n("Alpha Transparency") + "" + i18n("Make alpha channel transparent.") + "" + i18n("Direction") + "" + i18n("Rescale") + "" + i18n("Align") + ""; + ret.setContent(alphatrans); + transitions->append(ret.documentElement());*/ } -QDomElement initEffects::quickParameterFill(QDomDocument & doc, QString name, QString tag, QString type, QString def, QString min, QString max, QString list, QString listdisplaynames, QString factor, QString namedesc, QString format) { +QDomElement initEffects::quickParameterFill(QDomDocument & doc, QString name, QString tag, QString type, QString def, QString min, QString max, QString list, QString listdisplaynames, QString factor, QString namedesc, QString format, QString opacity) +{ QDomElement parameter = doc.createElement("parameter"); parameter.setAttribute("tag", tag); parameter.setAttribute("default", def); @@ -733,6 +829,8 @@ QDomElement initEffects::quickParameterFill(QDomDocument & doc, QString name, QS parameter.setAttribute("namedesc", namedesc); if (!format.isEmpty()) parameter.setAttribute("format", format); + if (!opacity.isEmpty()) + parameter.setAttribute("opacity", opacity); QDomElement pname = doc.createElement("name"); pname.appendChild(doc.createTextNode(name)); parameter.appendChild(pname);