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