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