]> git.sesse.net Git - kdenlive/blob - src/keyframeedit.cpp
Ensure keyframe list widget is not too small to be used
[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
27 KeyframeEdit::KeyframeEdit(QDomElement e, int minFrame, int maxFrame, int minVal, int maxVal, Timecode tc, int active_keyframe, QWidget* parent) :
28         QWidget(parent),
29         m_min(minFrame),
30         m_max(maxFrame),
31         m_minVal(minVal),
32         m_maxVal(maxVal),
33         m_timecode(tc),
34         m_previousPos(0),
35         m_active_keyframe(active_keyframe)
36 {
37     kDebug() << " / / / /MODIFIED KFR: " << m_active_keyframe;
38     setupUi(this);
39     m_params.append(e.cloneNode().toElement());
40     keyframe_list->setFont(KGlobalSettings::generalFont());
41     keyframe_seek->setChecked(KdenliveSettings::keyframeseek());
42     connect(keyframe_seek, SIGNAL(stateChanged(int)), this, SLOT(slotSetSeeking(int)));
43
44     button_add->setIcon(KIcon("list-add"));
45     button_add->setToolTip(i18n("Add keyframe"));
46     button_delete->setIcon(KIcon("list-remove"));
47     button_delete->setToolTip(i18n("Delete keyframe"));
48     connect(keyframe_list, SIGNAL(itemSelectionChanged()), this, SLOT(slotAdjustKeyframeInfo()));
49     connect(keyframe_list, SIGNAL(cellChanged(int, int)), this, SLOT(slotGenerateParams(int, int)));
50     setupParam();
51
52     keyframe_list->resizeRowsToContents();
53     keyframe_list->resizeColumnsToContents();
54     //keyframe_list->horizontalHeader()->resizeSections(QHeaderView::ResizeToContents);
55     connect(button_delete, SIGNAL(clicked()), this, SLOT(slotDeleteKeyframe()));
56     connect(button_add, SIGNAL(clicked()), this, SLOT(slotAddKeyframe()));
57     //connect(keyframe_list, SIGNAL(itemDoubleClicked(QTreeWidgetItem *, int)), this, SLOT(slotSaveCurrentParam(QTreeWidgetItem *, int)));
58     connect(keyframe_pos, SIGNAL(valueChanged(int)), this, SLOT(slotAdjustKeyframePos(int)));
59     //connect(keyframe_val, SIGNAL(valueChanged(int)), this, SLOT(slotAdjustKeyframeValue(int)));
60     keyframe_pos->setPageStep(1);
61     if (!keyframe_list->currentItem()) {
62         keyframe_list->setCurrentCell(0, 0);
63         keyframe_list->selectRow(0);
64     }
65     // ensure the keyframe list shows at least 3 lines
66     keyframe_list->setMinimumHeight(QFontInfo(keyframe_list->font()).pixelSize() * 9);
67     /*m_delegate = new KeyItemDelegate(minVal, maxVal);
68     keyframe_list->setItemDelegate(m_delegate);*/
69 }
70
71 KeyframeEdit::~KeyframeEdit()
72 {
73     keyframe_list->blockSignals(true);
74     keyframe_list->clear();
75     QLayoutItem *child;
76     while ((child = m_slidersLayout->takeAt(0)) != 0) {
77         QWidget *wid = child->widget();
78         delete child;
79         if (wid)
80             delete wid;
81     }
82     //delete m_delegate;
83 }
84
85 void KeyframeEdit::addParameter(QDomElement e)
86 {
87     keyframe_list->blockSignals(true);
88     m_params.append(e.cloneNode().toElement());
89     QDomNode na = e.firstChildElement("name");
90     QString paramName = i18n(na.toElement().text().toUtf8().data());
91     int columnId = keyframe_list->columnCount();
92     keyframe_list->insertColumn(columnId);
93     keyframe_list->setHorizontalHeaderItem(columnId, new QTableWidgetItem(paramName));
94
95     DoubleParameterWidget *doubleparam = new DoubleParameterWidget(paramName, 0,
96             m_params.at(columnId).attribute("min").toInt(), m_params.at(columnId).attribute("max").toInt(),
97             m_params.at(columnId).attribute("default").toInt(), m_params.at(columnId).attribute("suffix"), this);
98     connect(doubleparam, SIGNAL(valueChanged(int)), this, SLOT(slotAdjustKeyframeValue(int)));
99     m_slidersLayout->addWidget(doubleparam, columnId, 0);
100
101     QStringList frames = e.attribute("keyframes").split(";", QString::SkipEmptyParts);
102     for (int i = 0; i < frames.count(); i++) {
103         int frame = frames.at(i).section(':', 0, 0).toInt();
104         bool found = false;
105         int j;
106         for (j = 0; j < keyframe_list->rowCount(); j++) {
107             int currentPos = getPos(j);
108             if (frame == currentPos) {
109                 keyframe_list->setItem(j, columnId, new QTableWidgetItem(frames.at(i).section(':', 1, 1)));
110                 found = true;
111                 break;
112             } else if (currentPos > frame) {
113                 break;
114             }
115         }
116         if (!found) {
117             keyframe_list->insertRow(j);
118             keyframe_list->setVerticalHeaderItem(j, new QTableWidgetItem(getPosString(frame)));
119             keyframe_list->setItem(j, columnId, new QTableWidgetItem(frames.at(i).section(':', 1, 1)));
120             keyframe_list->resizeRowToContents(j);
121         }
122     }
123     keyframe_list->resizeColumnsToContents();
124     keyframe_list->blockSignals(false);
125     slotAdjustKeyframeInfo(false);
126 }
127
128 void KeyframeEdit::setupParam()
129 {
130     keyframe_list->blockSignals(true);
131     keyframe_list->clear();
132     int col = keyframe_list->columnCount();
133     QDomNode na = m_params.at(0).firstChildElement("name");
134     QString paramName = i18n(na.toElement().text().toUtf8().data());
135     kDebug() << " INSERT COL: " << col << ", " << paramName;
136     keyframe_list->insertColumn(col);
137     keyframe_list->setHorizontalHeaderItem(col, new QTableWidgetItem(paramName));
138     m_slidersLayout = new QGridLayout(param_sliders);
139
140     DoubleParameterWidget *doubleparam = new DoubleParameterWidget(paramName, 0,
141             m_params.at(0).attribute("min").toInt(), m_params.at(0).attribute("max").toInt(),
142             m_params.at(0).attribute("default").toInt(), m_params.at(0).attribute("suffix"), this);
143     connect(doubleparam, SIGNAL(valueChanged(int)), this, SLOT(slotAdjustKeyframeValue(int)));
144     m_slidersLayout->addWidget(doubleparam, 0, 0);
145
146     keyframe_list->setSelectionBehavior(QAbstractItemView::SelectRows);
147     keyframe_list->setSelectionMode(QAbstractItemView::SingleSelection);
148
149     QStringList frames = m_params.at(0).attribute("keyframes").split(";", QString::SkipEmptyParts);
150     for (int i = 0; i < frames.count(); i++) {
151         keyframe_list->insertRow(i);
152         int currentpos = frames.at(i).section(':', 0, 0).toInt();
153         keyframe_list->setVerticalHeaderItem(i, new QTableWidgetItem(getPosString(currentpos)));
154         keyframe_list->setItem(i, col, new QTableWidgetItem(frames.at(i).section(':', 1, 1)));
155         if ((m_active_keyframe > -1) && (m_active_keyframe == currentpos)) {
156             keyframe_list->setCurrentCell(i, 0);
157             keyframe_list->selectRow(i);
158         }
159         //item->setFlags(Qt::ItemIsSelectable | Qt::ItemIsEditable | Qt::ItemIsEnabled);
160     }
161     /*QTreeWidgetItem *first = keyframe_list->topLevelItem(0);
162     if (first) keyframe_list->setCurrentItem(first);*/
163     keyframe_list->blockSignals(false);
164     //keyframe_list->setCurrentCell(0, 0);
165     slotAdjustKeyframeInfo(false);
166     button_delete->setEnabled(keyframe_list->rowCount() > 1);
167 }
168
169 void KeyframeEdit::slotDeleteKeyframe()
170 {
171     if (keyframe_list->rowCount() < 2)
172         return;
173     int col = keyframe_list->currentColumn();
174     int row = keyframe_list->currentRow();
175     keyframe_list->removeRow(keyframe_list->currentRow());
176     row = qMin(row, keyframe_list->rowCount() - 1);
177     keyframe_list->setCurrentCell(row, col);
178     keyframe_list->selectRow(row);
179     generateAllParams();
180     button_delete->setEnabled(keyframe_list->rowCount() > 1);
181 }
182
183 void KeyframeEdit::slotAddKeyframe()
184 {
185     keyframe_list->blockSignals(true);
186     QTableWidgetItem *item = keyframe_list->currentItem();
187     int row = keyframe_list->currentRow();
188     int col = keyframe_list->currentColumn();
189     int newrow = row;
190     int pos1 = getPos(row);
191
192     int result;
193     kDebug() << "// ADD KF: " << row << ", MAX: " << keyframe_list->rowCount() << ", POS: " << pos1;
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     //keyframe_list->resizeRowToContents(newrow);
216     slotAdjustKeyframeInfo();
217     keyframe_list->blockSignals(false);
218     generateAllParams();
219     button_delete->setEnabled(keyframe_list->rowCount() > 1);
220     keyframe_list->setCurrentCell(newrow, col);
221     keyframe_list->selectRow(newrow);
222     //slotGenerateParams(newrow, 0);
223 }
224
225 void KeyframeEdit::slotGenerateParams(int row, int column)
226 {
227     if (column == -1) {
228         // position of keyframe changed
229         QTableWidgetItem *item = keyframe_list->item(row, 0);
230         if (item == NULL)
231             return;
232
233         int pos = getPos(row);
234         if (pos <= m_min)
235             pos = m_min;
236         if (pos > m_max)
237             pos = m_max;
238         QString val = getPosString(pos);
239         if (val != keyframe_list->verticalHeaderItem(row)->text())
240             keyframe_list->verticalHeaderItem(row)->setText(val);
241
242         for (int col = 0; col < keyframe_list->horizontalHeader()->count(); col++) {
243             item = keyframe_list->item(row, col);
244             int v = item->text().toInt();
245             if (v >= m_params.at(col).attribute("max").toInt())
246                 item->setText(m_params.at(col).attribute("max"));
247             if (v <= m_params.at(col).attribute("min").toInt())
248                 item->setText(m_params.at(col).attribute("min"));
249             QString keyframes;
250             for (int i = 0; i < keyframe_list->rowCount(); i++) {
251                 if (keyframe_list->item(i, col))
252                     keyframes.append(QString::number(getPos(i)) + ':' + keyframe_list->item(i, col)->text() + ';');
253             }
254             m_params[col].setAttribute("keyframes", keyframes);
255         }
256
257         emit parameterChanged();
258         return;
259
260     }
261     QTableWidgetItem *item = keyframe_list->item(row, column);
262     if (item == NULL)
263         return;
264
265     int pos = getPos(row);
266     if (pos <= m_min)
267         pos = m_min;
268     if (pos > m_max)
269         pos = m_max;
270     /*QList<QTreeWidgetItem *> duplicates = keyframe_list->findItems(val, Qt::MatchExactly, 0);
271     duplicates.removeAll(item);
272     if (!duplicates.isEmpty()) {
273         // Trying to insert a keyframe at existing value, revert it
274         val = m_timecode.getTimecodeFromFrames(m_previousPos);
275     }*/
276     QString val = getPosString(pos);
277     if (val != keyframe_list->verticalHeaderItem(row)->text())
278         keyframe_list->verticalHeaderItem(row)->setText(val);
279
280     int v = item->text().toInt();
281     if (v >= m_params.at(column).attribute("max").toInt())
282         item->setText(m_params.at(column).attribute("max"));
283     if (v <= m_params.at(column).attribute("min").toInt())
284         item->setText(m_params.at(column).attribute("min"));
285     slotAdjustKeyframeInfo(false);
286
287     QString keyframes;
288     for (int i = 0; i < keyframe_list->rowCount(); i++) {
289         if (keyframe_list->item(i, column))
290             keyframes.append(QString::number(getPos(i)) + ':' + keyframe_list->item(i, column)->text() + ';');
291     }
292     m_params[column].setAttribute("keyframes", keyframes);
293     emit parameterChanged();
294 }
295
296 void KeyframeEdit::generateAllParams()
297 {
298     for (int col = 0; col < keyframe_list->columnCount(); col++) {
299         QString keyframes;
300         for (int i = 0; i < keyframe_list->rowCount(); i++) {
301             if (keyframe_list->item(i, col))
302                 keyframes.append(QString::number(getPos(i)) + ':' + keyframe_list->item(i, col)->text() + ';');
303         }
304         m_params[col].setAttribute("keyframes", keyframes);
305     }
306     emit parameterChanged();
307 }
308
309 const QString KeyframeEdit::getValue(const QString &name)
310 {
311     for (int col = 0; col < keyframe_list->columnCount(); col++) {
312         QDomNode na = m_params.at(col).firstChildElement("name");
313         QString paramName = i18n(na.toElement().text().toUtf8().data());
314         kDebug() << paramName << " == " << name;
315         if (paramName == name)
316             return m_params.at(col).attribute("keyframes");
317     }
318     return QString();
319 }
320
321 void KeyframeEdit::slotAdjustKeyframeInfo(bool seek)
322 {
323     QTableWidgetItem *item = keyframe_list->currentItem();
324     if (!item)
325         return;
326     int min = m_min;
327     int max = m_max;
328     QTableWidgetItem *above = keyframe_list->item(item->row() - 1, item->column());
329     QTableWidgetItem *below = keyframe_list->item(item->row() + 1, item->column());
330     if (above)
331         min = getPos(above->row()) + 1;
332     if (below)
333         max = getPos(below->row()) - 1;
334
335     keyframe_pos->blockSignals(true);
336     keyframe_pos->setRange(min, max);
337     keyframe_pos->setValue(getPos(item->row()));
338     keyframe_pos->blockSignals(false);
339     for (int col = 0; col < keyframe_list->columnCount(); col++) {
340         DoubleParameterWidget *doubleparam = static_cast <DoubleParameterWidget*>(m_slidersLayout->itemAtPosition(col, 0)->widget());
341         if (!doubleparam)
342             continue;
343         doubleparam->blockSignals(true);
344         if (keyframe_list->item(item->row(), col)) {
345             doubleparam->setValue(keyframe_list->item(item->row(), col)->text().toInt());
346         } else {
347             kDebug() << "Null pointer exception caught: http://www.kdenlive.org/mantis/view.php?id=1771";
348         }
349         doubleparam->blockSignals(false);
350     }
351     if (KdenliveSettings::keyframeseek() && seek)
352         emit seekToPos(keyframe_pos->value() - m_min);
353 }
354
355 void KeyframeEdit::slotAdjustKeyframePos(int value)
356 {
357     QTableWidgetItem *item = keyframe_list->currentItem();
358     keyframe_list->verticalHeaderItem(item->row())->setText(getPosString(value));
359     slotGenerateParams(item->row(), -1);
360     if (KdenliveSettings::keyframeseek())
361         emit seekToPos(value - m_min);
362 }
363
364 void KeyframeEdit::slotAdjustKeyframeValue(int /*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(int state)
397 {
398     KdenliveSettings::setKeyframeseek(state == Qt::Checked);
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::slotSaveCurrentParam(QTreeWidgetItem *item, int column)
413 {
414     if (item && column == 0) m_previousPos = m_timecode.getFrameCount(item->text(0));
415 }*/
416