X-Git-Url: https://git.sesse.net/?a=blobdiff_plain;f=src%2Finiteffects.cpp;h=a8c8046609441cc56b0e590ddbc4b89860e1f400;hb=5505e4a10c0b0b372ed9f3c0b548f4d763fa5318;hp=6a759f20beb77dca3feac260a7400c3d7ca6998e;hpb=826f7539d163276fb0e4bc200aedf81547ce1c52;p=kdenlive diff --git a/src/initeffects.cpp b/src/initeffects.cpp index 6a759f20..a8c80466 100644 --- a/src/initeffects.cpp +++ b/src/initeffects.cpp @@ -20,7 +20,6 @@ #include "initeffects.h" #include "kdenlivesettings.h" #include "effectslist.h" -#include "effectstackedit.h" #include "mainwindow.h" #include @@ -28,28 +27,33 @@ #include #include -#include +#include #include #include -initEffectsThumbnailer::initEffectsThumbnailer() { +#include "locale.h" +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)) { + if (!MainWindow::m_lumacache.contains(entry)) { QImage pix(entry); //if (!pix.isNull()) - EffectStackEdit::iconCache[entry] = pix.scaled(30, 30); + MainWindow::m_lumacache.insert(entry, pix.scaled(30, 30, Qt::KeepAspectRatio, Qt::SmoothTransformation)); kDebug() << "stored"; } } @@ -58,414 +62,390 @@ 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++) { + for (int i = 0; i < params.count(); ++i) { QDomElement e = params.item(i).toElement(); if (e.attribute("tag") == "resource") { e.setAttribute("paramlistdisplay", imagenamelist.join(",")); - e.setAttribute("paramlist", imagefiles.join(",")); + e.setAttribute("paramlist", imagefiles.join(";")); break; } } + 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(const 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() { +void initEffects::parseEffectFiles(const QString &locale) +{ 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 " ; - return NULL; + kDebug() << "Repository didn't finish initialisation" ; + return; } + + // Warning: Mlt::Factory::init() resets the locale to the default system value, make sure we keep correct locale + if (!locale.isEmpty()) setlocale(LC_NUMERIC, locale.toUtf8().constData()); + + // Retrieve the list of MLT's available effects. Mlt::Properties *filters = repository->filters(); QStringList filtersList; + int max = filters->count(); + for (int i = 0; i < max; ++i) + filtersList << filters->get_name(i); + delete filters; - // Check for blacklisted effects - QString blacklist = KStandardDirs::locate("appdata", "blacklisted_effects.txt"); + // Retrieve the list of available producers. + Mlt::Properties *producers = repository->producers(); + QStringList producersList; + max = producers->count(); + for (int i = 0; i < max; ++i) + producersList << producers->get_name(i); + KdenliveSettings::setProducerslist(producersList); + delete producers; - QFile file(blacklist); - QStringList blackListed; + // Retrieve the list of available transitions. + Mlt::Properties *transitions = repository->transitions(); + QStringList transitionsItemList; + max = transitions->count(); + for (int i = 0; i < max; ++i) + transitionsItemList << transitions->get_name(i); + delete transitions; + // 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('#')) blackListed.append(black); + if (!black.isEmpty() && !black.startsWith('#') && + transitionsItemList.contains(black)) + transitionsItemList.removeAll(black); } + file.close(); } - file.close(); - - // Check for blacklisted transitions - blacklist = KStandardDirs::locate("appdata", "blacklisted_transitions.txt"); - QFile file2(blacklist); - QStringList blackListedtransitions; + // Fill transitions list. + fillTransitionsList(repository, &MainWindow::transitions, transitionsItemList); + // Remove blacklisted effects from the filters list. + QStringList mltFiltersList = filtersList; + QFile file2(KStandardDirs::locate("appdata", "blacklisted_effects.txt")); if (file2.open(QIODevice::ReadOnly)) { QTextStream in(&file2); while (!in.atEnd()) { QString black = in.readLine().simplified(); - if (!black.isEmpty() && !black.startsWith('#')) blackListedtransitions.append(black); + if (!black.isEmpty() && !black.startsWith('#') && + mltFiltersList.contains(black)) { + mltFiltersList.removeAll(black); + } + } + file2.close(); } - file2.close(); - - for (int i = 0 ; i < filters->count() ; i++) { - filtersList << filters->get_name(i); - } - - // Build effects. check producers first. - Mlt::Properties *producers = repository->producers(); - QStringList producersList; - for (int i = 0 ; i < producers->count() ; i++) { - producersList << producers->get_name(i); + /* + * Cleanup the global lists. We use QMap because of its automatic sorting + * (by key) and key uniqueness (using insert() instead of insertMulti()). + * This introduces some more cycles (while removing them from other parts of + * the code and centralising them), but due to the way this methods, QMap + * and EffectsList are implemented, there's no easy way to make it + * differently without reinplementing something (which should really be + * done). + */ + QDomElement effectInfo; + QMap effectsMap; + QMap videoEffectsMap; + QMap audioEffectsMap; + + // Create transitions + max = MainWindow::transitions.count(); + for (int i = 0; i < max; ++i) { + effectInfo = MainWindow::transitions.at(i); + effectsMap.insert(effectInfo.firstChildElement("name").text().toLower().toUtf8().data(), effectInfo); } - delete filters; - delete producers; - - Mlt::Properties *transitions = repository->transitions(); - QStringList transitionsItemList; - 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); + MainWindow::transitions.clearList(); + foreach(const QDomElement & effect, effectsMap) + MainWindow::transitions.append(effect); + effectsMap.clear(); + // Create effects from MLT + foreach(const QString & filtername, mltFiltersList) { + QDomDocument doc = createDescriptionFromMlt(repository, "filters", filtername); + //WARNING: TEMPORARY FIX for empty MLT effects descriptions - disable effects without parameters - jbm 09-06-2011 + if (!doc.isNull() && doc.elementsByTagName("parameter").count() > 0) { + if (doc.documentElement().attribute("type") == "audio") { + if (doc.elementsByTagName("description").count() > 0) { + QString desc = doc.documentElement().firstChildElement("description").text(); + //WARNING: TEMPORARY FIX for unusable MLT SOX parameters description + if (desc.startsWith("Process audio using a SoX")) { + // Remove MLT's SOX generated effects since the parameters properties are unusable for us + } + else { + audioEffectsMap.insert(doc.documentElement().firstChildElement("name").text().toLower().toUtf8().data(), doc.documentElement()); + } + } + } + else + videoEffectsMap.insert(doc.documentElement().firstChildElement("name").text().toLower().toUtf8().data(), doc.documentElement()); + } } - fillTransitionsList(repository, &MainWindow::transitions, transitionsItemList); - - 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"); - kDebug() << "// INIT EFFECT SEARCH" << endl; - - QStringList direc = KGlobal::dirs()->findDirs("data", "kdenlive/effects"); - - QDir directory; - for (more = direc.begin() ; more != direc.end() ; ++more) { - directory = QDir(*more); + // Set the directories to look into for effects. + QStringList direc = KGlobal::dirs()->findDirs("appdata", "effects"); + // 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; 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; + int unknownGroupCount = 0; + 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"); - if (effects.count() != 1) { - kDebug() << "More than one effect in file " << itemName << ", NOT SUPPORTED YET"; - } else { - QDomElement e = effects.item(0).toElement(); - MainWindow::customEffects.append(e); + QDomElement base = doc.documentElement(); + if (base.tagName() == "effectgroup") { + QString groupName = base.attribute("name"); + if (groupName.isEmpty()) { + groupName = i18n("Group %1", unknownGroupCount); + base.setAttribute("name", groupName); + unknownGroupCount++; + } + effectsMap.insert(groupName.toLower().toUtf8().data(), base); + } else if (base.tagName() == "effect") { + effectsMap.insert(base.firstChildElement("name").text().toLower().toUtf8().data(), base); } + else kDebug() << "Unsupported effect file: " << itemName; } + 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, const QString &name, QStringList filtersList, QStringList producersList, Mlt::Repository *repository) +{ QDomDocument doc; QFile file(name); doc.setContent(&file, false); file.close(); - QDomElement documentElement = doc.documentElement(); - QDomNodeList effects = doc.elementsByTagName("effect"); + QDomElement documentElement; + QDomNodeList effects; + QDomElement base = doc.documentElement(); + 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()); - } - - int i = 0; - while (!effects.item(i).isNull()) { + bool needsLocaleConversion = false; + for (int i = 0; !effects.item(i).isNull(); ++i) { + QLocale locale; documentElement = effects.item(i).toElement(); QString tag = documentElement.attribute("tag", QString()); - bool ladspaOk = true; - if (tag == "ladspa") { - QString library = documentElement.attribute("library", QString()); - if (KStandardDirs::locate("ladspa_plugin", library).isEmpty()) ladspaOk = false; + if (documentElement.hasAttribute("LC_NUMERIC")) { + // set a locale for that file + locale = QLocale(documentElement.attribute("LC_NUMERIC")); + if (locale.decimalPoint() != QLocale().decimalPoint()) { + needsLocaleConversion = true; + } } - - // Parse effect file - if ((filtersList.contains(tag) || producersList.contains(tag)) && ladspaOk) { - QString type = documentElement.attribute("type", QString()); - if (type == "audio") audioEffectList->append(documentElement); - else if (type == "custom") customEffectList->append(documentElement); - else videoEffectList->append(documentElement); + locale.setNumberOptions(QLocale::OmitGroupSeparator); + + if (needsLocaleConversion) { + // we need to convert all numbers to the system's locale (for example 0.5 -> 0,5) + QChar separator = QLocale().decimalPoint(); + QChar oldSeparator = locale.decimalPoint(); + QDomNodeList params = documentElement.elementsByTagName("parameter"); + for (int j = 0; j < params.count(); j++) { + QDomNamedNodeMap attrs = params.at(j).attributes(); + for (int k = 0; k < attrs.count(); k++) { + QString nodeName = attrs.item(k).nodeName(); + if (nodeName != "type" && nodeName != "name") { + QString val = attrs.item(k).nodeValue(); + if (val.contains(oldSeparator)) { + QString newVal = val.replace(oldSeparator, separator); + attrs.item(k).setNodeValue(newVal); + } + } + } + } } - /* - QDomNode n = documentElement.firstChild(); - QString id, effectName, effectTag, paramType; - int paramCount = 0; - EFFECTTYPE type; - - // Create Effect - EffectParamDescFactory effectDescParamFactory; - EffectDesc *effect = NULL; - - // parse effect file - QDomNode namenode = documentElement.elementsByTagName("name").item(0); - if (!namenode.isNull()) effectName = i18n(namenode.toElement().text()); - if (!groupName.isEmpty()) effectName.prepend("_" + groupName + "_"); - - QDomNode propsnode = documentElement.elementsByTagName("properties").item(0); - if (!propsnode.isNull()) { - QDomElement propselement = propsnode.toElement(); - 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; - } - - QString effectDescription; - QDomNode descnode = documentElement.elementsByTagName("description").item(0); - if (!descnode.isNull()) effectDescription = descnode.toElement().text() + "
"; - - QString effectAuthor; - QDomNode authnode = documentElement.elementsByTagName("author").item(0); - if (!authnode.isNull()) effectAuthor = authnode.toElement().text() + "
"; - - if (effectName.isEmpty() || id.isEmpty() || effectTag.isEmpty()) return; - - effect = new EffectDesc(effectName, id, effectTag, effectDescription, effectAuthor, type); - - QDomNodeList paramList = documentElement.elementsByTagName("parameter"); - if (paramList.count() == 0) { - QDomElement fixed = doc.createElement("parameter"); - fixed.setAttribute("type", "fixed"); - effect->addParameter(effectDescParamFactory.createParameter(fixed)); - } - else for (int i = 0; i < paramList.count(); i++) { - QDomElement e = paramList.item(i).toElement(); - if (!e.isNull()) { - paramCount++; - QDomNamedNodeMap attrs = e.attributes(); - int i = 0; - QString value; - while (!attrs.item(i).isNull()) { - QDomNode n = attrs.item(i); - value = n.nodeValue(); - if (value.find("MAX_WIDTH") != -1) - value.replace("MAX_WIDTH", QString::number(KdenliveSettings::defaultwidth())); - if (value.find("MID_WIDTH") != -1) - value.replace("MID_WIDTH", QString::number(KdenliveSettings::defaultwidth() / 2)); - if (value.find("MAX_HEIGHT") != -1) - value.replace("MAX_HEIGHT", QString::number(KdenliveSettings::defaultheight())); - if (value.find("MID_HEIGHT") != -1) - value.replace("MID_HEIGHT", QString::number(KdenliveSettings::defaultheight() / 2)); - n.setNodeValue(value); - i++; - } - effect->addParameter(effectDescParamFactory.createParameter(e)); - } - } - effectList->append(effect); - }*/ - i++; - } -} + double version = -1; + Mlt::Properties *metadata = repository->metadata(filter_type, tag.toUtf8().data()); + if (metadata && metadata->is_valid()) { + version = metadata->get_double("version"); + } + if (metadata) delete metadata; + if (documentElement.hasAttribute("version")) { + // a specific version of the filter is required + if (locale.toDouble(documentElement.attribute("version")) > version) { + return; + } + } + if (version > -1) { + // Add version info to XML + QDomNode versionNode = doc.createElement("version"); + versionNode.appendChild(doc.createTextNode(QLocale().toString(version))); + documentElement.appendChild(versionNode); + } -//static -const char* initEffects::ladspaEffectString(int ladspaId, QStringList params) { - if (ladspaId == 1433) //Pitch - return ladspaPitchEffectString(params); - else if (ladspaId == 1216) //Room Reverb - return ladspaRoomReverbEffectString(params); - else if (ladspaId == 1423) //Reverb - return ladspaReverbEffectString(params); - else if (ladspaId == 1901) //Reverb - return ladspaEqualizerEffectString(params); - else { - kDebug() << "++++++++++ ASKING FOR UNKNOWN LADSPA EFFECT: " << ladspaId << endl; - return ""; + // Parse effect information. + if (base.tagName() != "effectgroup" && (filtersList.contains(tag) || producersList.contains(tag))) { + QString type = documentElement.attribute("type", QString()); + if (type == "audio") + audioEffectList->append(documentElement); + else if (type == "custom") + customEffectList->append(documentElement); + else + videoEffectList->append(documentElement); + } } -} - -//static -void initEffects::ladspaEffectFile(const QString & fname, int ladspaId, QStringList params) { - const char *filterString; - switch (ladspaId) { - case 1433: //Pitch - filterString = ladspaPitchEffectString(params); - break; - case 1905: //Vinyl - filterString = ladspaVinylEffectString(params); - break; - case 1216 : //Room Reverb - filterString = ladspaRoomReverbEffectString(params); - break; - case 1423: //Reverb - filterString = ladspaReverbEffectString(params); - break; - case 1195: //Declipper - filterString = ladspaDeclipEffectString(params); - break; - case 1901: //Reverb - filterString = ladspaEqualizerEffectString(params); - break; - case 1913: // Limiter - filterString = ladspaLimiterEffectString(params); - break; - case 1193: // Pitch Shifter - filterString = ladspaPitchShifterEffectString(params); - break; - case 1417: // Rate Scaler - filterString = ladspaRateScalerEffectString(params); - break; - case 1217: // Phaser - filterString = ladspaPhaserEffectString(params); - break; - default: - kDebug() << "++++++++++ ASKING FOR UNKNOWN LADSPA EFFECT: " << ladspaId << endl; - return; - break; + if (base.tagName() == "effectgroup") { + QString type = base.attribute("type", QString()); + if (type == "audio") + audioEffectList->append(base); + else if (type == "custom") + customEffectList->append(base); + else + videoEffectList->append(base); } - - QFile f(fname); - if (f.open(QIODevice::WriteOnly)) { - QTextStream stream(&f); - stream << filterString; - f.close(); - } else kDebug() << "++++++++++ ERROR CANNOT WRITE TO: " << KdenliveSettings::currenttmpfolder() + fname << endl; - delete filterString; -} - -const QString jackString = "248000"; - - -const char* initEffects::ladspaDeclipEffectString(QStringList) { - return qstrdup(QString(jackString + "1195truefalsetrue1.0000001.000000true").toUtf8()); } -/* -const char* initEffects::ladspaVocoderEffectString(QStringList params) +QDomDocument initEffects::createDescriptionFromMlt(Mlt::Repository* repository, const QString& /*type*/, const QString& filtername) { - 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])); -}*/ - -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()); -} - -const char* initEffects::ladspaPitchEffectString(QStringList params) { - return qstrdup(QString(jackString + "1433truefalsetrue1.01.0truetrue%1%1true4.0000004.000000").arg(params[0]).toUtf8()); -} - -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()); -} - -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()); -} - -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()); -} - -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()); -} - -const char* initEffects::ladspaPitchShifterEffectString(QStringList params) { - return qstrdup(QString(jackString + "1193truefalsetrue1.0000001.000000truetrue%1%1").arg(params[0]).toUtf8()); -} - -const char* initEffects::ladspaRateScalerEffectString(QStringList params) { - return qstrdup(QString(jackString + "1417truefalsetrue1.0000001.000000truetrue%1%1").arg(params[0]).toUtf8()); -} - -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 ret; Mlt::Properties *metadata = repository->metadata(filter_type, filtername.toAscii().data()); @@ -473,8 +453,10 @@ QDomDocument initEffects::createDescriptionFromMlt(Mlt::Repository* repository, if (metadata && metadata->is_valid()) { if (metadata->get("title") && metadata->get("identifier")) { QDomElement eff = ret.createElement("effect"); - eff.setAttribute("tag", metadata->get("identifier")); - eff.setAttribute("id", metadata->get("identifier")); + QString id = metadata->get("identifier"); + eff.setAttribute("tag", id); + eff.setAttribute("id", id); + //kDebug()<<"Effect: "<get("title"))); @@ -485,37 +467,54 @@ QDomDocument initEffects::createDescriptionFromMlt(Mlt::Repository* repository, QDomElement author = ret.createElement("author"); author.appendChild(ret.createTextNode(metadata->get("creator"))); + QDomElement version = ret.createElement("version"); + version.appendChild(ret.createTextNode(metadata->get("version"))); + eff.appendChild(name); eff.appendChild(author); eff.appendChild(desc); + eff.appendChild(version); + + Mlt::Properties tags((mlt_properties) metadata->get_data("tags")); + if (QString(tags.get(0)) == "Audio") eff.setAttribute("type", "audio"); + /*for (int i = 0; i < tags.count(); ++i) + kDebug()<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))); - params.setAttribute("name", paramdesc.get("identifier")); + if (params.attribute("name") == "argument") { + // This parameter has to be given as attribute when using command line, do not show it in Kdenlive + continue; + } 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"); - if (QString(paramdesc.get("type")) == "float") { + + QString paramType = paramdesc.get("type"); + + if (paramType == "integer") { + if (params.attribute("min") == "0" && params.attribute("max") == "1") + params.setAttribute("type", "bool"); + else params.setAttribute("type", "constant"); + + } + else if (paramType == "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); + // param type is float, set default decimals to 3 + params.setAttribute("decimals", "3"); } - if (QString(paramdesc.get("type")) == "boolean") + else if (paramType == "boolean") params.setAttribute("type", "bool"); - if (!QString(paramdesc.get("format")).isEmpty() && QString(paramdesc.get("type")) != "geometry") { + else if (paramType == "geometry") { params.setAttribute("type", "geometry"); - params.setAttribute("format", paramdesc.get("format")); } - if (!QString(paramdesc.get("format")).isEmpty() && QString(paramdesc.get("type")) == "geometry") { - params.setAttribute("type", "geometry"); - //params.setAttribute("format", paramdesc.get("format")); + else { + params.setAttribute("type", paramType); + if (!QString(paramdesc.get("format")).isEmpty()) params.setAttribute("format", paramdesc.get("format")); } if (paramdesc.get("default")) params.setAttribute("default", paramdesc.get("default")); if (paramdesc.get("value")) { @@ -524,56 +523,99 @@ QDomDocument initEffects::createDescriptionFromMlt(Mlt::Repository* repository, params.setAttribute("value", paramdesc.get("default")); } - QDomElement pname = ret.createElement("name"); pname.appendChild(ret.createTextNode(paramdesc.get("title"))); params.appendChild(pname); + + if (paramdesc.get("description")) { + QDomElement desc = ret.createElement("comment"); + desc.appendChild(ret.createTextNode(paramdesc.get("description"))); + params.appendChild(desc); + } eff.appendChild(params); } ret.appendChild(eff); } } - /* QString outstr; + delete metadata; + metadata = 0; + /*QString outstr; QTextStream str(&outstr); ret.save(str, 2); kDebug() << outstr;*/ 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()); + } + + //WARNING: this is a hack to get around temporary invalid metadata in MLT, 2nd of june 2011 JBM + QStringList customTransitions; + customTransitions << "composite" << "luma" << "affine" << "mix" << "region"; + + 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 = NULL; + if (!customTransitions.contains(name)) 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"); @@ -583,138 +625,124 @@ 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(); } else { - if (name == "luma") { + /* + * Check for Kdenlive installed luma files, add empty string at + * start for no luma file. + */ - 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 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")); - ktrans.setAttribute("id", "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", i18n("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"))); + ret.documentElement().setAttribute("showrotation", "1"); + /*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, "keyed", "keyed", "fixed", "1", "1", "1")); + paramList.append(quickParameterFill(ret, i18n("Geometry"), "geometry", "geometry", "0/0:100%x100%:100%", "0/0:100%x100%:100%", "0/0:100%x100%:100%", "", "", "", "", "", "true")); + paramList.append(quickParameterFill(ret, i18n("Distort"), "distort", "bool", "0", "0", "1")); + paramList.append(quickParameterFill(ret, i18n("Rotate X"), "rotate_x", "addedgeometry", "0", "-1800", "1800", QString(), QString(), "10")); + paramList.append(quickParameterFill(ret, i18n("Rotate Y"), "rotate_y", "addedgeometry", "0", "-1800", "1800", QString(), QString(), "10")); + paramList.append(quickParameterFill(ret, i18n("Rotate Z"), "rotate_z", "addedgeometry", "0", "-1800", "1800", QString(), QString(), "10")); + /*paramList.append(quickParameterFill(ret, i18n("Rotate Y"), "rotate_y", "simplekeyframe", "0", "-1800", "1800", QString(), QString(), "10")); + paramList.append(quickParameterFill(ret, i18n("Rotate Z"), "rotate_z", "simplekeyframe", "0", "-1800", "1800", QString(), QString(), "10"));*/ + + paramList.append(quickParameterFill(ret, i18n("Fix Shear Y"), "shear_y", "double", "0", "0", "360")); + paramList.append(quickParameterFill(ret, i18n("Fix Shear X"), "shear_x", "double", "0", "0", "360")); + paramList.append(quickParameterFill(ret, i18n("Fix Shear Z"), "shear_z", "double", "0", "0", "360")); + } 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", i18n("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); } - - + delete metadata; + metadata = 0; + // 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 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 = "Alpha transparencyMake alpha channel transparentDirectionRescaleAlign"; - QDomDocument ret2; - ret2.setContent(alphatrans); - transitions->append(ret2.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, const QString &name, const QString &tag, const QString &type, const QString &def, const QString &min, const QString &max, const QString &list, const QString &listdisplaynames, const QString &factor, const QString &namedesc, const QString &format, const QString &opacity) +{ QDomElement parameter = doc.createElement("parameter"); parameter.setAttribute("tag", tag); parameter.setAttribute("default", def); @@ -732,9 +760,13 @@ 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); return parameter; } + +#include "initeffects.moc"