]> git.sesse.net Git - kdenlive/blob - src/keyframeedit.cpp
keyframe editor: when adding a keyframe with last keyframe selected and not at the...
[kdenlive] / src / keyframeedit.cpp
1 /***************************************************************************
2                           keyframedit.cpp  -  description
3                              -------------------
4     begin                : 03 Aug 2008
5     copyright            : (C) 2008 by Marco Gittler
6     email                : g.marco@freenet.de
7  ***************************************************************************/
8
9 /***************************************************************************
10  *                                                                         *
11  *   This program is free software; you can redistribute it and/or modify  *
12  *   it under the terms of the GNU General Public License as published by  *
13  *   the Free Software Foundation; either version 2 of the License, or     *
14  *   (at your option) any later version.                                   *
15  *                                                                         *
16  ***************************************************************************/
17
18 #include "keyframeedit.h"
19 #include "doubleparameterwidget.h"
20 #include "positionedit.h"
21 #include "kdenlivesettings.h"
22
23 #include <KDebug>
24 #include <KGlobalSettings>
25
26 #include <QHeaderView>
27 #include <QButtonGroup>
28 #include <QRadioButton>
29
30 KeyframeEdit::KeyframeEdit(QDomElement e, int minFrame, int maxFrame, Timecode tc, int activeKeyframe, QWidget* parent) :
31         QWidget(parent),
32         m_min(minFrame),
33         m_max(maxFrame),
34         m_timecode(tc)
35 {
36     setupUi(this);
37     if (m_max == -1) {
38         // special case: keyframe for tracks, do not allow keyframes
39         widgetTable->setHidden(true);
40     }
41     keyframe_list->setFont(KGlobalSettings::generalFont());
42     buttonSeek->setChecked(KdenliveSettings::keyframeseek());
43     connect(buttonSeek, SIGNAL(toggled(bool)), this, SLOT(slotSetSeeking(bool)));
44
45     buttonKeyframes->setIcon(KIcon("chronometer"));
46     button_add->setIcon(KIcon("list-add"));
47     button_add->setToolTip(i18n("Add keyframe"));
48     button_delete->setIcon(KIcon("list-remove"));
49     button_delete->setToolTip(i18n("Delete keyframe"));
50     buttonResetKeyframe->setIcon(KIcon("edit-undo"));
51     buttonSeek->setIcon(KIcon("insert-link"));
52     connect(keyframe_list, SIGNAL(itemSelectionChanged()), this, SLOT(slotAdjustKeyframeInfo()));
53     connect(keyframe_list, SIGNAL(cellChanged(int, int)), this, SLOT(slotGenerateParams(int, int)));
54
55     m_position = new PositionEdit(i18n("Position"), 0, 0, 1, tc, widgetTable);
56     ((QGridLayout*)widgetTable->layout())->addWidget(m_position, 3, 0, 1, -1);
57
58     m_showButtons = new QButtonGroup(this);
59     m_slidersLayout = new QGridLayout(param_sliders);
60     keyframe_list->setSelectionBehavior(QAbstractItemView::SelectRows);
61     keyframe_list->setSelectionMode(QAbstractItemView::SingleSelection);
62     addParameter(e, activeKeyframe);
63     keyframe_list->resizeRowsToContents();
64
65     //keyframe_list->horizontalHeader()->resizeSections(QHeaderView::ResizeToContents);
66     connect(button_delete, SIGNAL(clicked()), this, SLOT(slotDeleteKeyframe()));
67     connect(button_add, SIGNAL(clicked()), this, SLOT(slotAddKeyframe()));
68     connect(buttonKeyframes, SIGNAL(clicked()), this, SLOT(slotKeyframeMode()));
69     connect(buttonResetKeyframe, SIGNAL(clicked()), this, SLOT(slotResetKeyframe()));
70     connect(m_position, SIGNAL(parameterChanged(int)), this, SLOT(slotAdjustKeyframePos(int)));
71     connect(m_showButtons, SIGNAL(buttonClicked(int)), this, SLOT(slotUpdateVisibleParameter(int)));
72
73     //connect(keyframe_list, SIGNAL(itemDoubleClicked(QTreeWidgetItem *, int)), this, SLOT(slotSaveCurrentParam(QTreeWidgetItem *, int)));
74
75     if (!keyframe_list->currentItem()) {
76         keyframe_list->setCurrentCell(0, 0);
77         keyframe_list->selectRow(0);
78     }
79     // ensure the keyframe list shows at least 3 lines
80     keyframe_list->setMinimumHeight(QFontInfo(keyframe_list->font()).pixelSize() * 9);
81
82     // Do not show keyframe table if only one keyframe exists at the beginning
83     if (keyframe_list->rowCount() < 2 && getPos(0) == m_min && m_max != -1)
84         widgetTable->setHidden(true);
85     else
86         buttonKeyframes->setHidden(true);
87 }
88
89 KeyframeEdit::~KeyframeEdit()
90 {
91     keyframe_list->blockSignals(true);
92     keyframe_list->clear();
93     QLayoutItem *child;
94     while ((child = m_slidersLayout->takeAt(0)) != 0) {
95         QWidget *wid = child->widget();
96         delete child;
97         if (wid)
98             delete wid;
99     }
100 }
101
102 void KeyframeEdit::addParameter(QDomElement e, int activeKeyframe)
103 {
104     keyframe_list->blockSignals(true);
105     m_params.append(e.cloneNode().toElement());
106
107     QDomElement na = e.firstChildElement("name");
108     QString paramName = i18n(na.text().toUtf8().data());
109     QDomElement commentElem = e.firstChildElement("comment");
110     QString comment;
111     if (!commentElem.isNull())
112         comment = i18n(commentElem.text().toUtf8().data());
113
114     int columnId = keyframe_list->columnCount();
115     keyframe_list->insertColumn(columnId);
116     keyframe_list->setHorizontalHeaderItem(columnId, new QTableWidgetItem(paramName));
117
118     DoubleParameterWidget *doubleparam = new DoubleParameterWidget(paramName, 0,
119             m_params.at(columnId).attribute("min").toInt(), m_params.at(columnId).attribute("max").toInt(),
120             m_params.at(columnId).attribute("default").toInt(), comment, m_params.at(columnId).attribute("suffix"), this);
121     connect(doubleparam, SIGNAL(valueChanged(int)), this, SLOT(slotAdjustKeyframeValue(int)));
122     connect(this, SIGNAL(showComments(bool)), doubleparam, SLOT(slotShowComment(bool)));
123     m_slidersLayout->addWidget(doubleparam, columnId, 0);
124
125     QRadioButton *radio = new QRadioButton(this);
126     radio->setToolTip(i18n("Show %1 in timeline").arg(paramName));
127     m_showButtons->addButton(radio, columnId);
128     if (e.attribute("intimeline") == "1")
129         radio->setChecked(true);
130
131     QVBoxLayout *radioLayout = new QVBoxLayout;
132     radioLayout->addWidget(radio, 0, Qt::AlignTop);
133     m_slidersLayout->addLayout(radioLayout, columnId, 1);
134
135     QStringList frames = e.attribute("keyframes").split(";", QString::SkipEmptyParts);
136     for (int i = 0; i < frames.count(); i++) {
137         int frame = frames.at(i).section(':', 0, 0).toInt();
138         bool found = false;
139         int j;
140         for (j = 0; j < keyframe_list->rowCount(); j++) {
141             int currentPos = getPos(j);
142             if (frame == currentPos) {
143                 keyframe_list->setItem(j, columnId, new QTableWidgetItem(frames.at(i).section(':', 1, 1)));
144                 found = true;
145                 break;
146             } else if (currentPos > frame) {
147                 break;
148             }
149         }
150         if (!found) {
151             keyframe_list->insertRow(j);
152             keyframe_list->setVerticalHeaderItem(j, new QTableWidgetItem(getPosString(frame)));
153             keyframe_list->setItem(j, columnId, new QTableWidgetItem(frames.at(i).section(':', 1, 1)));
154             keyframe_list->resizeRowToContents(j);
155         }
156         if ((activeKeyframe > -1) && (activeKeyframe == frame)) {
157             keyframe_list->setCurrentCell(i, columnId);
158             keyframe_list->selectRow(i);
159         }
160     }
161     keyframe_list->resizeColumnsToContents();
162     keyframe_list->blockSignals(false);
163     slotAdjustKeyframeInfo(false);
164     button_delete->setEnabled(keyframe_list->rowCount() > 1);
165 }
166
167 void KeyframeEdit::slotDeleteKeyframe()
168 {
169     if (keyframe_list->rowCount() < 2)
170         return;
171     int col = keyframe_list->currentColumn();
172     int row = keyframe_list->currentRow();
173     keyframe_list->removeRow(keyframe_list->currentRow());
174     row = qMin(row, keyframe_list->rowCount() - 1);
175     keyframe_list->setCurrentCell(row, col);
176     keyframe_list->selectRow(row);
177     generateAllParams();
178
179     bool disable = keyframe_list->rowCount() < 2;
180     button_delete->setEnabled(!disable);
181     disable &= getPos(0) == m_min;
182     widgetTable->setHidden(disable);
183     buttonKeyframes->setHidden(!disable);
184 }
185
186 void KeyframeEdit::slotAddKeyframe()
187 {
188     keyframe_list->blockSignals(true);
189     QTableWidgetItem *item = keyframe_list->currentItem();
190     int row = keyframe_list->currentRow();
191     int col = keyframe_list->currentColumn();
192     int newrow = row;
193     int pos1 = getPos(row);
194
195     int result;
196     if (row < (keyframe_list->rowCount() - 1)) {
197         result = pos1 + (getPos(row + 1) - pos1) / 2;
198         newrow++;
199     } else if (row == 0) {
200         if (pos1 == m_min) {
201             result = m_max - 1;
202             newrow++;
203         } else {
204             result = m_min;
205         }
206     } else {
207         if (pos1 < m_max - 1) {
208             // last keyframe selected and it is not at end of clip -> add keyframe at the end
209             result = m_max - 1;
210             newrow++;
211         } else {
212             int pos2 = getPos(row - 1);
213             result = pos2 + (pos1 - pos2) / 2;
214         }
215     }
216
217     keyframe_list->insertRow(newrow);
218     keyframe_list->setVerticalHeaderItem(newrow, new QTableWidgetItem(getPosString(result)));
219     for (int i = 0; i < keyframe_list->columnCount(); i++)
220         keyframe_list->setItem(newrow, i, new QTableWidgetItem(keyframe_list->item(item->row(), i)->text()));
221
222     keyframe_list->resizeRowsToContents();
223     slotAdjustKeyframeInfo();
224     keyframe_list->blockSignals(false);
225     generateAllParams();
226     button_delete->setEnabled(keyframe_list->rowCount() > 1);
227     keyframe_list->setCurrentCell(newrow, col);
228     keyframe_list->selectRow(newrow);
229     //slotGenerateParams(newrow, 0);
230 }
231
232 void KeyframeEdit::slotGenerateParams(int row, int column)
233 {
234     if (column == -1) {
235         // position of keyframe changed
236         QTableWidgetItem *item = keyframe_list->item(row, 0);
237         if (item == NULL)
238             return;
239
240         int pos = getPos(row);
241         if (pos <= m_min)
242             pos = m_min;
243         if (m_max != -1 && pos > m_max)
244             pos = m_max;
245         QString val = getPosString(pos);
246         if (val != keyframe_list->verticalHeaderItem(row)->text())
247             keyframe_list->verticalHeaderItem(row)->setText(val);
248
249         for (int col = 0; col < keyframe_list->horizontalHeader()->count(); col++) {
250             item = keyframe_list->item(row, col);
251             int v = item->text().toInt();
252             if (v >= m_params.at(col).attribute("max").toInt())
253                 item->setText(m_params.at(col).attribute("max"));
254             if (v <= m_params.at(col).attribute("min").toInt())
255                 item->setText(m_params.at(col).attribute("min"));
256             QString keyframes;
257             for (int i = 0; i < keyframe_list->rowCount(); i++) {
258                 if (keyframe_list->item(i, col))
259                     keyframes.append(QString::number(getPos(i)) + ':' + keyframe_list->item(i, col)->text() + ';');
260             }
261             m_params[col].setAttribute("keyframes", keyframes);
262         }
263
264         emit parameterChanged();
265         return;
266
267     }
268     QTableWidgetItem *item = keyframe_list->item(row, column);
269     if (item == NULL)
270         return;
271
272     int pos = getPos(row);
273     if (pos <= m_min)
274         pos = m_min;
275     if (m_max != -1 && pos > m_max)
276         pos = m_max;
277     /*QList<QTreeWidgetItem *> duplicates = keyframe_list->findItems(val, Qt::MatchExactly, 0);
278     duplicates.removeAll(item);
279     if (!duplicates.isEmpty()) {
280         // Trying to insert a keyframe at existing value, revert it
281         val = m_timecode.getTimecodeFromFrames(m_previousPos);
282     }*/
283     QString val = getPosString(pos);
284     if (val != keyframe_list->verticalHeaderItem(row)->text())
285         keyframe_list->verticalHeaderItem(row)->setText(val);
286
287     int v = item->text().toInt();
288     if (v >= m_params.at(column).attribute("max").toInt())
289         item->setText(m_params.at(column).attribute("max"));
290     if (v <= m_params.at(column).attribute("min").toInt())
291         item->setText(m_params.at(column).attribute("min"));
292     slotAdjustKeyframeInfo(false);
293
294     QString keyframes;
295     for (int i = 0; i < keyframe_list->rowCount(); i++) {
296         if (keyframe_list->item(i, column))
297             keyframes.append(QString::number(getPos(i)) + ':' + keyframe_list->item(i, column)->text() + ';');
298     }
299     m_params[column].setAttribute("keyframes", keyframes);
300     emit parameterChanged();
301 }
302
303 void KeyframeEdit::generateAllParams()
304 {
305     for (int col = 0; col < keyframe_list->columnCount(); col++) {
306         QString keyframes;
307         for (int i = 0; i < keyframe_list->rowCount(); i++) {
308             if (keyframe_list->item(i, col))
309                 keyframes.append(QString::number(getPos(i)) + ':' + keyframe_list->item(i, col)->text() + ';');
310         }
311         m_params[col].setAttribute("keyframes", keyframes);
312     }
313     emit parameterChanged();
314 }
315
316 const QString KeyframeEdit::getValue(const QString &name)
317 {
318     for (int col = 0; col < keyframe_list->columnCount(); col++) {
319         QDomNode na = m_params.at(col).firstChildElement("name");
320         QString paramName = i18n(na.toElement().text().toUtf8().data());
321         if (paramName == name)
322             return m_params.at(col).attribute("keyframes");
323     }
324     return QString();
325 }
326
327 void KeyframeEdit::slotAdjustKeyframeInfo(bool seek)
328 {
329     QTableWidgetItem *item = keyframe_list->currentItem();
330     if (!item)
331         return;
332     int min = m_min;
333     int max = m_max;
334     QTableWidgetItem *above = keyframe_list->item(item->row() - 1, item->column());
335     QTableWidgetItem *below = keyframe_list->item(item->row() + 1, item->column());
336     if (above)
337         min = getPos(above->row()) + 1;
338     if (below)
339         max = getPos(below->row()) - 1;
340
341     m_position->blockSignals(true);
342     m_position->setRange(min, max);
343     m_position->setPosition(getPos(item->row()));
344     m_position->blockSignals(false);
345
346     for (int col = 0; col < keyframe_list->columnCount(); col++) {
347         DoubleParameterWidget *doubleparam = static_cast <DoubleParameterWidget*>(m_slidersLayout->itemAtPosition(col, 0)->widget());
348         if (!doubleparam)
349             continue;
350         doubleparam->blockSignals(true);
351         if (keyframe_list->item(item->row(), col)) {
352             doubleparam->setValue(keyframe_list->item(item->row(), col)->text().toInt());
353         } else {
354             kDebug() << "Null pointer exception caught: http://www.kdenlive.org/mantis/view.php?id=1771";
355         }
356         doubleparam->blockSignals(false);
357     }
358     if (KdenliveSettings::keyframeseek() && seek)
359         emit seekToPos(m_position->getPosition() - m_min);
360 }
361
362 void KeyframeEdit::slotAdjustKeyframePos(int value)
363 {
364     QTableWidgetItem *item = keyframe_list->currentItem();
365     keyframe_list->verticalHeaderItem(item->row())->setText(getPosString(value));
366     slotGenerateParams(item->row(), -1);
367     if (KdenliveSettings::keyframeseek())
368         emit seekToPos(value - m_min);
369 }
370
371 void KeyframeEdit::slotAdjustKeyframeValue(int value)
372 {
373     Q_UNUSED(value);
374
375     QTableWidgetItem *item = keyframe_list->currentItem();
376     for (int col = 0; col < keyframe_list->columnCount(); col++) {
377         DoubleParameterWidget *doubleparam = static_cast <DoubleParameterWidget*>(m_slidersLayout->itemAtPosition(col, 0)->widget());
378         if (!doubleparam)
379             continue;
380         int val = doubleparam->getValue();
381         QTableWidgetItem *nitem = keyframe_list->item(item->row(), col);
382         if (nitem && nitem->text().toInt() != val)
383             nitem->setText(QString::number(val));
384     }
385     //keyframe_list->item(item->row() - 1, item->column());
386
387 }
388
389 int KeyframeEdit::getPos(int row)
390 {
391     if (KdenliveSettings::frametimecode())
392         return keyframe_list->verticalHeaderItem(row)->text().toInt();
393     else
394         return m_timecode.getFrameCount(keyframe_list->verticalHeaderItem(row)->text());
395 }
396
397 QString KeyframeEdit::getPosString(int pos)
398 {
399     if (KdenliveSettings::frametimecode())
400         return QString::number(pos);
401     else
402         return m_timecode.getTimecodeFromFrames(pos);
403 }
404
405 void KeyframeEdit::slotSetSeeking(bool seek)
406 {
407     KdenliveSettings::setKeyframeseek(seek);
408 }
409
410 void KeyframeEdit::updateTimecodeFormat()
411 {
412     for (int row = 0; row < keyframe_list->rowCount(); ++row) {
413         QString pos = keyframe_list->verticalHeaderItem(row)->text();
414         if (KdenliveSettings::frametimecode())
415             keyframe_list->verticalHeaderItem(row)->setText(QString::number(m_timecode.getFrameCount(pos)));
416         else
417             keyframe_list->verticalHeaderItem(row)->setText(m_timecode.getTimecodeFromFrames(pos.toInt()));
418     }
419
420     m_position->updateTimecodeFormat();
421 }
422
423 void KeyframeEdit::slotKeyframeMode()
424 {
425     widgetTable->setHidden(false);
426     buttonKeyframes->setHidden(true);
427     slotAddKeyframe();
428 }
429
430 void KeyframeEdit::slotResetKeyframe()
431 {
432     for (int col = 0; col < keyframe_list->columnCount(); ++col) {
433         DoubleParameterWidget *doubleparam = static_cast<DoubleParameterWidget*>(m_slidersLayout->itemAtPosition(col, 0)->widget());
434         if (doubleparam)
435             doubleparam->slotReset();
436     }
437 }
438
439 void KeyframeEdit::slotUpdateVisibleParameter(int id, bool update)
440 {
441     for (int i = 0; i < m_params.count(); ++i)
442         m_params[i].setAttribute("intimeline", (i == id ? "1" : "0"));
443     if (update) emit parameterChanged();
444 }
445
446 bool KeyframeEdit::isVisibleParam(const QString& name)
447 {
448     for (int col = 0; col < keyframe_list->columnCount(); ++col) {
449         QDomNode na = m_params.at(col).firstChildElement("name");
450         QString paramName = i18n(na.toElement().text().toUtf8().data());
451         if (paramName == name)
452             return m_params.at(col).attribute("intimeline") == "1";
453     }
454     return false;
455 }
456
457 void KeyframeEdit::checkVisibleParam()
458 {
459     if (m_params.count() == 0)
460         return;
461
462     foreach(QDomElement elem, m_params) {
463         if (elem.attribute("intimeline") == "1")
464             return;
465     }
466
467     slotUpdateVisibleParameter(0, false);
468     QRadioButton *radio = static_cast<QRadioButton*>(m_slidersLayout->itemAtPosition(0, 1)->widget());
469     if (radio)
470         radio->setChecked(true);
471 }
472
473 #include "keyframeedit.moc"