X-Git-Url: https://git.sesse.net/?a=blobdiff_plain;f=src%2Feffectstackview.cpp;h=87e5511f0c95db534087677e5c529296517dc482;hb=08bebc6a135e4297359a361d2f17c02bf593a74a;hp=717586b6c166efa09d6a8ad85da27b5ba54187f0;hpb=2a223cff6e45c560c28857b72c0cb7e584f9a4ef;p=kdenlive diff --git a/src/effectstackview.cpp b/src/effectstackview.cpp index 717586b6..87e5511f 100644 --- a/src/effectstackview.cpp +++ b/src/effectstackview.cpp @@ -15,157 +15,245 @@ * * ***************************************************************************/ -#include -#include #include "effectstackview.h" #include "effectslist.h" #include "clipitem.h" -#include -#include +#include "mainwindow.h" +#include "kdenlivesettings.h" -EffectStackView::EffectStackView(EffectsList *audioEffectList, EffectsList *videoEffectList, EffectsList *customEffectList, QWidget *parent) - : QWidget(parent) { - ui.setupUi(this); - effectedit = new EffectStackEdit(ui.frame, this); - //ui.effectlist->horizontalHeader()->setVisible(false); - //ui.effectlist->verticalHeader()->setVisible(false); - clipref = NULL; - - ui.buttonNew->setIcon(KIcon("document-new")); - ui.buttonNew->setToolTip(i18n("Add new effect")); - ui.buttonUp->setIcon(KIcon("go-up")); - ui.buttonUp->setToolTip(i18n("Move effect up")); - ui.buttonDown->setIcon(KIcon("go-down")); - ui.buttonDown->setToolTip(i18n("Move effect down")); - ui.buttonDel->setIcon(KIcon("trash-empty")); - ui.buttonDel->setToolTip(i18n("Delete effect")); - ui.buttonReset->setIcon(KIcon("view-refresh")); - ui.buttonReset->setToolTip(i18n("Reset effect")); - - - ui.effectlist->setDragDropMode(QAbstractItemView::NoDragDrop);//use internal if drop is recognised right - - connect(ui.effectlist, SIGNAL(itemSelectionChanged()), this , SLOT(slotItemSelectionChanged())); - connect(ui.effectlist, SIGNAL(itemChanged(QListWidgetItem *)), this , SLOT(slotItemChanged(QListWidgetItem *))); - connect(ui.buttonNew, SIGNAL(clicked()), this, SLOT(slotNewEffect())); - connect(ui.buttonUp, SIGNAL(clicked()), this, SLOT(slotItemUp())); - connect(ui.buttonDown, SIGNAL(clicked()), this, SLOT(slotItemDown())); - connect(ui.buttonDel, SIGNAL(clicked()), this, SLOT(slotItemDel())); - connect(ui.buttonReset, SIGNAL(clicked()), this, SLOT(slotResetEffect())); - connect(this, SIGNAL(transferParamDesc(const QDomElement&, int , int)), effectedit , SLOT(transferParamDesc(const QDomElement&, int , int))); - connect(effectedit, SIGNAL(parameterChanged(const QDomElement&, const QDomElement&)), this , SLOT(slotUpdateEffectParams(const QDomElement&, const QDomElement&))); - effectLists["audio"] = audioEffectList; - effectLists["video"] = videoEffectList; - effectLists["custom"] = customEffectList; - - ui.infoBox->hide(); - setEnabled(false); +#include +#include +#include +#include + +#include +#include +#include +#include + + +EffectStackView::EffectStackView(QWidget *parent) : + QWidget(parent) +{ + m_ui.setupUi(this); + m_effectedit = new EffectStackEdit(m_ui.frame); + //m_ui.effectlist->horizontalHeader()->setVisible(false); + //m_ui.effectlist->verticalHeader()->setVisible(false); + m_clipref = NULL; + + m_ui.buttonNew->setIcon(KIcon("document-new")); + m_ui.buttonNew->setToolTip(i18n("Add new effect")); + m_ui.buttonUp->setIcon(KIcon("go-up")); + m_ui.buttonUp->setToolTip(i18n("Move effect up")); + m_ui.buttonDown->setIcon(KIcon("go-down")); + m_ui.buttonDown->setToolTip(i18n("Move effect down")); + m_ui.buttonDel->setIcon(KIcon("edit-delete")); + m_ui.buttonDel->setToolTip(i18n("Delete effect")); + m_ui.buttonSave->setIcon(KIcon("document-save")); + m_ui.buttonSave->setToolTip(i18n("Save effect")); + m_ui.buttonReset->setIcon(KIcon("view-refresh")); + m_ui.buttonReset->setToolTip(i18n("Reset effect")); + + + m_ui.effectlist->setDragDropMode(QAbstractItemView::NoDragDrop);//use internal if drop is recognised right + + connect(m_ui.effectlist, SIGNAL(itemSelectionChanged()), this , SLOT(slotItemSelectionChanged())); + connect(m_ui.effectlist, SIGNAL(itemChanged(QListWidgetItem *)), this , SLOT(slotItemChanged(QListWidgetItem *))); + connect(m_ui.buttonUp, SIGNAL(clicked()), this, SLOT(slotItemUp())); + connect(m_ui.buttonDown, SIGNAL(clicked()), this, SLOT(slotItemDown())); + connect(m_ui.buttonDel, SIGNAL(clicked()), this, SLOT(slotItemDel())); + connect(m_ui.buttonSave, SIGNAL(clicked()), this, SLOT(slotSaveEffect())); + connect(m_ui.buttonReset, SIGNAL(clicked()), this, SLOT(slotResetEffect())); + connect(this, SIGNAL(transferParamDesc(const QDomElement&, int , int)), m_effectedit , SLOT(transferParamDesc(const QDomElement&, int , int))); + connect(m_effectedit, SIGNAL(parameterChanged(const QDomElement&, const QDomElement&)), this , SLOT(slotUpdateEffectParams(const QDomElement&, const QDomElement&))); + m_effectLists["audio"] = &MainWindow::audioEffects; + m_effectLists["video"] = &MainWindow::videoEffects; + m_effectLists["custom"] = &MainWindow::customEffects; + m_ui.splitter->setStretchFactor(1, 10); + m_ui.splitter->setStretchFactor(0, 1); setEnabled(false); +} +void EffectStackView::setMenu(QMenu *menu) +{ + m_ui.buttonNew->setMenu(menu); } -void EffectStackView::slotUpdateEffectParams(const QDomElement& old, const QDomElement& e) { - if (clipref) - emit updateClipEffect(clipref, old, e); +void EffectStackView::updateProjectFormat(MltVideoProfile profile, Timecode t) +{ + m_effectedit->updateProjectFormat(profile, t); } -void EffectStackView::slotClipItemSelected(ClipItem* c) { - clipref = c; - if (clipref == NULL) { +void EffectStackView::slotSaveEffect() +{ + QString name = QInputDialog::getText(this, i18n("Save Effect"), i18n("Name for saved effect: ")); + if (name.isEmpty()) return; + QString path = KStandardDirs::locateLocal("appdata", "effects/", true); + path = path + name + ".xml"; + if (QFile::exists(path)) if (KMessageBox::questionYesNo(this, i18n("File already exists.\nDo you want to overwrite it?")) == KMessageBox::No) return; + + int i = m_ui.effectlist->currentRow(); + QDomDocument doc; + QDomElement effect = m_clipref->effectAt(i).cloneNode().toElement(); + doc.appendChild(doc.importNode(effect, true)); + effect = doc.firstChild().toElement(); + effect.removeAttribute("kdenlive_ix"); + effect.setAttribute("id", name); + effect.setAttribute("type", "custom"); + QDomElement effectname = effect.firstChildElement("name"); + effect.removeChild(effectname); + effectname = doc.createElement("name"); + QDomText nametext = doc.createTextNode(name); + effectname.appendChild(nametext); + effect.insertBefore(effectname, QDomNode()); + QDomElement effectprops = effect.firstChildElement("properties"); + effectprops.setAttribute("id", name); + effectprops.setAttribute("type", "custom"); + + + QFile file(path); + if (file.open(QFile::WriteOnly | QFile::Truncate)) { + QTextStream out(&file); + out << doc.toString(); + } + file.close(); + emit reloadEffects(); +} + +void EffectStackView::slotUpdateEffectParams(const QDomElement& old, const QDomElement& e) +{ + if (m_clipref) + emit updateClipEffect(m_clipref, old, e, m_ui.effectlist->currentRow()); +} + +void EffectStackView::slotClipItemSelected(ClipItem* c, int ix) +{ + if (c && c == m_clipref) { + if (ix == -1) ix = m_ui.effectlist->currentRow(); + } else { + m_clipref = c; + if (c) ix = c->selectedEffectIndex(); + else ix = 0; + } + if (m_clipref == NULL) { + m_ui.effectlist->clear(); + m_effectedit->transferParamDesc(QDomElement(), 0, 0); setEnabled(false); return; } setEnabled(true); - setupListView(); - + setupListView(ix); } -void EffectStackView::slotItemChanged(QListWidgetItem *item) { +void EffectStackView::slotItemChanged(QListWidgetItem *item) +{ bool disable = true; if (item->checkState() == Qt::Checked) disable = false; - ui.buttonReset->setEnabled(!disable); - int activeRow = ui.effectlist->currentRow(); + m_ui.buttonReset->setEnabled(!disable); + int activeRow = m_ui.effectlist->currentRow(); if (activeRow >= 0) { - emit changeEffectState(clipref, clipref->effectAt(activeRow), disable); + emit changeEffectState(m_clipref, activeRow, disable); } } -void EffectStackView::setupListView() { +void EffectStackView::setupListView(int ix) +{ + m_ui.effectlist->clear(); + + // Issue 238: Add icons for effect type in effectstack. + KIcon videoIcon("kdenlive-show-video"); + KIcon audioIcon("kdenlive-show-audio"); + QListWidgetItem* item; + + for (int i = 0; i < m_clipref->effectsCount(); i++) { + QDomElement d = m_clipref->effectAt(i); - ui.effectlist->clear(); - for (int i = 0;i < clipref->effectsCount();i++) { - QDomElement d = clipref->effectAt(i); QDomNode namenode = d.elementsByTagName("name").item(0); if (!namenode.isNull()) { - QListWidgetItem* item = new QListWidgetItem(namenode.toElement().text(), ui.effectlist); + // Issue 238: Add icons for effect type in effectstack. + // Logic more or less copied from initeffects.cpp + QString type = d.attribute("type", QString()); + if ("audio" == type) { + item = new QListWidgetItem(audioIcon, i18n(namenode.toElement().text().toUtf8().data()), m_ui.effectlist); + } else if ("custom" == type) { + item = new QListWidgetItem(i18n(namenode.toElement().text().toUtf8().data()), m_ui.effectlist); + } else { + item = new QListWidgetItem(videoIcon, i18n(namenode.toElement().text().toUtf8().data()), m_ui.effectlist); + } item->setFlags(Qt::ItemIsSelectable | Qt::ItemIsDragEnabled | Qt::ItemIsUserCheckable | Qt::ItemIsEnabled); if (d.attribute("disabled") == "1") item->setCheckState(Qt::Unchecked); else item->setCheckState(Qt::Checked); } } - if (clipref->effectsCount() == 0) - emit transferParamDesc(QDomElement(), 0, 100); - ui.effectlist->setCurrentRow(0); - + if (m_clipref->effectsCount() == 0) { + emit transferParamDesc(QDomElement(), 0, 0); + m_ui.buttonDel->setEnabled(false); + m_ui.buttonSave->setEnabled(false); + m_ui.buttonReset->setEnabled(false); + m_ui.buttonUp->setEnabled(false); + m_ui.buttonDown->setEnabled(false); + } else { + if (ix < 0) ix = 0; + if (ix > m_ui.effectlist->count() - 1) ix = m_ui.effectlist->count() - 1; + m_ui.effectlist->setCurrentRow(ix); + m_ui.buttonDel->setEnabled(true); + m_ui.buttonSave->setEnabled(true); + m_ui.buttonReset->setEnabled(true); + m_ui.buttonUp->setEnabled(ix > 0); + m_ui.buttonDown->setEnabled(ix < m_clipref->effectsCount() - 1); + } } -void EffectStackView::slotItemSelectionChanged() { - bool hasItem = ui.effectlist->currentItem(); - int activeRow = ui.effectlist->currentRow(); - bool isChecked = ui.effectlist->currentItem()->checkState() == Qt::Checked; - if (hasItem && ui.effectlist->currentItem()->isSelected()) { - emit transferParamDesc(clipref->effectAt(activeRow), 0, 100);//minx max frame +void EffectStackView::slotItemSelectionChanged() +{ + bool hasItem = m_ui.effectlist->currentItem(); + int activeRow = m_ui.effectlist->currentRow(); + bool isChecked = false; + if (hasItem && m_ui.effectlist->currentItem()->checkState() == Qt::Checked) isChecked = true; + if (hasItem && m_ui.effectlist->currentItem()->isSelected()) { + emit transferParamDesc(m_clipref->effectAt(activeRow), m_clipref->cropStart().frames(KdenliveSettings::project_fps()), m_clipref->cropDuration().frames(KdenliveSettings::project_fps()));//minx max frame } - ui.buttonDel->setEnabled(hasItem); - ui.buttonReset->setEnabled(hasItem && isChecked); - ui.buttonUp->setEnabled(activeRow > 0); - ui.buttonDown->setEnabled((activeRow < ui.effectlist->count() - 1) && hasItem); + if (m_clipref) m_clipref->setSelectedEffect(activeRow); + m_ui.buttonDel->setEnabled(hasItem); + m_ui.buttonSave->setEnabled(hasItem); + m_ui.buttonReset->setEnabled(hasItem && isChecked); + m_ui.buttonUp->setEnabled(activeRow > 0); + m_ui.buttonDown->setEnabled((activeRow < m_ui.effectlist->count() - 1) && hasItem); } -void EffectStackView::slotItemUp() { - int activeRow = ui.effectlist->currentRow(); - if (activeRow > 0) { - QDomElement act = clipref->effectAt(activeRow).cloneNode().toElement(); - QDomElement before = clipref->effectAt(activeRow - 1).cloneNode().toElement(); - clipref->setEffectAt(activeRow - 1, act); - clipref->setEffectAt(activeRow, before); - } - QListWidgetItem *item = ui.effectlist->takeItem(activeRow); - ui.effectlist->insertItem(activeRow - 1, item); - ui.effectlist->setCurrentItem(item); - emit refreshEffectStack(clipref); +void EffectStackView::slotItemUp() +{ + int activeRow = m_ui.effectlist->currentRow(); + if (activeRow <= 0) return; + emit changeEffectPosition(m_clipref, activeRow + 1, activeRow); } -void EffectStackView::slotItemDown() { - int activeRow = ui.effectlist->currentRow(); - if (activeRow < ui.effectlist->count() - 1) { - QDomElement act = clipref->effectAt(activeRow).cloneNode().toElement(); - QDomElement after = clipref->effectAt(activeRow + 1).cloneNode().toElement(); - clipref->setEffectAt(activeRow + 1, act); - clipref->setEffectAt(activeRow, after); - } - QListWidgetItem *item = ui.effectlist->takeItem(activeRow); - ui.effectlist->insertItem(activeRow + 1, item); - ui.effectlist->setCurrentItem(item); - emit refreshEffectStack(clipref); +void EffectStackView::slotItemDown() +{ + int activeRow = m_ui.effectlist->currentRow(); + if (activeRow >= m_ui.effectlist->count() - 1) return; + emit changeEffectPosition(m_clipref, activeRow + 1, activeRow + 2); } -void EffectStackView::slotItemDel() { - int activeRow = ui.effectlist->currentRow(); +void EffectStackView::slotItemDel() +{ + int activeRow = m_ui.effectlist->currentRow(); if (activeRow >= 0) { - emit removeEffect(clipref, clipref->effectAt(activeRow)); + emit removeEffect(m_clipref, m_clipref->effectAt(activeRow)); } } -void EffectStackView::slotResetEffect() { - int activeRow = ui.effectlist->currentRow(); - QDomElement old = clipref->effectAt(activeRow).cloneNode().toElement(); +void EffectStackView::slotResetEffect() +{ + int activeRow = m_ui.effectlist->currentRow(); + if (activeRow < 0) return; + QDomElement old = m_clipref->effectAt(activeRow).cloneNode().toElement(); QDomElement dom; - QString effectName = ui.effectlist->currentItem()->text(); - foreach(QString type, effectLists.keys()) { - EffectsList *list = effectLists[type]; + QString effectName = m_ui.effectlist->currentItem()->text(); + foreach(const QString &type, m_effectLists.keys()) { + EffectsList *list = m_effectLists[type]; if (list->effectNames().contains(effectName)) { dom = list->getEffectByName(effectName); break; @@ -173,50 +261,27 @@ void EffectStackView::slotResetEffect() { } if (!dom.isNull()) { dom.setAttribute("kdenlive_ix", old.attribute("kdenlive_ix")); - emit transferParamDesc(dom, 0, 100);//minx max frame - emit updateClipEffect(clipref, old, dom); + emit transferParamDesc(dom, m_clipref->cropStart().frames(KdenliveSettings::project_fps()), m_clipref->cropDuration().frames(KdenliveSettings::project_fps()));//minx max frame + emit updateClipEffect(m_clipref, old, dom, activeRow); } } -void EffectStackView::slotNewEffect() { - - - QMenu *displayMenu = new QMenu(this); - displayMenu->setTitle("Filters"); - foreach(QString type, effectLists.keys()) { - QAction *a = new QAction(type, displayMenu); - EffectsList *list = effectLists[type]; - - QMenu *parts = new QMenu(type, displayMenu); - parts->setTitle(type); - foreach(QString name, list->effectNames()) { - QAction *entry = new QAction(name, parts); - entry->setData(name); - entry->setToolTip(list->getInfo(name)); - entry->setStatusTip(list->getInfo(name)); - parts->addAction(entry); - //QAction - } - displayMenu->addMenu(parts); - - } - - QAction *result = displayMenu->exec(mapToGlobal(ui.buttonNew->pos() + ui.buttonNew->rect().bottomRight())); - if (result) { - //TODO effects.append(result->data().toString()); - foreach(EffectsList* e, effectLists.values()) { - QDomElement dom = e->getEffectByName(result->data().toString()); - if (clipref) - clipref->addEffect(dom); - slotClipItemSelected(clipref); - } - - setupListView(); - //kDebug()<< result->data(); - } - delete displayMenu; +void EffectStackView::raiseWindow(QWidget* dock) +{ + if (m_clipref && dock) + dock->raise(); +} +void EffectStackView::clear() +{ + m_ui.effectlist->clear(); + m_ui.buttonDel->setEnabled(false); + m_ui.buttonSave->setEnabled(false); + m_ui.buttonReset->setEnabled(false); + m_ui.buttonUp->setEnabled(false); + m_ui.buttonDown->setEnabled(false); + m_effectedit->transferParamDesc(QDomElement(), 0, 0); } #include "effectstackview.moc"