]> git.sesse.net Git - kdenlive/blobdiff - src/keyframeedit.cpp
Fix seek to pos in composite transition:
[kdenlive] / src / keyframeedit.cpp
index 26eaf86cf648bc38d89d8fdadfa45eb0a025557d..8f14b44f18c0a74af86b215dc4d6c50dbf534f63 100644 (file)
 #include "kdenlivesettings.h"
 
 #include <KDebug>
+#include <KGlobalSettings>
 
+#include <QHeaderView>
 
-KeyframeEdit::KeyframeEdit(Timecode tc, QWidget* parent) :
+
+KeyframeEdit::KeyframeEdit(QDomElement e, int minFrame, int maxFrame, int minVal, int maxVal, Timecode tc, int active_keyframe, QWidget* parent) :
         QWidget(parent),
-        m_timecode(tc)
+        m_min(minFrame),
+        m_max(maxFrame),
+        m_minVal(minVal),
+        m_maxVal(maxVal),
+        m_timecode(tc),
+        m_previousPos(0),
+        m_active_keyframe(active_keyframe)
 {
-    m_ui.setupUi(this);
-    m_ui.keyframe_list->setHeaderLabels(QStringList() << i18n("Position") << i18n("Value"));
-    m_ui.button_add->setIcon(KIcon("document-new"));
-    m_ui.button_delete->setIcon(KIcon("edit-delete"));
-    setEnabled(false);
+    kDebug() << " / / / /MODIFIED KFR: " << m_active_keyframe;
+    setupUi(this);
+    m_params.append(e.cloneNode().toElement());
+    keyframe_list->setFont(KGlobalSettings::generalFont());
+    keyframe_seek->setChecked(KdenliveSettings::keyframeseek());
+    connect(keyframe_seek, SIGNAL(stateChanged(int)), this, SLOT(slotSetSeeking(int)));
+
+    button_add->setIcon(KIcon("list-add"));
+    button_add->setToolTip(i18n("Add keyframe"));
+    button_delete->setIcon(KIcon("list-remove"));
+    button_delete->setToolTip(i18n("Delete keyframe"));
+    connect(keyframe_list, SIGNAL(itemSelectionChanged()), this, SLOT(slotAdjustKeyframeInfo()));
+    connect(keyframe_list, SIGNAL(cellChanged(int, int)), this, SLOT(slotGenerateParams(int, int)));
+    setupParam();
+
+    keyframe_list->resizeRowsToContents();
+    keyframe_list->resizeColumnsToContents();
+    //keyframe_list->horizontalHeader()->resizeSections(QHeaderView::ResizeToContents);
+    connect(button_delete, SIGNAL(clicked()), this, SLOT(slotDeleteKeyframe()));
+    connect(button_add, SIGNAL(clicked()), this, SLOT(slotAddKeyframe()));
+    //connect(keyframe_list, SIGNAL(itemDoubleClicked(QTreeWidgetItem *, int)), this, SLOT(slotSaveCurrentParam(QTreeWidgetItem *, int)));
+    connect(keyframe_pos, SIGNAL(valueChanged(int)), this, SLOT(slotAdjustKeyframePos(int)));
+    //connect(keyframe_val, SIGNAL(valueChanged(int)), this, SLOT(slotAdjustKeyframeValue(int)));
+    keyframe_pos->setPageStep(1);
+    if (!keyframe_list->currentItem()) {
+        keyframe_list->setCurrentCell(0, 0);
+        keyframe_list->selectRow(0);
+    }
+    /*m_delegate = new KeyItemDelegate(minVal, maxVal);
+    keyframe_list->setItemDelegate(m_delegate);*/
 }
 
