1 /***************************************************************************
2 initeffects.cpp - description
5 copyright : (C) 2006 by Jean-Baptiste Mardelle
7 copyright : (C) 2008 Marco Gittler
8 email : g.marco@freenet.de
9 ***************************************************************************/
11 /***************************************************************************
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. *
18 ***************************************************************************/
20 #include "initeffects.h"
21 #include "kdenlivesettings.h"
22 #include "effectslist.h"
23 #include "mainwindow.h"
27 #include <KStandardDirs>
36 initEffectsThumbnailer::initEffectsThumbnailer() :
41 void initEffectsThumbnailer::prepareThumbnailsCall(const QStringList& list)
48 void initEffectsThumbnailer::run()
50 foreach(const QString & entry, m_list) {
52 if (!entry.isEmpty() && (entry.endsWith(".png") || entry.endsWith(".pgm"))) {
53 if (!MainWindow::m_lumacache.contains(entry)) {
56 MainWindow::m_lumacache.insert(entry, pix.scaled(30, 30, Qt::KeepAspectRatio, Qt::SmoothTransformation));
63 initEffectsThumbnailer initEffects::thumbnailer;
66 void initEffects::refreshLumas()
68 // Check for Kdenlive installed luma files, add empty string at start for no luma
69 QStringList imagenamelist = QStringList() << i18n("None");
70 QStringList imagefiles = QStringList() << QString();
72 filters << "*.pgm" << "*.png";
74 QStringList customLumas = KGlobal::dirs()->findDirs("appdata", "lumas");
75 foreach(const QString & folder, customLumas) {
76 QStringList filesnames = QDir(folder).entryList(filters, QDir::Files);
77 foreach(const QString & fname, filesnames) {
78 imagenamelist.append(fname);
79 imagefiles.append(KUrl(folder).path(KUrl::AddTrailingSlash) + fname);
83 // Check for MLT lumas
84 KUrl folder(mlt_environment("MLT_DATA"));
85 folder.addPath("lumas");
86 folder.addPath(mlt_environment("MLT_NORMALISATION"));
87 QDir lumafolder(folder.path());
88 QStringList filesnames = lumafolder.entryList(filters, QDir::Files);
89 foreach(const QString & fname, filesnames) {
90 imagenamelist.append(fname);
93 imagefiles.append(path.toLocalFile());
95 QDomElement lumaTransition = MainWindow::transitions.getEffectByTag("luma", "luma");
96 QDomNodeList params = lumaTransition.elementsByTagName("parameter");
97 for (int i = 0; i < params.count(); i++) {
98 QDomElement e = params.item(i).toElement();
99 if (e.attribute("tag") == "resource") {
100 e.setAttribute("paramlistdisplay", imagenamelist.join(","));
101 e.setAttribute("paramlist", imagefiles.join(";"));
106 QDomElement compositeTransition = MainWindow::transitions.getEffectByTag("composite", "composite");
107 params = compositeTransition.elementsByTagName("parameter");
108 for (int i = 0; i < params.count(); i++) {
109 QDomElement e = params.item(i).toElement();
110 if (e.attribute("tag") == "luma") {
111 e.setAttribute("paramlistdisplay", imagenamelist.join(","));
112 e.setAttribute("paramlist", imagefiles.join(";"));
119 QDomDocument initEffects::getUsedCustomEffects(QMap <QString, QString> effectids)
121 QMapIterator<QString, QString> i(effectids);
124 QDomElement list = doc.createElement("customeffects");
125 doc.appendChild(list);
126 while (i.hasNext()) {
128 ix = MainWindow::customEffects.hasEffect(i.value(), i.key());
130 QDomElement e = MainWindow::customEffects.at(ix);
131 list.appendChild(doc.importNode(e, true));
138 void initEffects::parseEffectFiles(const QString &locale)
140 QStringList::Iterator more;
141 QStringList::Iterator it;
142 QStringList fileList;
145 Mlt::Repository *repository = Mlt::Factory::init();
147 kDebug() << "Repository didn't finish initialisation" ;
151 // Warning: Mlt::Factory::init() resets the locale to the default system value, make sure we keep correct locale
152 if (!locale.isEmpty()) setlocale(LC_NUMERIC, locale.toUtf8().constData());
154 // Retrieve the list of MLT's available effects.
155 Mlt::Properties *filters = repository->filters();
156 QStringList filtersList;
157 int max = filters->count();
158 for (int i = 0; i < max; ++i)
159 filtersList << filters->get_name(i);
162 // Retrieve the list of available producers.
163 Mlt::Properties *producers = repository->producers();
164 QStringList producersList;
165 max = producers->count();
166 for (int i = 0; i < max; ++i)
167 producersList << producers->get_name(i);
168 KdenliveSettings::setProducerslist(producersList);
171 // Retrieve the list of available transitions.
172 Mlt::Properties *transitions = repository->transitions();
173 QStringList transitionsItemList;
174 max = transitions->count();
175 for (int i = 0; i < max; ++i)
176 transitionsItemList << transitions->get_name(i);
179 // Remove blacklisted transitions from the list.
180 QFile file(KStandardDirs::locate("appdata", "blacklisted_transitions.txt"));
181 if (file.open(QIODevice::ReadOnly)) {
182 QTextStream in(&file);
183 while (!in.atEnd()) {
184 QString black = in.readLine().simplified();
185 if (!black.isEmpty() && !black.startsWith('#') &&
186 transitionsItemList.contains(black))
187 transitionsItemList.removeAll(black);
192 // Fill transitions list.
193 fillTransitionsList(repository, &MainWindow::transitions, transitionsItemList);
195 // Remove blacklisted effects from the filters list.
196 QStringList mltFiltersList = filtersList;
197 QFile file2(KStandardDirs::locate("appdata", "blacklisted_effects.txt"));
198 if (file2.open(QIODevice::ReadOnly)) {
199 QTextStream in(&file2);
200 while (!in.atEnd()) {
201 QString black = in.readLine().simplified();
202 if (!black.isEmpty() && !black.startsWith('#') &&
203 mltFiltersList.contains(black)) {
204 mltFiltersList.removeAll(black);
212 * Cleanup the global lists. We use QMap because of its automatic sorting
213 * (by key) and key uniqueness (using insert() instead of insertMulti()).
214 * This introduces some more cycles (while removing them from other parts of
215 * the code and centralising them), but due to the way this methods, QMap
216 * and EffectsList are implemented, there's no easy way to make it
217 * differently without reinplementing something (which should really be
220 QDomElement effectInfo;
221 QMap<QString, QDomElement> effectsMap;
222 QMap<QString, QDomElement> videoEffectsMap;
223 QMap<QString, QDomElement> audioEffectsMap;
225 // Create transitions
226 max = MainWindow::transitions.count();
227 for (int i = 0; i < max; ++i) {
228 effectInfo = MainWindow::transitions.at(i);
229 effectsMap.insert(effectInfo.firstChildElement("name").text().toLower().toUtf8().data(), effectInfo);
231 MainWindow::transitions.clearList();
232 foreach(const QDomElement & effect, effectsMap)
233 MainWindow::transitions.append(effect);
235 // Create effects from MLT
236 foreach(const QString & filtername, mltFiltersList) {
237 QDomDocument doc = createDescriptionFromMlt(repository, "filters", filtername);
238 //WARNING: TEMPORARY FIX for empty MLT effects descriptions - disable effects without parameters - jbm 09-06-2011
239 if (!doc.isNull() && doc.elementsByTagName("parameter").count() > 0) {
240 if (doc.documentElement().attribute("type") == "audio") {
241 if (doc.elementsByTagName("description").count() > 0) {
242 QString desc = doc.documentElement().firstChildElement("description").text();
243 //WARNING: TEMPORARY FIX for unusable MLT SOX parameters description
244 if (desc.startsWith("Process audio using a SoX")) {
245 // Remove MLT's SOX generated effects since the parameters properties are unusable for us
248 audioEffectsMap.insert(doc.documentElement().firstChildElement("name").text().toLower().toUtf8().data(), doc.documentElement());
253 videoEffectsMap.insert(doc.documentElement().firstChildElement("name").text().toLower().toUtf8().data(), doc.documentElement());
257 // Set the directories to look into for effects.
258 QStringList direc = KGlobal::dirs()->findDirs("appdata", "effects");
259 // Iterate through effects directories to parse all XML files.
260 for (more = direc.begin(); more != direc.end(); ++more) {
261 QDir directory(*more);
264 fileList = directory.entryList(filter, QDir::Files);
265 for (it = fileList.begin(); it != fileList.end(); ++it) {
266 itemName = KUrl(*more + *it).path();
267 parseEffectFile(&MainWindow::customEffects,
268 &MainWindow::audioEffects,
269 &MainWindow::videoEffects,
270 itemName, filtersList, producersList, repository);
274 // Create custom effects
275 max = MainWindow::customEffects.count();
276 for (int i = 0; i < max; ++i) {
277 effectInfo = MainWindow::customEffects.at(i);
278 if (effectInfo.tagName() == "effectgroup") {
279 effectsMap.insert(effectInfo.attribute("name").toUtf8().data(), effectInfo);
281 else effectsMap.insert(effectInfo.firstChildElement("name").text().toUtf8().data(), effectInfo);
283 MainWindow::customEffects.clearList();
284 foreach(const QDomElement & effect, effectsMap)
285 MainWindow::customEffects.append(effect);
288 // Create audio effects
289 max = MainWindow::audioEffects.count();
290 for (int i = 0; i < max; ++i) {
291 effectInfo = MainWindow::audioEffects.at(i);
292 audioEffectsMap.insert(effectInfo.firstChildElement("name").text().toLower().toUtf8().data(), effectInfo);
294 MainWindow::audioEffects.clearList();
295 foreach(const QDomElement & effect, audioEffectsMap)
296 MainWindow::audioEffects.append(effect);
298 // Create video effects
299 max = MainWindow::videoEffects.count();
300 for (int i = 0; i < max; ++i) {
301 effectInfo = MainWindow::videoEffects.at(i);
302 videoEffectsMap.insert(effectInfo.firstChildElement("name").text().toLower().toUtf8().data(), effectInfo);
304 MainWindow::videoEffects.clearList();
305 foreach(const QDomElement & effect, videoEffectsMap)
306 MainWindow::videoEffects.append(effect);
310 void initEffects::parseCustomEffectsFile()
312 MainWindow::customEffects.clearList();
314 * Why a QMap? See parseEffectFiles(). It's probably useless here, but we
315 * cannot be sure about it.
317 QMap<QString, QDomElement> effectsMap;
318 QString path = KStandardDirs::locateLocal("appdata", "effects/", true);
319 QDir directory = QDir(path);
322 const QStringList fileList = directory.entryList(filter, QDir::Files);
324 * We need to declare these variables outside the foreach, or the QMap will
325 * refer to non existing variables (QMap::insert() takes references as
329 QDomNodeList effects;
331 int unknownGroupCount = 0;
332 foreach(const QString & filename, fileList) {
333 QString itemName = KUrl(path + filename).path();
334 QFile file(itemName);
335 doc.setContent(&file, false);
337 QDomElement base = doc.documentElement();
338 if (base.tagName() == "effectgroup") {
339 QString groupName = base.attribute("name");
340 if (groupName.isEmpty()) {
341 groupName = i18n("Group %1", unknownGroupCount);
342 base.setAttribute("name", groupName);
345 effectsMap.insert(groupName.toLower().toUtf8().data(), base);
346 } else if (base.tagName() == "effect") {
347 effectsMap.insert(base.firstChildElement("name").text().toLower().toUtf8().data(), base);
349 else kDebug() << "Unsupported effect file: " << itemName;
351 foreach(const QDomElement & effect, effectsMap)
352 MainWindow::customEffects.append(effect);
356 void initEffects::parseEffectFile(EffectsList *customEffectList, EffectsList *audioEffectList, EffectsList *videoEffectList, QString name, QStringList filtersList, QStringList producersList, Mlt::Repository *repository)
360 doc.setContent(&file, false);
362 QDomElement documentElement;
363 QDomNodeList effects;
364 QDomElement base = doc.documentElement();
365 effects = doc.elementsByTagName("effect");
367 if (effects.count() == 0) {
368 kDebug() << "Effect broken: " << name;
372 bool needsLocaleConversion = false;
373 for (int i = 0; !effects.item(i).isNull(); ++i) {
375 documentElement = effects.item(i).toElement();
376 QString tag = documentElement.attribute("tag", QString());
377 if (documentElement.hasAttribute("LC_NUMERIC")) {
378 // set a locale for that file
379 locale = QLocale(documentElement.attribute("LC_NUMERIC"));
380 if (locale.decimalPoint() != QLocale().decimalPoint()) {
381 needsLocaleConversion = true;
384 locale.setNumberOptions(QLocale::OmitGroupSeparator);
386 if (needsLocaleConversion) {
387 // we need to convert all numbers to the system's locale (for example 0.5 -> 0,5)
388 QChar separator = QLocale().decimalPoint();
389 QChar oldSeparator = locale.decimalPoint();
390 QDomNodeList params = documentElement.elementsByTagName("parameter");
391 for (int j = 0; j < params.count(); j++) {
392 QDomNamedNodeMap attrs = params.at(j).attributes();
393 for (int k = 0; k < attrs.count(); k++) {
394 QString name = attrs.item(k).nodeName();
395 if (name != "type" && name != "name") {
396 QString val = attrs.item(k).nodeValue();
397 if (val.contains(oldSeparator)) {
398 QString newVal = val.replace(oldSeparator, separator);
399 attrs.item(k).setNodeValue(newVal);
407 Mlt::Properties *metadata = repository->metadata(filter_type, tag.toUtf8().data());
408 if (metadata && metadata->is_valid()) {
409 version = metadata->get_double("version");
411 if (metadata) delete metadata;
412 if (documentElement.hasAttribute("version")) {
413 // a specific version of the filter is required
414 if (locale.toDouble(documentElement.attribute("version")) > version) {
419 // Add version info to XML
420 QDomNode versionNode = doc.createElement("version");
421 versionNode.appendChild(doc.createTextNode(QLocale().toString(version)));
422 documentElement.appendChild(versionNode);
425 // Parse effect information.
426 if (base.tagName() != "effectgroup" && (filtersList.contains(tag) || producersList.contains(tag))) {
427 QString type = documentElement.attribute("type", QString());
429 audioEffectList->append(documentElement);
430 else if (type == "custom")
431 customEffectList->append(documentElement);
433 videoEffectList->append(documentElement);
436 if (base.tagName() == "effectgroup") {
437 QString type = base.attribute("type", QString());
439 audioEffectList->append(base);
440 else if (type == "custom")
441 customEffectList->append(base);
443 videoEffectList->append(base);
447 QDomDocument initEffects::createDescriptionFromMlt(Mlt::Repository* repository, const QString& /*type*/, const QString& filtername)
451 Mlt::Properties *metadata = repository->metadata(filter_type, filtername.toAscii().data());
452 //kDebug() << filtername;
453 if (metadata && metadata->is_valid()) {
454 if (metadata->get("title") && metadata->get("identifier")) {
455 QDomElement eff = ret.createElement("effect");
456 QString id = metadata->get("identifier");
457 eff.setAttribute("tag", id);
458 eff.setAttribute("id", id);
459 //kDebug()<<"Effect: "<<id;
461 QDomElement name = ret.createElement("name");
462 name.appendChild(ret.createTextNode(metadata->get("title")));
464 QDomElement desc = ret.createElement("description");
465 desc.appendChild(ret.createTextNode(metadata->get("description")));
467 QDomElement author = ret.createElement("author");
468 author.appendChild(ret.createTextNode(metadata->get("creator")));
470 QDomElement version = ret.createElement("version");
471 version.appendChild(ret.createTextNode(metadata->get("version")));
473 eff.appendChild(name);
474 eff.appendChild(author);
475 eff.appendChild(desc);
476 eff.appendChild(version);
478 Mlt::Properties tags((mlt_properties) metadata->get_data("tags"));
479 if (QString(tags.get(0)) == "Audio") eff.setAttribute("type", "audio");
480 /*for (int i = 0; i < tags.count(); i++)
481 kDebug()<<tags.get_name(i)<<"="<<tags.get(i);*/
483 Mlt::Properties param_props((mlt_properties) metadata->get_data("parameters"));
484 for (int j = 0; param_props.is_valid() && j < param_props.count(); j++) {
485 QDomElement params = ret.createElement("parameter");
487 Mlt::Properties paramdesc((mlt_properties) param_props.get_data(param_props.get_name(j)));
488 params.setAttribute("name", paramdesc.get("identifier"));
489 if (params.attribute("name") == "argument") {
490 // This parameter has to be given as attribute when using command line, do not show it in Kdenlive
494 if (paramdesc.get("maximum")) params.setAttribute("max", paramdesc.get("maximum"));
495 if (paramdesc.get("minimum")) params.setAttribute("min", paramdesc.get("minimum"));
497 QString paramType = paramdesc.get("type");
499 if (paramType == "integer") {
500 if (params.attribute("min") == "0" && params.attribute("max") == "1")
501 params.setAttribute("type", "bool");
502 else params.setAttribute("type", "constant");
505 else if (paramType == "float") {
506 params.setAttribute("type", "constant");
507 // param type is float, set default decimals to 3
508 params.setAttribute("decimals", "3");
510 else if (paramType == "boolean")
511 params.setAttribute("type", "bool");
512 else if (paramType == "geometry") {
513 params.setAttribute("type", "geometry");
516 params.setAttribute("type", paramType);
517 if (!QString(paramdesc.get("format")).isEmpty()) params.setAttribute("format", paramdesc.get("format"));
519 if (paramdesc.get("default")) params.setAttribute("default", paramdesc.get("default"));
520 if (paramdesc.get("value")) {
521 params.setAttribute("value", paramdesc.get("value"));
523 params.setAttribute("value", paramdesc.get("default"));
526 QDomElement pname = ret.createElement("name");
527 pname.appendChild(ret.createTextNode(paramdesc.get("title")));
528 params.appendChild(pname);
530 if (paramdesc.get("description")) {
531 QDomElement desc = ret.createElement("comment");
532 desc.appendChild(ret.createTextNode(paramdesc.get("description")));
533 params.appendChild(desc);
536 eff.appendChild(params);
538 ret.appendChild(eff);
544 QTextStream str(&outstr);
546 kDebug() << outstr;*/
550 void initEffects::fillTransitionsList(Mlt::Repository *repository, EffectsList *transitions, QStringList names)
552 // Remove transitions that are not implemented.
553 int pos = names.indexOf("mix");
557 QStringList imagenamelist = QStringList() << i18n("None");
558 QStringList imagefiles = QStringList() << QString();
560 filters << "*.pgm" << "*.png";
561 QStringList customLumas = KGlobal::dirs()->findDirs("appdata", "lumas");
562 foreach(QString folder, customLumas) {
563 if (!folder.endsWith('/'))
565 QStringList filesnames = QDir(folder).entryList(filters, QDir::Files);
566 foreach(const QString & fname, filesnames) {
567 imagenamelist.append(fname);
568 imagefiles.append(folder + fname);
572 // Check for MLT luma files.
573 KUrl folder(mlt_environment("MLT_DATA"));
574 folder.addPath("lumas");
575 folder.addPath(mlt_environment("MLT_NORMALISATION"));
576 QDir lumafolder(folder.path());
577 QStringList filesnames = lumafolder.entryList(filters, QDir::Files);
578 foreach(const QString & fname, filesnames) {
579 imagenamelist.append(fname);
582 imagefiles.append(path.toLocalFile());
585 //WARNING: this is a hack to get around temporary invalid metadata in MLT, 2nd of june 2011 JBM
586 QStringList customTransitions;
587 customTransitions << "composite" << "luma" << "affine" << "mix" << "region";
589 foreach(const QString & name, names) {
591 QDomElement ktrans = ret.createElement("ktransition");
592 ret.appendChild(ktrans);
593 ktrans.setAttribute("tag", name);
595 QDomElement tname = ret.createElement("name");
596 QDomElement desc = ret.createElement("description");
597 ktrans.appendChild(tname);
598 ktrans.appendChild(desc);
599 Mlt::Properties *metadata = NULL;
600 if (!customTransitions.contains(name)) metadata = repository->metadata(transition_type, name.toUtf8().data());
601 if (metadata && metadata->is_valid()) {
602 // If possible, set name and description.
603 if (metadata->get("title") && metadata->get("identifier"))
604 tname.appendChild(ret.createTextNode(metadata->get("title")));
605 desc.appendChild(ret.createTextNode(metadata->get("description")));
607 Mlt::Properties param_props((mlt_properties) metadata->get_data("parameters"));
608 for (int i = 0; param_props.is_valid() && i < param_props.count(); ++i) {
609 QDomElement params = ret.createElement("parameter");
611 Mlt::Properties paramdesc((mlt_properties) param_props.get_data(param_props.get_name(i)));
613 params.setAttribute("name", paramdesc.get("identifier"));
615 if (paramdesc.get("maximum"))
616 params.setAttribute("max", paramdesc.get("maximum"));
617 if (paramdesc.get("minimum"))
618 params.setAttribute("min", paramdesc.get("minimum"));
619 if (QString(paramdesc.get("type")) == "integer") {
620 params.setAttribute("type", "constant");
621 params.setAttribute("factor", "100");
623 if (QString(paramdesc.get("type")) == "boolean")
624 params.setAttribute("type", "bool");
625 if (!QString(paramdesc.get("format")).isEmpty()) {
626 params.setAttribute("type", "complex");
627 params.setAttribute("format", paramdesc.get("format"));
629 if (paramdesc.get("default"))
630 params.setAttribute("default", paramdesc.get("default"));
631 if (paramdesc.get("value"))
632 params.setAttribute("value", paramdesc.get("value"));
634 params.setAttribute("value", paramdesc.get("default"));
636 QDomElement pname = ret.createElement("name");
637 pname.appendChild(ret.createTextNode(paramdesc.get("title")));
638 params.appendChild(pname);
639 ktrans.appendChild(params);
643 * Check for Kdenlive installed luma files, add empty string at
644 * start for no luma file.
647 // Implement default transitions.
648 QList<QDomElement> paramList;
649 if (name == "luma") {
650 ktrans.setAttribute("id", name);
651 tname.appendChild(ret.createTextNode(i18n("Wipe")));
652 desc.appendChild(ret.createTextNode(i18n("Applies a stationary transition between the current and next frames.")));
654 paramList.append(quickParameterFill(ret, i18n("Softness"), "softness", "double", "0", "0", "100", "", "", "100"));
655 paramList.append(quickParameterFill(ret, i18nc("@property: means that the image is inverted", "Invert"), "invert", "bool", "0", "0", "1"));
656 paramList.append(quickParameterFill(ret, i18n("Image File"), "resource", "list", "", "", "", imagefiles.join(";"), imagenamelist.join(",")));
657 paramList.append(quickParameterFill(ret, i18n("Reverse Transition"), "reverse", "bool", "0", "0", "1"));
658 //thumbnailer.prepareThumbnailsCall(imagelist);
659 } else if (name == "composite") {
660 ktrans.setAttribute("id", name);
661 tname.appendChild(ret.createTextNode(i18n("Composite")));
662 desc.appendChild(ret.createTextNode(i18n("A key-framable alpha-channel compositor for two frames.")));
663 paramList.append(quickParameterFill(ret, i18n("Geometry"), "geometry", "geometry", "0%/0%:100%x100%:100", "-500;-500;-500;-500;0", "500;500;500;500;100"));
664 paramList.append(quickParameterFill(ret, i18n("Alpha Channel Operation"), "operator", "list", "over", "", "", "over,and,or,xor", i18n("Over,And,Or,Xor")));
665 paramList.append(quickParameterFill(ret, i18n("Align"), "aligned", "bool", "1", "0", "1"));
666 paramList.append(quickParameterFill(ret, i18n("Fill"), "fill", "bool", "1", "0", "1"));
667 paramList.append(quickParameterFill(ret, i18n("Distort"), "distort", "bool", "0", "0", "1"));
668 paramList.append(quickParameterFill(ret, i18n("Wipe File"), "luma", "list", "", "", "", imagefiles.join(";"), imagenamelist.join(",")));
669 paramList.append(quickParameterFill(ret, i18n("Wipe Softness"), "softness", "double", "0", "0", "100", "", "", "100"));
670 paramList.append(quickParameterFill(ret, i18n("Wipe Invert"), "luma_invert", "bool", "0", "0", "1"));
671 paramList.append(quickParameterFill(ret, i18n("Force Progressive Rendering"), "progressive", "bool", "1", "0", "1"));
672 paramList.append(quickParameterFill(ret, i18n("Force Deinterlace Overlay"), "deinterlace", "bool", "0", "0", "1"));
673 } else if (name == "affine") {
674 tname.appendChild(ret.createTextNode(i18n("Affine")));
675 ret.documentElement().setAttribute("showrotation", "1");
676 /*paramList.append(quickParameterFill(ret, i18n("Rotate Y"), "rotate_y", "double", "0", "0", "360"));
677 paramList.append(quickParameterFill(ret, i18n("Rotate X"), "rotate_x", "double", "0", "0", "360"));
678 paramList.append(quickParameterFill(ret, i18n("Rotate Z"), "rotate_z", "double", "0", "0", "360"));
679 paramList.append(quickParameterFill(ret, i18n("Fix Rotate Y"), "fix_rotate_y", "double", "0", "0", "360"));
680 paramList.append(quickParameterFill(ret, i18n("Fix Rotate X"), "fix_rotate_x", "double", "0", "0", "360"));
681 paramList.append(quickParameterFill(ret, i18n("Fix Rotate Z"), "fix_rotate_z", "double", "0", "0", "360"));
682 paramList.append(quickParameterFill(ret, i18n("Shear Y"), "shear_y", "double", "0", "0", "360"));
683 paramList.append(quickParameterFill(ret, i18n("Shear X"), "shear_x", "double", "0", "0", "360"));
684 paramList.append(quickParameterFill(ret, i18n("Shear Z"), "shear_z", "double", "0", "0", "360"));*/
685 /*paramList.append(quickParameterFill(ret, i18n("Fix Shear Y"), "fix_shear_y", "double", "0", "0", "360"));
686 paramList.append(quickParameterFill(ret, i18n("Fix Shear X"), "fix_shear_x", "double", "0", "0", "360"));
687 paramList.append(quickParameterFill(ret, i18n("Fix Shear Z"), "fix_shear_z", "double", "0", "0", "360"));*/
689 paramList.append(quickParameterFill(ret, "keyed", "keyed", "fixed", "1", "1", "1"));
690 paramList.append(quickParameterFill(ret, i18n("Geometry"), "geometry", "geometry", "0/0:100%x100%:100%", "0/0:100%x100%:100%", "0/0:100%x100%:100%", "", "", "", "", "", "true"));
691 paramList.append(quickParameterFill(ret, i18n("Distort"), "distort", "bool", "0", "0", "1"));
692 paramList.append(quickParameterFill(ret, i18n("Rotate X"), "rotate_x", "addedgeometry", "0", "-1800", "1800", QString(), QString(), "10"));
693 paramList.append(quickParameterFill(ret, i18n("Rotate Y"), "rotate_y", "addedgeometry", "0", "-1800", "1800", QString(), QString(), "10"));
694 paramList.append(quickParameterFill(ret, i18n("Rotate Z"), "rotate_z", "addedgeometry", "0", "-1800", "1800", QString(), QString(), "10"));
695 /*paramList.append(quickParameterFill(ret, i18n("Rotate Y"), "rotate_y", "simplekeyframe", "0", "-1800", "1800", QString(), QString(), "10"));
696 paramList.append(quickParameterFill(ret, i18n("Rotate Z"), "rotate_z", "simplekeyframe", "0", "-1800", "1800", QString(), QString(), "10"));*/
698 paramList.append(quickParameterFill(ret, i18n("Fix Shear Y"), "shear_y", "double", "0", "0", "360"));
699 paramList.append(quickParameterFill(ret, i18n("Fix Shear X"), "shear_x", "double", "0", "0", "360"));
700 paramList.append(quickParameterFill(ret, i18n("Fix Shear Z"), "shear_z", "double", "0", "0", "360"));
701 } else if (name == "mix") {
702 tname.appendChild(ret.createTextNode(i18n("Mix")));
703 } else if (name == "region") {
704 ktrans.setAttribute("id", name);
705 tname.appendChild(ret.createTextNode(i18n("Region")));
706 desc.appendChild(ret.createTextNode(i18n("Use alpha channel of another clip to create a transition.")));
707 paramList.append(quickParameterFill(ret, i18n("Transparency clip"), "resource", "url", "", "", "", "", "", ""));
708 paramList.append(quickParameterFill(ret, i18n("Geometry"), "composite.geometry", "geometry", "0%/0%:100%x100%:100", "-500;-500;-500;-500;0", "500;500;500;500;100"));
709 paramList.append(quickParameterFill(ret, i18n("Alpha Channel Operation"), "composite.operator", "list", "over", "", "", "over,and,or,xor", i18n("Over,And,Or,Xor")));
710 paramList.append(quickParameterFill(ret, i18n("Align"), "composite.aligned", "bool", "1", "0", "1"));
711 paramList.append(quickParameterFill(ret, i18n("Fill"), "composite.fill", "bool", "1", "0", "1"));
712 paramList.append(quickParameterFill(ret, i18n("Distort"), "composite.distort", "bool", "0", "0", "1"));
713 paramList.append(quickParameterFill(ret, i18n("Wipe File"), "composite.luma", "list", "", "", "", imagefiles.join(";"), imagenamelist.join(",")));
714 paramList.append(quickParameterFill(ret, i18n("Wipe Softness"), "composite.softness", "double", "0", "0", "100", "", "", "100"));
715 paramList.append(quickParameterFill(ret, i18n("Wipe Invert"), "composite.luma_invert", "bool", "0", "0", "1"));
716 paramList.append(quickParameterFill(ret, i18n("Force Progressive Rendering"), "composite.progressive", "bool", "1", "0", "1"));
717 paramList.append(quickParameterFill(ret, i18n("Force Deinterlace Overlay"), "composite.deinterlace", "bool", "0", "0", "1"));
719 foreach(const QDomElement & e, paramList)
720 ktrans.appendChild(e);
724 // Add the transition to the global list.
725 transitions->append(ret.documentElement());
726 //kDebug() << ret.toString();
729 // Add some virtual transitions.
730 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>";
732 ret.setContent(slidetrans);
733 transitions->append(ret.documentElement());
735 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>";
736 ret.setContent(dissolve);
737 transitions->append(ret.documentElement());
739 /*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>";
740 ret.setContent(alphatrans);
741 transitions->append(ret.documentElement());*/
744 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)
746 QDomElement parameter = doc.createElement("parameter");
747 parameter.setAttribute("tag", tag);
748 parameter.setAttribute("default", def);
749 parameter.setAttribute("type", type);
750 parameter.setAttribute("name", tag);
751 parameter.setAttribute("max", max);
752 parameter.setAttribute("min", min);
754 parameter.setAttribute("paramlist", list);
755 if (!listdisplaynames.isEmpty())
756 parameter.setAttribute("paramlistdisplay", listdisplaynames);
757 if (!factor.isEmpty())
758 parameter.setAttribute("factor", factor);
759 if (!namedesc.isEmpty())
760 parameter.setAttribute("namedesc", namedesc);
761 if (!format.isEmpty())
762 parameter.setAttribute("format", format);
763 if (!opacity.isEmpty())
764 parameter.setAttribute("opacity", opacity);
765 QDomElement pname = doc.createElement("name");
766 pname.appendChild(doc.createTextNode(name));
767 parameter.appendChild(pname);