]> git.sesse.net Git - kdenlive/blob - src/initeffects.cpp
Fix saved effect groups not showing in effects list
[kdenlive] / src / initeffects.cpp
1 /***************************************************************************
2                           initeffects.cpp  -  description
3                              -------------------
4     begin                :  Jul 2006
5     copyright            : (C) 2006 by Jean-Baptiste Mardelle
6     email                : jb@ader.ch
7     copyright            : (C) 2008 Marco Gittler
8     email                : g.marco@freenet.de
9  ***************************************************************************/
10
11 /***************************************************************************
12  *                                                                         *
13  *   This program is free software; you can redistribute it and/or modify  *
14  *   it under the terms of the GNU General Public License as published by  *
15  *   the Free Software Foundation; either version 2 of the License, or     *
16  *   (at your option) any later version.                                   *
17  *                                                                         *
18  ***************************************************************************/
19
20 #include "initeffects.h"
21 #include "kdenlivesettings.h"
22 #include "effectslist.h"
23 #include "effectstackedit.h"
24 #include "mainwindow.h"
25
26 #include <KDebug>
27 #include <kglobal.h>
28 #include <KStandardDirs>
29
30 #include <QFile>
31 #include <qregexp.h>
32 #include <QDir>
33 #include <QIcon>
34
35 initEffectsThumbnailer::initEffectsThumbnailer() :
36     QThread()
37 {
38 }
39
40 void initEffectsThumbnailer::prepareThumbnailsCall(const QStringList& list)
41 {
42     m_list = list;
43     start();
44     kDebug() << "done";
45 }
46
47 void initEffectsThumbnailer::run()
48 {
49     foreach(const QString & entry, m_list) {
50         kDebug() << entry;
51         if (!entry.isEmpty() && (entry.endsWith(".png") || entry.endsWith(".pgm"))) {
52             if (!EffectStackEdit::iconCache.contains(entry)) {
53                 QImage pix(entry);
54                 //if (!pix.isNull())
55                 EffectStackEdit::iconCache[entry] = pix.scaled(30, 30);
56                 kDebug() << "stored";
57             }
58         }
59     }
60 }
61
62 initEffectsThumbnailer initEffects::thumbnailer;
63
64 // static
65 void initEffects::refreshLumas()
66 {
67     // Check for Kdenlive installed luma files, add empty string at start for no luma
68     QStringList imagenamelist = QStringList() << i18n("None");
69     QStringList imagefiles = QStringList() << QString();
70     QStringList filters;
71     filters << "*.pgm" << "*.png";
72
73     QStringList customLumas = KGlobal::dirs()->findDirs("appdata", "lumas");
74     foreach(const QString & folder, customLumas) {
75         QStringList filesnames = QDir(folder).entryList(filters, QDir::Files);
76         foreach(const QString & fname, filesnames) {
77             imagenamelist.append(fname);
78             imagefiles.append(KUrl(folder).path(KUrl::AddTrailingSlash) + fname);
79         }
80     }
81
82     // Check for MLT lumas
83     KUrl folder(mlt_environment("MLT_DATA"));
84     folder.addPath("lumas");
85     folder.addPath(mlt_environment("MLT_NORMALISATION"));
86     QDir lumafolder(folder.path());
87     QStringList filesnames = lumafolder.entryList(filters, QDir::Files);
88     foreach(const QString & fname, filesnames) {
89         imagenamelist.append(fname);
90         KUrl path(folder);
91         path.addPath(fname);
92         imagefiles.append(path.toLocalFile());
93     }
94     QDomElement lumaTransition = MainWindow::transitions.getEffectByTag("luma", "luma");
95     QDomNodeList params = lumaTransition.elementsByTagName("parameter");
96     for (int i = 0; i < params.count(); i++) {
97         QDomElement e = params.item(i).toElement();
98         if (e.attribute("tag") == "resource") {
99             e.setAttribute("paramlistdisplay", imagenamelist.join(","));
100             e.setAttribute("paramlist", imagefiles.join(";"));
101             break;
102         }
103     }
104
105     QDomElement compositeTransition = MainWindow::transitions.getEffectByTag("composite", "composite");
106     params = compositeTransition.elementsByTagName("parameter");
107     for (int i = 0; i < params.count(); i++) {
108         QDomElement e = params.item(i).toElement();
109         if (e.attribute("tag") == "luma") {
110             e.setAttribute("paramlistdisplay", imagenamelist.join(","));
111             e.setAttribute("paramlist", imagefiles.join(";"));
112             break;
113         }
114     }
115 }
116
117 // static
118 QDomDocument initEffects::getUsedCustomEffects(QMap <QString, QString> effectids)
119 {
120     QMapIterator<QString, QString> i(effectids);
121     int ix;
122     QDomDocument doc;
123     QDomElement list = doc.createElement("customeffects");
124     doc.appendChild(list);
125     while (i.hasNext()) {
126         i.next();
127         ix = MainWindow::customEffects.hasEffect(i.value(), i.key());
128         if (ix > -1) {
129             QDomElement e = MainWindow::customEffects.at(ix);
130             list.appendChild(doc.importNode(e, true));
131         }
132     }
133     return doc;
134 }
135
136 //static
137 void initEffects::parseEffectFiles()
138 {
139     QStringList::Iterator more;
140     QStringList::Iterator it;
141     QStringList fileList;
142     QString itemName;
143
144     Mlt::Repository *repository = Mlt::Factory::init();
145     if (!repository) {
146         kDebug() << "Repository didn't finish initialisation" ;
147         return;
148     }
149
150     // Retrieve the list of MLT's available effects.
151     Mlt::Properties *filters = repository->filters();
152     QStringList filtersList;
153     int max = filters->count();
154     for (int i = 0; i < max; ++i)
155         filtersList << filters->get_name(i);
156     delete filters;
157
158     // Retrieve the list of available producers.
159     Mlt::Properties *producers = repository->producers();
160     QStringList producersList;
161     max = producers->count();
162     for (int i = 0; i < max; ++i)
163         producersList << producers->get_name(i);
164     KdenliveSettings::setProducerslist(producersList);
165     delete producers;
166
167     // Retrieve the list of available transitions.
168     Mlt::Properties *transitions = repository->transitions();
169     QStringList transitionsItemList;
170     max = transitions->count();
171     for (int i = 0; i < max; ++i)
172         transitionsItemList << transitions->get_name(i);
173     delete transitions;
174
175     // Remove blacklisted transitions from the list.
176     QFile file(KStandardDirs::locate("appdata", "blacklisted_transitions.txt"));
177     if (file.open(QIODevice::ReadOnly)) {
178         QTextStream in(&file);
179         while (!in.atEnd()) {
180             QString black = in.readLine().simplified();
181             if (!black.isEmpty() && !black.startsWith('#') &&
182                     transitionsItemList.contains(black))
183                 transitionsItemList.removeAll(black);
184         }
185         file.close();
186     }
187
188     // Fill transitions list.
189     fillTransitionsList(repository, &MainWindow::transitions, transitionsItemList);
190
191     // Remove blacklisted effects from the filters list.
192     QStringList mltFiltersList = filtersList;
193     QFile file2(KStandardDirs::locate("appdata", "blacklisted_effects.txt"));
194     if (file2.open(QIODevice::ReadOnly)) {
195         QTextStream in(&file2);
196         while (!in.atEnd()) {
197             QString black = in.readLine().simplified();
198             if (!black.isEmpty() && !black.startsWith('#') &&
199                     mltFiltersList.contains(black))
200                 mltFiltersList.removeAll(black);
201         }
202         file2.close();
203     }
204
205     /*
206      * Cleanup the global lists. We use QMap because of its automatic sorting
207      * (by key) and key uniqueness (using insert() instead of insertMulti()).
208      * This introduces some more cycles (while removing them from other parts of
209      * the code and centralising them), but due to the way this methods, QMap
210      * and EffectsList are implemented, there's no easy way to make it
211      * differently without reinplementing something (which should really be
212      * done).
213      */
214     QDomElement effectInfo;
215     QMap<QString, QDomElement> effectsMap;
216     QMap<QString, QDomElement> videoEffectsMap;
217     QMap<QString, QDomElement> audioEffectsMap;
218
219     // Create transitions
220     max = MainWindow::transitions.count();
221     for (int i = 0; i < max; ++i) {
222         effectInfo = MainWindow::transitions.at(i);
223         effectsMap.insert(effectInfo.firstChildElement("name").text().toLower().toUtf8().data(), effectInfo);
224     }
225     MainWindow::transitions.clearList();
226     foreach(const QDomElement & effect, effectsMap)
227         MainWindow::transitions.append(effect);
228     effectsMap.clear();
229
230     // Create effects from MLT
231     foreach(const QString & filtername, mltFiltersList) {
232         QDomDocument doc = createDescriptionFromMlt(repository, "filters", filtername);
233         //WARNING: TEMPORARY FIX for empty MLT effects descriptions - disable effects without parameters - jbm 09-06-2011
234         if (!doc.isNull() && doc.elementsByTagName("parameter").count() > 0) {
235             if (doc.documentElement().attribute("type") == "audio") {
236                 if (doc.elementsByTagName("description").count() > 0) {
237                     QString desc = doc.documentElement().firstChildElement("description").text();
238                     //WARNING: TEMPORARY FIX for unusable MLT SOX parameters description
239                     if (desc.startsWith("Process audio using a SoX")) {
240                         // Remove MLT's SOX generated effects since the parameters properties are unusable for us
241                     }
242                     else audioEffectsMap.insert(doc.documentElement().firstChildElement("name").text().toLower().toUtf8().data(), doc.documentElement());
243                 }
244             }
245             else
246                 videoEffectsMap.insert(doc.documentElement().firstChildElement("name").text().toLower().toUtf8().data(), doc.documentElement());
247         }
248     }
249
250     // Set the directories to look into for effects.
251     QStringList direc = KGlobal::dirs()->findDirs("appdata", "effects");
252     // Iterate through effects directories to parse all XML files.
253     for (more = direc.begin(); more != direc.end(); ++more) {
254         QDir directory(*more);
255         QStringList filter;
256         filter << "*.xml";
257         fileList = directory.entryList(filter, QDir::Files);
258         for (it = fileList.begin(); it != fileList.end(); ++it) {
259             itemName = KUrl(*more + *it).path();
260             parseEffectFile(&MainWindow::customEffects,
261                             &MainWindow::audioEffects,
262                             &MainWindow::videoEffects,
263                             itemName, filtersList, producersList, repository);
264         }
265     }
266
267     // Create custom effects
268     max = MainWindow::customEffects.count();
269     for (int i = 0; i < max; ++i) {
270         effectInfo = MainWindow::customEffects.at(i);
271         if (effectInfo.tagName() == "effectgroup") {
272             effectsMap.insert(effectInfo.attribute("name").toLower().toUtf8().data(), effectInfo);
273         }
274         else effectsMap.insert(effectInfo.firstChildElement("name").text().toLower().toUtf8().data(), effectInfo);
275     }
276     MainWindow::customEffects.clearList();
277     foreach(const QDomElement & effect, effectsMap)
278         MainWindow::customEffects.append(effect);
279     effectsMap.clear();
280
281     // Create audio effects
282     max = MainWindow::audioEffects.count();
283     for (int i = 0; i < max; ++i) {
284         effectInfo = MainWindow::audioEffects.at(i);
285         audioEffectsMap.insert(effectInfo.firstChildElement("name").text().toLower().toUtf8().data(), effectInfo);
286     }
287     MainWindow::audioEffects.clearList();
288     foreach(const QDomElement & effect, audioEffectsMap)
289         MainWindow::audioEffects.append(effect);
290
291     // Create video effects
292     max = MainWindow::videoEffects.count();
293     for (int i = 0; i < max; ++i) {
294         effectInfo = MainWindow::videoEffects.at(i);
295         videoEffectsMap.insert(effectInfo.firstChildElement("name").text().toLower().toUtf8().data(), effectInfo);
296     }
297     MainWindow::videoEffects.clearList();
298     foreach(const QDomElement & effect, videoEffectsMap)
299         MainWindow::videoEffects.append(effect);
300 }
301
302 // static
303 void initEffects::parseCustomEffectsFile()
304 {
305     MainWindow::customEffects.clearList();
306     /*
307      * Why a QMap? See parseEffectFiles(). It's probably useless here, but we
308      * cannot be sure about it.
309      */
310     QMap<QString, QDomElement> effectsMap;
311     QString path = KStandardDirs::locateLocal("appdata", "effects/", true);
312     QDir directory = QDir(path);
313     QStringList filter;
314     filter << "*.xml";
315     const QStringList fileList = directory.entryList(filter, QDir::Files);
316     /*
317      * We need to declare these variables outside the foreach, or the QMap will
318      * refer to non existing variables (QMap::insert() takes references as
319      * parameters).
320      */
321     QDomDocument doc;
322     QDomNodeList effects;
323     QDomElement e;
324     int unknownGroupCount = 0;
325     foreach(const QString & filename, fileList) {
326         QString itemName = KUrl(path + filename).path();
327         QFile file(itemName);
328         doc.setContent(&file, false);
329         file.close();
330         QDomElement base = doc.documentElement();
331         if (base.tagName() == "effectgroup") {
332             QString groupName = base.attribute("name");
333             if (groupName.isEmpty()) {
334                 groupName = i18n("Group %1", unknownGroupCount);
335                 base.setAttribute("name", groupName);
336                 unknownGroupCount++;
337             }
338             effectsMap.insert(groupName.toLower().toUtf8().data(), base);
339         } else if (base.tagName() == "effect") {
340             effectsMap.insert(e.firstChildElement("name").text().toLower().toUtf8().data(), base);
341         }
342         else kDebug() << "Unsupported effect file: " << itemName;
343     }
344     foreach(const QDomElement & effect, effectsMap)
345         MainWindow::customEffects.append(effect);
346 }
347
348 // static
349 void initEffects::parseEffectFile(EffectsList *customEffectList, EffectsList *audioEffectList, EffectsList *videoEffectList, QString name, QStringList filtersList, QStringList producersList, Mlt::Repository *repository)
350 {
351     QDomDocument doc;
352     QFile file(name);
353     doc.setContent(&file, false);
354     file.close();
355     QDomElement documentElement;
356     QDomNodeList effects;
357     QDomElement base = doc.documentElement();
358     effects = doc.elementsByTagName("effect");
359
360     if (effects.count() == 0) {
361         kDebug() << "Effect broken: " << name;
362         return;
363     }
364
365     bool needsLocaleConversion = false;
366     for (int i = 0; !effects.item(i).isNull(); ++i) {
367         QLocale locale;
368         documentElement = effects.item(i).toElement();
369         QString tag = documentElement.attribute("tag", QString());
370         if (documentElement.hasAttribute("LC_NUMERIC")) {
371             // set a locale for that file
372             locale = QLocale(documentElement.attribute("LC_NUMERIC"));
373             if (locale.decimalPoint() != QLocale().decimalPoint()) {
374                 needsLocaleConversion = true;
375             }
376         }
377         locale.setNumberOptions(QLocale::OmitGroupSeparator);
378
379         if (needsLocaleConversion) {
380             // we need to convert all numbers to the system's locale (for example 0.5 -> 0,5)
381             QChar separator = QLocale().decimalPoint();
382             QChar oldSeparator = locale.decimalPoint();
383             QDomNodeList params = documentElement.elementsByTagName("parameter");
384             for (int j = 0; j < params.count(); j++) {
385                 QDomNamedNodeMap attrs = params.at(j).attributes();
386                 for (int k = 0; k < attrs.count(); k++) {
387                     QString name = attrs.item(k).nodeName();
388                     if (name != "type" && name != "name") {
389                             QString val = attrs.item(k).nodeValue();
390                             if (val.contains(oldSeparator)) {
391                                 QString newVal = val.replace(oldSeparator, separator);
392                                 attrs.item(k).setNodeValue(newVal);
393                             }
394                     }
395                 }
396             }
397         }
398
399         double version = -1;
400         Mlt::Properties *metadata = repository->metadata(filter_type, tag.toUtf8().data());
401         if (metadata && metadata->is_valid()) {
402             version = metadata->get_double("version");
403         }
404         if (metadata) delete metadata;
405         if (documentElement.hasAttribute("version")) {
406             // a specific version of the filter is required
407             if (locale.toDouble(documentElement.attribute("version")) > version) {
408                 return;
409             }
410         }
411         if (version > -1) {
412             // Add version info to XML
413             QDomNode versionNode = doc.createElement("version");
414             versionNode.appendChild(doc.createTextNode(QLocale().toString(version)));
415             documentElement.appendChild(versionNode);
416         }
417
418         // Parse effect information.
419         if (base.tagName() != "effectgroup" && (filtersList.contains(tag) || producersList.contains(tag))) {
420             QString type = documentElement.attribute("type", QString());
421             if (type == "audio")
422                 audioEffectList->append(documentElement);
423             else if (type == "custom")
424                 customEffectList->append(documentElement);
425             else
426                 videoEffectList->append(documentElement);
427         }
428     }
429     if (base.tagName() == "effectgroup") {
430         QString type = base.attribute("type", QString());
431         if (type == "audio")
432             audioEffectList->append(base);
433         else if (type == "custom")
434             customEffectList->append(base);
435         else
436             videoEffectList->append(base);
437     }
438 }
439
440 QDomDocument initEffects::createDescriptionFromMlt(Mlt::Repository* repository, const QString& /*type*/, const QString& filtername)
441 {
442
443     QDomDocument ret;
444     Mlt::Properties *metadata = repository->metadata(filter_type, filtername.toAscii().data());
445     //kDebug() << filtername;
446     if (metadata && metadata->is_valid()) {
447         if (metadata->get("title") && metadata->get("identifier")) {
448             QDomElement eff = ret.createElement("effect");
449             QString id = metadata->get("identifier");
450             eff.setAttribute("tag", id);
451             eff.setAttribute("id", id);
452             //kDebug()<<"Effect: "<<id;
453
454             QDomElement name = ret.createElement("name");
455             name.appendChild(ret.createTextNode(metadata->get("title")));
456
457             QDomElement desc = ret.createElement("description");
458             desc.appendChild(ret.createTextNode(metadata->get("description")));
459
460             QDomElement author = ret.createElement("author");
461             author.appendChild(ret.createTextNode(metadata->get("creator")));
462
463             QDomElement version = ret.createElement("version");
464             version.appendChild(ret.createTextNode(metadata->get("version")));
465
466             eff.appendChild(name);
467             eff.appendChild(author);
468             eff.appendChild(desc);
469             eff.appendChild(version);
470
471             Mlt::Properties tags((mlt_properties) metadata->get_data("tags"));
472             if (QString(tags.get(0)) == "Audio") eff.setAttribute("type", "audio");
473             /*for (int i = 0; i < tags.count(); i++)
474                 kDebug()<<tags.get_name(i)<<"="<<tags.get(i);*/
475
476             Mlt::Properties param_props((mlt_properties) metadata->get_data("parameters"));
477             for (int j = 0; param_props.is_valid() && j < param_props.count(); j++) {
478                 QDomElement params = ret.createElement("parameter");
479
480                 Mlt::Properties paramdesc((mlt_properties) param_props.get_data(param_props.get_name(j)));
481
482                 params.setAttribute("name", paramdesc.get("identifier"));
483
484                 if (paramdesc.get("maximum")) params.setAttribute("max", paramdesc.get("maximum"));
485                 if (paramdesc.get("minimum")) params.setAttribute("min", paramdesc.get("minimum"));
486
487                 QString paramType = paramdesc.get("type");
488                 
489                 if (paramType == "integer")
490                     params.setAttribute("type", "constant");
491                 else if (paramType == "float") {
492                     params.setAttribute("type", "constant");
493                     // param type is float, set default decimals to 3
494                     params.setAttribute("decimals", "3");
495                 }
496                 else if (paramType == "boolean")
497                     params.setAttribute("type", "bool");
498                 else if (paramType == "geometry") {
499                     params.setAttribute("type", "geometry");
500                 }
501                 else {
502                     params.setAttribute("type", paramType);
503                     if (!QString(paramdesc.get("format")).isEmpty()) params.setAttribute("format", paramdesc.get("format"));
504                 }
505                 if (paramdesc.get("default")) params.setAttribute("default", paramdesc.get("default"));
506                 if (paramdesc.get("value")) {
507                     params.setAttribute("value", paramdesc.get("value"));
508                 } else {
509                     params.setAttribute("value", paramdesc.get("default"));
510                 }
511
512                 QDomElement pname = ret.createElement("name");
513                 pname.appendChild(ret.createTextNode(paramdesc.get("title")));
514                 params.appendChild(pname);
515
516                 eff.appendChild(params);
517             }
518             ret.appendChild(eff);
519         }
520     }
521     delete metadata;
522     metadata = 0;
523     /*QString outstr;
524      QTextStream str(&outstr);
525      ret.save(str, 2);
526      kDebug() << outstr;*/
527     return ret;
528 }
529
530 void initEffects::fillTransitionsList(Mlt::Repository *repository, EffectsList *transitions, QStringList names)
531 {
532     // Remove transitions that are not implemented.
533     int pos = names.indexOf("mix");
534     if (pos != -1)
535         names.takeAt(pos);
536
537     QStringList imagenamelist = QStringList() << i18n("None");
538     QStringList imagefiles = QStringList() << QString();
539     QStringList filters;
540     filters << "*.pgm" << "*.png";
541     QStringList customLumas = KGlobal::dirs()->findDirs("appdata", "lumas");
542     foreach(QString folder, customLumas) {
543         if (!folder.endsWith('/'))
544             folder.append('/');
545         QStringList filesnames = QDir(folder).entryList(filters, QDir::Files);
546         foreach(const QString & fname, filesnames) {
547             imagenamelist.append(fname);
548             imagefiles.append(folder + fname);
549         }
550     }
551
552     // Check for MLT luma files.
553     KUrl folder(mlt_environment("MLT_DATA"));
554     folder.addPath("lumas");
555     folder.addPath(mlt_environment("MLT_NORMALISATION"));
556     QDir lumafolder(folder.path());
557     QStringList filesnames = lumafolder.entryList(filters, QDir::Files);
558     foreach(const QString & fname, filesnames) {
559         imagenamelist.append(fname);
560         KUrl path(folder);
561         path.addPath(fname);
562         imagefiles.append(path.toLocalFile());
563     }
564     
565     //WARNING: this is a hack to get around temporary invalid metadata in MLT, 2nd of june 2011 JBM
566     QStringList customTransitions;
567     customTransitions << "composite" << "luma" << "affine" << "mix" << "region";
568
569     foreach(const QString & name, names) {
570         QDomDocument ret;
571         QDomElement ktrans = ret.createElement("ktransition");
572         ret.appendChild(ktrans);
573         ktrans.setAttribute("tag", name);
574
575         QDomElement tname = ret.createElement("name");
576         QDomElement desc = ret.createElement("description");
577         ktrans.appendChild(tname);
578         ktrans.appendChild(desc);
579         Mlt::Properties *metadata = NULL;
580         if (!customTransitions.contains(name)) metadata = repository->metadata(transition_type, name.toUtf8().data());
581         if (metadata && metadata->is_valid()) {
582             // If possible, set name and description.
583             if (metadata->get("title") && metadata->get("identifier"))
584                 tname.appendChild(ret.createTextNode(metadata->get("title")));
585             desc.appendChild(ret.createTextNode(metadata->get("description")));
586
587             Mlt::Properties param_props((mlt_properties) metadata->get_data("parameters"));
588             for (int i = 0; param_props.is_valid() && i < param_props.count(); ++i) {
589                 QDomElement params = ret.createElement("parameter");
590
591                 Mlt::Properties paramdesc((mlt_properties) param_props.get_data(param_props.get_name(i)));
592
593                 params.setAttribute("name", paramdesc.get("identifier"));
594
595                 if (paramdesc.get("maximum"))
596                     params.setAttribute("max", paramdesc.get("maximum"));
597                 if (paramdesc.get("minimum"))
598                     params.setAttribute("min", paramdesc.get("minimum"));
599                 if (QString(paramdesc.get("type")) == "integer") {
600                     params.setAttribute("type", "constant");
601                     params.setAttribute("factor", "100");
602                 }
603                 if (QString(paramdesc.get("type")) == "boolean")
604                     params.setAttribute("type", "bool");
605                 if (!QString(paramdesc.get("format")).isEmpty()) {
606                     params.setAttribute("type", "complex");
607                     params.setAttribute("format", paramdesc.get("format"));
608                 }
609                 if (paramdesc.get("default"))
610                     params.setAttribute("default", paramdesc.get("default"));
611                 if (paramdesc.get("value"))
612                     params.setAttribute("value", paramdesc.get("value"));
613                 else
614                     params.setAttribute("value", paramdesc.get("default"));
615
616                 QDomElement pname = ret.createElement("name");
617                 pname.appendChild(ret.createTextNode(paramdesc.get("title")));
618                 params.appendChild(pname);
619                 ktrans.appendChild(params);
620             }
621         } else {
622             /*
623              * Check for Kdenlive installed luma files, add empty string at
624              * start for no luma file.
625              */
626
627             // Implement default transitions.
628             QList<QDomElement> paramList;
629             if (name == "luma") {
630                 ktrans.setAttribute("id", name);
631                 tname.appendChild(ret.createTextNode(i18n("Wipe")));
632                 desc.appendChild(ret.createTextNode(i18n("Applies a stationary transition between the current and next frames.")));
633
634                 paramList.append(quickParameterFill(ret, i18n("Softness"), "softness", "double", "0", "0", "100", "", "", "100"));
635                 paramList.append(quickParameterFill(ret, i18nc("@property: means that the image is inverted", "Invert"), "invert", "bool", "0", "0", "1"));
636                 paramList.append(quickParameterFill(ret, i18n("Image File"), "resource", "list", "", "", "", imagefiles.join(";"), imagenamelist.join(",")));
637                 paramList.append(quickParameterFill(ret, i18n("Reverse Transition"), "reverse", "bool", "0", "0", "1"));
638                 //thumbnailer.prepareThumbnailsCall(imagelist);
639             } else if (name == "composite") {
640                 ktrans.setAttribute("id", name);
641                 tname.appendChild(ret.createTextNode(i18n("Composite")));
642                 desc.appendChild(ret.createTextNode(i18n("A key-framable alpha-channel compositor for two frames.")));
643                 paramList.append(quickParameterFill(ret, i18n("Geometry"), "geometry", "geometry", "0%/0%:100%x100%:100", "-500;-500;-500;-500;0", "500;500;500;500;100"));
644                 paramList.append(quickParameterFill(ret, i18n("Alpha Channel Operation"), "operator", "list", "over", "", "", "over,and,or,xor", i18n("Over,And,Or,Xor")));
645                 paramList.append(quickParameterFill(ret, i18n("Align"), "aligned", "bool", "1", "0", "1"));
646                 paramList.append(quickParameterFill(ret, i18n("Fill"), "fill", "bool", "1", "0", "1"));
647                 paramList.append(quickParameterFill(ret, i18n("Distort"), "distort", "bool", "0", "0", "1"));
648                 paramList.append(quickParameterFill(ret, i18n("Wipe File"), "luma", "list", "", "", "", imagefiles.join(";"), imagenamelist.join(",")));
649                 paramList.append(quickParameterFill(ret, i18n("Wipe Softness"), "softness", "double", "0", "0", "100", "", "", "100"));
650                 paramList.append(quickParameterFill(ret, i18n("Wipe Invert"), "luma_invert", "bool", "0", "0", "1"));
651                 paramList.append(quickParameterFill(ret, i18n("Force Progressive Rendering"), "progressive", "bool", "1", "0", "1"));
652                 paramList.append(quickParameterFill(ret, i18n("Force Deinterlace Overlay"), "deinterlace", "bool", "0", "0", "1"));
653             } else if (name == "affine") {
654                 tname.appendChild(ret.createTextNode(i18n("Affine")));
655                 ret.documentElement().setAttribute("showrotation", "1");
656                 /*paramList.append(quickParameterFill(ret, i18n("Rotate Y"), "rotate_y", "double", "0", "0", "360"));
657                 paramList.append(quickParameterFill(ret, i18n("Rotate X"), "rotate_x", "double", "0", "0", "360"));
658                 paramList.append(quickParameterFill(ret, i18n("Rotate Z"), "rotate_z", "double", "0", "0", "360"));
659                 paramList.append(quickParameterFill(ret, i18n("Fix Rotate Y"), "fix_rotate_y", "double", "0", "0", "360"));
660                 paramList.append(quickParameterFill(ret, i18n("Fix Rotate X"), "fix_rotate_x", "double", "0", "0", "360"));
661                 paramList.append(quickParameterFill(ret, i18n("Fix Rotate Z"), "fix_rotate_z", "double", "0", "0", "360"));
662                 paramList.append(quickParameterFill(ret, i18n("Shear Y"), "shear_y", "double", "0", "0", "360"));
663                 paramList.append(quickParameterFill(ret, i18n("Shear X"), "shear_x", "double", "0", "0", "360"));
664                 paramList.append(quickParameterFill(ret, i18n("Shear Z"), "shear_z", "double", "0", "0", "360"));*/
665                 /*paramList.append(quickParameterFill(ret, i18n("Fix Shear Y"), "fix_shear_y", "double", "0", "0", "360"));
666                 paramList.append(quickParameterFill(ret, i18n("Fix Shear X"), "fix_shear_x", "double", "0", "0", "360"));
667                 paramList.append(quickParameterFill(ret, i18n("Fix Shear Z"), "fix_shear_z", "double", "0", "0", "360"));*/
668
669                 paramList.append(quickParameterFill(ret, "keyed", "keyed", "fixed", "1", "1", "1"));
670                 paramList.append(quickParameterFill(ret, i18n("Geometry"), "geometry", "geometry",  "0/0:100%x100%:100%", "0/0:100%x100%:100%", "0/0:100%x100%:100%", "", "", "", "", "", "true"));
671                 paramList.append(quickParameterFill(ret, i18n("Distort"), "distort", "bool", "0", "0", "1"));
672                 paramList.append(quickParameterFill(ret, i18n("Rotate X"), "rotate_x", "addedgeometry", "0", "-1800", "1800", QString(), QString(), "10"));
673                 paramList.append(quickParameterFill(ret, i18n("Rotate Y"), "rotate_y", "addedgeometry", "0", "-1800", "1800", QString(), QString(), "10"));
674                 paramList.append(quickParameterFill(ret, i18n("Rotate Z"), "rotate_z", "addedgeometry", "0", "-1800", "1800", QString(), QString(), "10"));
675                 /*paramList.append(quickParameterFill(ret, i18n("Rotate Y"), "rotate_y", "simplekeyframe", "0", "-1800", "1800", QString(), QString(), "10"));
676                 paramList.append(quickParameterFill(ret, i18n("Rotate Z"), "rotate_z", "simplekeyframe", "0", "-1800", "1800", QString(), QString(), "10"));*/
677                 
678                 paramList.append(quickParameterFill(ret, i18n("Fix Shear Y"), "shear_y", "double", "0", "0", "360"));
679                 paramList.append(quickParameterFill(ret, i18n("Fix Shear X"), "shear_x", "double", "0", "0", "360"));
680                 paramList.append(quickParameterFill(ret, i18n("Fix Shear Z"), "shear_z", "double", "0", "0", "360"));
681             } else if (name == "mix") {
682                 tname.appendChild(ret.createTextNode(i18n("Mix")));
683             } else if (name == "region") {
684                 ktrans.setAttribute("id", name);
685                 tname.appendChild(ret.createTextNode(i18n("Region")));
686                 desc.appendChild(ret.createTextNode(i18n("Use alpha channel of another clip to create a transition.")));
687                 paramList.append(quickParameterFill(ret, i18n("Transparency clip"), "resource", "url", "", "", "", "", "", ""));
688                 paramList.append(quickParameterFill(ret, i18n("Geometry"), "composite.geometry", "geometry", "0%/0%:100%x100%:100", "-500;-500;-500;-500;0", "500;500;500;500;100"));
689                 paramList.append(quickParameterFill(ret, i18n("Alpha Channel Operation"), "composite.operator", "list", "over", "", "", "over,and,or,xor", i18n("Over,And,Or,Xor")));
690                 paramList.append(quickParameterFill(ret, i18n("Align"), "composite.aligned", "bool", "1", "0", "1"));
691                 paramList.append(quickParameterFill(ret, i18n("Fill"), "composite.fill", "bool", "1", "0", "1"));
692                 paramList.append(quickParameterFill(ret, i18n("Distort"), "composite.distort", "bool", "0", "0", "1"));
693                 paramList.append(quickParameterFill(ret, i18n("Wipe File"), "composite.luma", "list", "", "", "", imagefiles.join(";"), imagenamelist.join(",")));
694                 paramList.append(quickParameterFill(ret, i18n("Wipe Softness"), "composite.softness", "double", "0", "0", "100", "", "", "100"));
695                 paramList.append(quickParameterFill(ret, i18n("Wipe Invert"), "composite.luma_invert", "bool", "0", "0", "1"));
696                 paramList.append(quickParameterFill(ret, i18n("Force Progressive Rendering"), "composite.progressive", "bool", "1", "0", "1"));
697                 paramList.append(quickParameterFill(ret, i18n("Force Deinterlace Overlay"), "composite.deinterlace", "bool", "0", "0", "1"));
698             }
699             foreach(const QDomElement & e, paramList)
700             ktrans.appendChild(e);
701         }
702         delete metadata;
703         metadata = 0;
704         // Add the transition to the global list.
705         transitions->append(ret.documentElement());
706         //kDebug() << ret.toString();
707     }
708
709     // Add some virtual transitions.
710     QString slidetrans = "<ktransition tag=\"composite\" id=\"slide\"><name>" + i18n("Slide") + "</name><description>" + i18n("Slide image from one side to another.") + "</description><parameter tag=\"geometry\" type=\"wipe\" default=\"-100%,0%:100%x100%;-1=0%,0%:100%x100%\" name=\"geometry\"><name>" + i18n("Direction") + "</name>                                               </parameter><parameter tag=\"aligned\" default=\"0\" type=\"bool\" name=\"aligned\" ><name>" + i18n("Align") + "</name></parameter><parameter tag=\"progressive\" default=\"1\" type=\"bool\" name=\"progressive\" ><name>" + i18n("Force Progressive Rendering") + "</name></parameter><parameter tag=\"deinterlace\" default=\"0\" type=\"bool\" name=\"deinterlace\" ><name>" + i18n("Force Deinterlace Overlay") + "</name></parameter><parameter tag=\"invert\" default=\"0\" type=\"bool\" name=\"invert\" ><name>" + i18nc("@property: means that the image is inverted", "Invert") + "</name></parameter></ktransition>";
711     QDomDocument ret;
712     ret.setContent(slidetrans);
713     transitions->append(ret.documentElement());
714
715     QString dissolve = "<ktransition tag=\"luma\" id=\"dissolve\"><name>" + i18n("Dissolve") + "</name><description>" + i18n("Fade out one video while fading in the other video.") + "</description><parameter tag=\"reverse\" default=\"0\" type=\"bool\" name=\"reverse\" ><name>" + i18n("Reverse") + "</name></parameter></ktransition>";
716     ret.setContent(dissolve);
717     transitions->append(ret.documentElement());
718
719     /*QString alphatrans = "<ktransition tag=\"composite\" id=\"alphatransparency\" ><name>" + i18n("Alpha Transparency") + "</name><description>" + i18n("Make alpha channel transparent.") + "</description><parameter tag=\"geometry\" type=\"fixed\" default=\"0%,0%:100%x100%\" name=\"geometry\"><name>" + i18n("Direction") + "</name></parameter><parameter tag=\"fill\" default=\"0\" type=\"bool\" name=\"fill\" ><name>" + i18n("Rescale") + "</name></parameter><parameter tag=\"aligned\" default=\"0\" type=\"bool\" name=\"aligned\" ><name>" + i18n("Align") + "</name></parameter></ktransition>";
720     ret.setContent(alphatrans);
721     transitions->append(ret.documentElement());*/
722 }
723
724 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)
725 {
726     QDomElement parameter = doc.createElement("parameter");
727     parameter.setAttribute("tag", tag);
728     parameter.setAttribute("default", def);
729     parameter.setAttribute("type", type);
730     parameter.setAttribute("name", tag);
731     parameter.setAttribute("max", max);
732     parameter.setAttribute("min", min);
733     if (!list.isEmpty())
734         parameter.setAttribute("paramlist", list);
735     if (!listdisplaynames.isEmpty())
736         parameter.setAttribute("paramlistdisplay", listdisplaynames);
737     if (!factor.isEmpty())
738         parameter.setAttribute("factor", factor);
739     if (!namedesc.isEmpty())
740         parameter.setAttribute("namedesc", namedesc);
741     if (!format.isEmpty())
742         parameter.setAttribute("format", format);
743     if (!opacity.isEmpty())
744         parameter.setAttribute("opacity", opacity);
745     QDomElement pname = doc.createElement("name");
746     pname.appendChild(doc.createTextNode(name));
747     parameter.appendChild(pname);
748
749     return parameter;
750 }