-void KeyframeEdit::setupParam(int maxFrame, int minValue, int maxValue, QString keyframes)
+KeyframeEdit::~KeyframeEdit()
 {
-    m_maxFrame = maxFrame;
-    m_min = minValue;
-    m_max = maxValue;
-    m_ui.keyframe_list->clear();
-    QStringList frames = keyframes.split(";");
+    keyframe_list->blockSignals(true);
+    keyframe_list->clear();
+    QLayoutItem *child;
+    while ((child = m_slidersLayout->takeAt(0)) != 0) {
+        QWidget *wid = child->widget();
+        delete child;
+        if (wid) delete wid;
+    }
+    //delete m_delegate;
+}
+
+void KeyframeEdit::addParameter(QDomElement e)
+{
+    keyframe_list->blockSignals(true);
+    m_params.append(e.cloneNode().toElement());
+    QDomNode na = e.firstChildElement("name");
+    QString paramName = i18n(na.toElement().text().toUtf8().data());
+    int columnId = keyframe_list->columnCount();
+    keyframe_list->insertColumn(columnId);
+    keyframe_list->setHorizontalHeaderItem(columnId, new QTableWidgetItem(paramName));
+    m_slidersLayout->addWidget(new QLabel(paramName), columnId, 0);
+    QSlider *sl = new QSlider(Qt::Horizontal, this);
+    sl->setRange(m_params.at(columnId).attribute("min").toInt(), m_params.at(columnId).attribute("max").toInt());
+    connect(sl, SIGNAL(valueChanged(int)), this, SLOT(slotAdjustKeyframeValue(int)));
+    m_slidersLayout->addWidget(sl, columnId, 1);
+
+    QStringList frames = e.attribute("keyframes").split(";", QString::SkipEmptyParts);
     for (int i = 0; i < frames.count(); i++) {
-        QString framePos = m_timecode.getTimecodeFromFrames(frames.at(i).section(':', 0, 0).toInt());
-        m_ui.keyframe_list->addTopLevelItem(new QTreeWidgetItem(QStringList() << framePos << frames.at(i).section(':', 1, 1)));
+        int frame = frames.at(i).section(':', 0, 0).toInt();
+        bool found = false;
+        int j;
+        for (j = 0; j < keyframe_list->rowCount(); j++) {
+            int currentPos;
+            if (KdenliveSettings::frametimecode()) currentPos = keyframe_list->verticalHeaderItem(j)->text().toInt();
+            else currentPos = m_timecode.getFrameCount(keyframe_list->verticalHeaderItem(j)->text());
+            if (frame == currentPos) {
+                keyframe_list->setItem(j, columnId, new QTableWidgetItem(frames.at(i).section(':', 1, 1)));
+                found = true;
+                break;
+            } else if (currentPos > frame) {
+                break;
+            }
+        }
+        if (!found) {
+            keyframe_list->insertRow(j);
+            QString currentPos;
+            if (KdenliveSettings::frametimecode()) currentPos = QString::number(frame);
+            else currentPos = m_timecode.getTimecodeFromFrames(frame);
+            keyframe_list->setVerticalHeaderItem(j, new QTableWidgetItem(currentPos));
+            keyframe_list->setItem(j, columnId, new QTableWidgetItem(frames.at(i).section(':', 1, 1)));
+            keyframe_list->resizeRowToContents(j);
+        }
+    }
+    keyframe_list->resizeColumnsToContents();
+    keyframe_list->blockSignals(false);
+    slotAdjustKeyframeInfo(false);
+}
+
+void KeyframeEdit::setupParam()
+{
+    keyframe_list->blockSignals(true);
+    keyframe_list->clear();
+    int col = keyframe_list->columnCount();
+    QDomNode na = m_params.at(0).firstChildElement("name");
+    QString paramName = i18n(na.toElement().text().toUtf8().data());
+    kDebug() << " INSERT COL: " << col << ", " << paramName;
+    keyframe_list->insertColumn(col);
+    keyframe_list->setHorizontalHeaderItem(col, new QTableWidgetItem(paramName));
+    m_slidersLayout = new QGridLayout;
+    m_slidersLayout->addWidget(new QLabel(paramName), 0, 0);
+    QSlider *sl = new QSlider(Qt::Horizontal, this);
+    sl->setRange(m_params.at(0).attribute("min").toInt(), m_params.at(0).attribute("max").toInt());
+    connect(sl, SIGNAL(valueChanged(int)), this, SLOT(slotAdjustKeyframeValue(int)));
+    m_slidersLayout->addWidget(sl, 0, 1);
+    param_sliders->setLayout(m_slidersLayout);
+    keyframe_list->setSelectionBehavior(QAbstractItemView::SelectRows);
+    keyframe_list->setSelectionMode(QAbstractItemView::SingleSelection);
+
+    QStringList frames = m_params.at(0).attribute("keyframes").split(";", QString::SkipEmptyParts);
+    for (int i = 0; i < frames.count(); i++) {
+        keyframe_list->insertRow(i);
+        int currentpos = frames.at(i).section(':', 0, 0).toInt();
+        QString framePos;
+        if (KdenliveSettings::frametimecode()) framePos  = QString::number(currentpos);
+        else framePos = m_timecode.getTimecodeFromFrames(currentpos);
+        keyframe_list->setVerticalHeaderItem(i, new QTableWidgetItem(framePos));
+        keyframe_list->setItem(i, col, new QTableWidgetItem(frames.at(i).section(':', 1, 1)));
+        if ((m_active_keyframe > -1) && (m_active_keyframe == currentpos)) {
+            keyframe_list->setCurrentCell(i, 0);
+            keyframe_list->selectRow(i);
+        }
+        //item->setFlags(Qt::ItemIsSelectable | Qt::ItemIsEditable | Qt::ItemIsEnabled);
+    }
+    /*QTreeWidgetItem *first = keyframe_list->topLevelItem(0);
+    if (first) keyframe_list->setCurrentItem(first);*/
+    keyframe_list->blockSignals(false);
+    //keyframe_list->setCurrentCell(0, 0);
+    slotAdjustKeyframeInfo(false);
+    button_delete->setEnabled(keyframe_list->rowCount() > 1);
+}
+
+void KeyframeEdit::slotDeleteKeyframe()
+{
+    if (keyframe_list->rowCount() < 2) return;
+    int col = keyframe_list->currentColumn();
+    int row = keyframe_list->currentRow();
+    keyframe_list->removeRow(keyframe_list->currentRow());
+    row = qMin(row, keyframe_list->rowCount() - 1);
+    keyframe_list->setCurrentCell(row, col);
+    keyframe_list->selectRow(row);
+    generateAllParams();
+    button_delete->setEnabled(keyframe_list->rowCount() > 1);
+}
+
+void KeyframeEdit::slotAddKeyframe()
+{
+    keyframe_list->blockSignals(true);
+    QTableWidgetItem *item = keyframe_list->currentItem();
+    int row = keyframe_list->currentRow();
+    int col = keyframe_list->currentColumn();
+    int newrow = row;
+    int pos1;
+    if (KdenliveSettings::frametimecode()) pos1 = keyframe_list->verticalHeaderItem(row)->text().toInt();
+    else pos1 = m_timecode.getFrameCount(keyframe_list->verticalHeaderItem(row)->text());
+
+    int result;
+    kDebug() << "// ADD KF: " << row << ", MAX: " << keyframe_list->rowCount() << ", POS: " << pos1;
+    if (row < (keyframe_list->rowCount() - 1)) {
+        int pos2;
+        if (KdenliveSettings::frametimecode()) pos2 = keyframe_list->verticalHeaderItem(row + 1)->text().toInt();
+        else pos2 = m_timecode.getFrameCount(keyframe_list->verticalHeaderItem(row + 1)->text());
+        result = pos1 + (pos2 - pos1) / 2;
+        newrow++;
+    } else if (row == 0) {
+        if (pos1 == m_min) {
+            result = m_max - 1;
+            newrow++;
+        } else {
+            result = m_min;
+        }
+    } else {
+        int pos2;
+        if (KdenliveSettings::frametimecode()) pos2 = keyframe_list->verticalHeaderItem(row - 1)->text().toInt();
+        else pos2 = m_timecode.getFrameCount(keyframe_list->verticalHeaderItem(row - 1)->text());
+        result = pos2 + (pos1 - pos2) / 2;
+    }
+
+    keyframe_list->insertRow(newrow);
+    QString currentPos;
+    if (KdenliveSettings::frametimecode()) currentPos = QString::number(result);
+    else currentPos = m_timecode.getTimecodeFromFrames(result);
+    keyframe_list->setVerticalHeaderItem(newrow, new QTableWidgetItem(currentPos));
+    for (int i = 0; i < keyframe_list->columnCount(); i++) {
+        keyframe_list->setItem(newrow, i, new QTableWidgetItem(keyframe_list->item(item->row(), i)->text()));
     }
+    keyframe_list->resizeRowsToContents();
+    //keyframe_list->resizeRowToContents(newrow);
+    slotAdjustKeyframeInfo();
+    keyframe_list->blockSignals(false);
+    generateAllParams();
+    button_delete->setEnabled(keyframe_list->rowCount() > 1);
+    keyframe_list->setCurrentCell(newrow, col);
+    keyframe_list->selectRow(newrow);
+    //slotGenerateParams(newrow, 0);
 }
 
+void KeyframeEdit::slotGenerateParams(int row, int column)
+{
+    if (column == -1) {
+        // position of keyframe changed
+        QTableWidgetItem *item = keyframe_list->item(row, 0);
+        if (item == NULL) return;
+        QString val = keyframe_list->verticalHeaderItem(row)->text();
+        int pos;
+        if (KdenliveSettings::frametimecode()) pos = val.toInt();
+        else pos = m_timecode.getFrameCount(val);
+
+        if (pos <= m_min) {
+            pos = m_min;
+            if (KdenliveSettings::frametimecode()) val = QString::number(pos);
+            else val = m_timecode.getTimecodeFromFrames(pos);
+        }
+        if (pos > m_max) {
+            pos = m_max;
+            if (KdenliveSettings::frametimecode()) val = QString::number(pos);
+            else val = m_timecode.getTimecodeFromFrames(pos);
+        }
+        if (val != keyframe_list->verticalHeaderItem(row)->text()) keyframe_list->verticalHeaderItem(row)->setText(val);
+
+        for (int col = 0; col < keyframe_list->horizontalHeader()->count(); col++) {
+            item = keyframe_list->item(row, col);
+            int v = item->text().toInt();
+            if (v >= m_params.at(col).attribute("max").toInt()) item->setText(m_params.at(col).attribute("max"));
+            if (v <= m_params.at(col).attribute("min").toInt()) item->setText(m_params.at(col).attribute("min"));
+            QString keyframes;
+            for (int i = 0; i < keyframe_list->rowCount(); i++) {
+                if (keyframe_list->item(i, col)) {
+                    int pos3;
+                    if (KdenliveSettings::frametimecode()) pos3 = keyframe_list->verticalHeaderItem(i)->text().toInt();
+                    else pos3 = m_timecode.getFrameCount(keyframe_list->verticalHeaderItem(i)->text());
+                    keyframes.append(QString::number(pos3) + ':' + keyframe_list->item(i, col)->text() + ';');
+                }
+            }
+            m_params[col].setAttribute("keyframes", keyframes);
+        }
+
+        emit parameterChanged();
+        return;
+
+    }
+    QTableWidgetItem *item = keyframe_list->item(row, column);
+    if (item == NULL) return;
+    QString val = keyframe_list->verticalHeaderItem(row)->text();
+    ;
+    int pos;
+    if (KdenliveSettings::frametimecode()) pos = val.toInt();
+    else pos = m_timecode.getFrameCount(val);
+
+    if (pos <= m_min) {
+        pos = m_min;
+        if (KdenliveSettings::frametimecode()) val = QString::number(pos);
+        else val = m_timecode.getTimecodeFromFrames(pos);
+    }
+    if (pos > m_max) {
+        pos = m_max;
+        if (KdenliveSettings::frametimecode()) val = QString::number(pos);
+        else val = m_timecode.getTimecodeFromFrames(pos);
+    }
+    /*QList<QTreeWidgetItem *> duplicates = keyframe_list->findItems(val, Qt::MatchExactly, 0);
+    duplicates.removeAll(item);
+    if (!duplicates.isEmpty()) {
+        // Trying to insert a keyframe at existing value, revert it
+        val = m_timecode.getTimecodeFromFrames(m_previousPos);
+    }*/
+    if (val != keyframe_list->verticalHeaderItem(row)->text()) keyframe_list->verticalHeaderItem(row)->setText(val);
+
+    int v = item->text().toInt();
+    if (v >= m_params.at(column).attribute("max").toInt()) item->setText(m_params.at(column).attribute("max"));
+    if (v <= m_params.at(column).attribute("min").toInt()) item->setText(m_params.at(column).attribute("min"));
+    slotAdjustKeyframeInfo(false);
+
+    QString keyframes;
+    for (int i = 0; i < keyframe_list->rowCount(); i++) {
+        if (keyframe_list->item(i, column)) {
+            int pos3;
+            if (KdenliveSettings::frametimecode()) pos3 = keyframe_list->verticalHeaderItem(i)->text().toInt();
+            else pos3 = m_timecode.getFrameCount(keyframe_list->verticalHeaderItem(i)->text());
+            keyframes.append(QString::number(pos3) + ':' + keyframe_list->item(i, column)->text() + ';');
+        }
+    }
+    m_params[column].setAttribute("keyframes", keyframes);
+    emit parameterChanged();
+}
+
+void KeyframeEdit::generateAllParams()
+{
+    for (int col = 0; col < keyframe_list->columnCount(); col++) {
+        QString keyframes;
+        for (int i = 0; i < keyframe_list->rowCount(); i++) {
+            if (keyframe_list->item(i, col)) {
+                int pos3;
+                if (KdenliveSettings::frametimecode()) pos3 = keyframe_list->verticalHeaderItem(i)->text().toInt();
+                else pos3 = m_timecode.getFrameCount(keyframe_list->verticalHeaderItem(i)->text());
+                keyframes.append(QString::number(pos3) + ':' + keyframe_list->item(i, col)->text() + ';');
+            }
+        }
+        m_params[col].setAttribute("keyframes", keyframes);
+    }
+    emit parameterChanged();
+}
+
+const QString KeyframeEdit::getValue(const QString &name)
+{
+    for (int col = 0; col < keyframe_list->columnCount(); col++) {
+        QDomNode na = m_params.at(col).firstChildElement("name");
+        QString paramName = i18n(na.toElement().text().toUtf8().data());
+        kDebug() << paramName << " == " << name;
+        if (paramName == name) return m_params.at(col).attribute("keyframes");
+    }
+    return QString();
+}
+
+void KeyframeEdit::slotAdjustKeyframeInfo(bool seek)
+{
+    QTableWidgetItem *item = keyframe_list->currentItem();
+    if (!item) return;
+    int min = m_min;
+    int max = m_max;
+    QTableWidgetItem *above = keyframe_list->item(item->row() - 1, item->column());
+    QTableWidgetItem *below = keyframe_list->item(item->row() + 1, item->column());
+    if (above) {
+        if (KdenliveSettings::frametimecode()) min = keyframe_list->verticalHeaderItem(above->row())->text().toInt();
+        else min = m_timecode.getFrameCount(keyframe_list->verticalHeaderItem(above->row())->text());
+        min++;
+    }
+    if (below) {
+        if (KdenliveSettings::frametimecode()) max = keyframe_list->verticalHeaderItem(below->row())->text().toInt();
+        else max = m_timecode.getFrameCount(keyframe_list->verticalHeaderItem(below->row())->text());
+        max--;
+    }
+    keyframe_pos->blockSignals(true);
+    keyframe_pos->setRange(min, max);
+    int pos;
+    if (KdenliveSettings::frametimecode()) pos = keyframe_list->verticalHeaderItem(item->row())->text().toInt();
+    else pos = m_timecode.getFrameCount(keyframe_list->verticalHeaderItem(item->row())->text());
+
+    keyframe_pos->setValue(pos);
+    keyframe_pos->blockSignals(false);
+    for (int col = 0; col < keyframe_list->columnCount(); col++) {
+        QSlider *sl = static_cast <QSlider*>(m_slidersLayout->itemAtPosition(col, 1)->widget());
+        if (!sl) continue;
+        sl->blockSignals(true);
+        sl->setValue(keyframe_list->item(item->row(), col)->text().toInt());
+        sl->blockSignals(false);
+    }
+    if (KdenliveSettings::keyframeseek() && seek) emit seekToPos(keyframe_pos->value() - m_min);
+}
+
+void KeyframeEdit::slotAdjustKeyframePos(int value)
+{
+    QTableWidgetItem *item = keyframe_list->currentItem();
+    QString val;
+    if (KdenliveSettings::frametimecode()) val = QString::number(value);
+    else val = m_timecode.getTimecodeFromFrames(value);
+    keyframe_list->verticalHeaderItem(item->row())->setText(val);
+    slotGenerateParams(item->row(), -1);
+    if (KdenliveSettings::keyframeseek()) emit seekToPos(value - m_min);
+}
+
+void KeyframeEdit::slotAdjustKeyframeValue(int /*value*/)
+{
+    QTableWidgetItem *item = keyframe_list->currentItem();
+    for (int col = 0; col < keyframe_list->columnCount(); col++) {
+        QSlider *sl = static_cast <QSlider*>(m_slidersLayout->itemAtPosition(col, 1)->widget());
+        if (!sl) continue;
+        int val = sl->value();
+        QTableWidgetItem *nitem = keyframe_list->item(item->row(), col);
+        if (nitem->text().toInt() != val) {
+            nitem->setText(QString::number(val));
+        }
+    }
+    //keyframe_list->item(item->row() - 1, item->column());
+
+}
+
+void KeyframeEdit::slotSetSeeking(int state)
+{
+    KdenliveSettings::setKeyframeseek(state == Qt::Checked);
+}
+
+
+/*void KeyframeEdit::slotSaveCurrentParam(QTreeWidgetItem *item, int column)
+{
+    if (item && column == 0) m_previousPos = m_timecode.getFrameCount(item->text(0));
+}*/