]> git.sesse.net Git - kdenlive/blob - src/geometryval.cpp
Fix mem leak + possible crash with transition:
[kdenlive] / src / geometryval.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 "geometryval.h"
19 #include "graphicsscenerectmove.h"
20 #include "kdenlivesettings.h"
21
22 #include <KDebug>
23
24 #include <QGraphicsView>
25 #include <QVBoxLayout>
26 #include <QGraphicsRectItem>
27 #include <QMenu>
28 #include <QInputDialog>
29
30
31 Geometryval::Geometryval(const MltVideoProfile profile, QPoint frame_size, QWidget* parent) :
32         QWidget(parent),
33         m_profile(profile),
34         m_paramRect(NULL),
35         m_geom(NULL),
36         m_path(NULL),
37         m_fixedMode(false),
38         m_frameSize(frame_size)
39 {
40     m_ui.setupUi(this);
41     QVBoxLayout* vbox = new QVBoxLayout(m_ui.widget);
42     QGraphicsView *view = new QGraphicsView(this);
43     view->setBackgroundBrush(QBrush(Qt::black));
44     vbox->addWidget(view);
45     vbox->setContentsMargins(0, 0, 0, 0);
46
47     QVBoxLayout* vbox2 = new QVBoxLayout(m_ui.keyframeWidget);
48     m_helper = new KeyframeHelper(this);
49     vbox2->addWidget(m_helper);
50     vbox2->setContentsMargins(0, 0, 0, 0);
51
52     connect(m_helper, SIGNAL(positionChanged(int)), this, SLOT(slotPositionChanged(int)));
53
54     m_scene = new GraphicsSceneRectMove(this);
55     m_scene->setTool(TITLE_SELECT);
56     view->setScene(m_scene);
57     QGraphicsRectItem *m_frameBorder = new QGraphicsRectItem(QRectF(0, 0, profile.width, profile.height));
58     m_frameBorder->setZValue(-1100);
59     m_frameBorder->setBrush(QColor(255, 255, 0, 30));
60     m_frameBorder->setPen(QPen(QBrush(QColor(255, 255, 255, 255)), 1.0, Qt::DashLine));
61     m_scene->addItem(m_frameBorder);
62
63     m_ui.buttonNext->setIcon(KIcon("media-skip-forward"));
64     m_ui.buttonNext->setToolTip(i18n("Go to next keyframe"));
65     m_ui.buttonPrevious->setIcon(KIcon("media-skip-backward"));
66     m_ui.buttonPrevious->setToolTip(i18n("Go to previous keyframe"));
67     m_ui.buttonAdd->setIcon(KIcon("document-new"));
68     m_ui.buttonAdd->setToolTip(i18n("Add keyframe"));
69     m_ui.buttonDelete->setIcon(KIcon("edit-delete"));
70     m_ui.buttonDelete->setToolTip(i18n("Delete keyframe"));
71
72     QMenu *configMenu = new QMenu(i18n("Misc..."), this);
73     m_ui.buttonMenu->setIcon(KIcon("system-run"));
74     m_ui.buttonMenu->setMenu(configMenu);
75     m_ui.buttonMenu->setPopupMode(QToolButton::QToolButton::InstantPopup);
76
77
78     configMenu->addAction(i18n("Geometry"), this, SLOT(slotGeometry()));
79
80     m_scaleMenu = new QMenu(i18n("Resize..."), this);
81     configMenu->addMenu(m_scaleMenu);
82     m_scaleMenu->addAction(i18n("50%"), this, SLOT(slotResize50()));
83     m_scaleMenu->addAction(i18n("100%"), this, SLOT(slotResize100()));
84     m_scaleMenu->addAction(i18n("200%"), this, SLOT(slotResize200()));
85     m_scaleMenu->addAction(i18n("Original size"), this, SLOT(slotResizeOriginal()));
86     m_scaleMenu->addAction(i18n("Custom"), this, SLOT(slotResizeCustom()));
87
88     m_alignMenu = new QMenu(i18n("Align..."), this);
89     configMenu->addMenu(m_alignMenu);
90     m_alignMenu->addAction(i18n("Center"), this, SLOT(slotAlignCenter()));
91     m_alignMenu->addAction(i18n("Hor. Center"), this, SLOT(slotAlignHCenter()));
92     m_alignMenu->addAction(i18n("Vert. Center"), this, SLOT(slotAlignVCenter()));
93     m_alignMenu->addAction(i18n("Right"), this, SLOT(slotAlignRight()));
94     m_alignMenu->addAction(i18n("Left"), this, SLOT(slotAlignLeft()));
95     m_alignMenu->addAction(i18n("Top"), this, SLOT(slotAlignTop()));
96     m_alignMenu->addAction(i18n("Bottom"), this, SLOT(slotAlignBottom()));
97
98
99     m_syncAction = configMenu->addAction(i18n("Sync timeline cursor"), this, SLOT(slotSyncCursor()));
100     m_syncAction->setCheckable(true);
101     m_syncAction->setChecked(KdenliveSettings::transitionfollowcursor());
102
103     //scene->setSceneRect(0, 0, profile.width * 2, profile.height * 2);
104     //view->fitInView(m_frameBorder, Qt::KeepAspectRatio);
105     const double sc = 100.0 / profile.height * 0.8;
106     QRectF srect = view->sceneRect();
107     view->setSceneRect(srect.x(), -srect.height() / 3 + 10, srect.width(), srect.height() + srect.height() / 3 * 2 - 10);
108     m_scene->setZoom(sc);
109     view->centerOn(m_frameBorder);
110     connect(m_ui.buttonNext , SIGNAL(clicked()) , this , SLOT(slotNextFrame()));
111     connect(m_ui.buttonPrevious , SIGNAL(clicked()) , this , SLOT(slotPreviousFrame()));
112     connect(m_ui.buttonDelete , SIGNAL(clicked()) , this , SLOT(slotDeleteFrame()));
113     connect(m_ui.buttonAdd , SIGNAL(clicked()) , this , SLOT(slotAddFrame()));
114     connect(m_scene, SIGNAL(actionFinished()), this, SLOT(slotUpdateTransitionProperties()));
115     connect(m_scene, SIGNAL(doubleClickEvent()), this, SLOT(slotGeometry()));
116
117 }
118
119
120 Geometryval::~Geometryval()
121 {
122     m_scene->disconnect();
123     delete m_paramRect;
124     delete m_path;
125     delete m_helper;
126     delete m_geom;
127     delete m_scene;
128 }
129
130
131 void Geometryval::slotAlignCenter()
132 {
133     int pos = m_ui.spinPos->value();
134     Mlt::GeometryItem item;
135     int error = m_geom->fetch(&item, pos);
136     if (error || item.key() == false) {
137         // no keyframe under cursor
138         return;
139     }
140     m_paramRect->setPos((m_profile.width - m_paramRect->rect().width()) / 2, (m_profile.height - m_paramRect->rect().height()) / 2);
141     slotUpdateTransitionProperties();
142 }
143
144 void Geometryval::slotAlignHCenter()
145 {
146     int pos = m_ui.spinPos->value();
147     Mlt::GeometryItem item;
148     int error = m_geom->fetch(&item, pos);
149     if (error || item.key() == false) {
150         // no keyframe under cursor
151         return;
152     }
153     m_paramRect->setPos((m_profile.width - m_paramRect->rect().width()) / 2, m_paramRect->pos().y());
154     slotUpdateTransitionProperties();
155 }
156
157 void Geometryval::slotAlignVCenter()
158 {
159     int pos = m_ui.spinPos->value();
160     Mlt::GeometryItem item;
161     int error = m_geom->fetch(&item, pos);
162     if (error || item.key() == false) {
163         // no keyframe under cursor
164         return;
165     }
166     m_paramRect->setPos(m_paramRect->pos().x(), (m_profile.height - m_paramRect->rect().height()) / 2);
167     slotUpdateTransitionProperties();
168 }
169
170 void Geometryval::slotAlignTop()
171 {
172     int pos = m_ui.spinPos->value();
173     Mlt::GeometryItem item;
174     int error = m_geom->fetch(&item, pos);
175     if (error || item.key() == false) {
176         // no keyframe under cursor
177         return;
178     }
179     m_paramRect->setPos(m_paramRect->pos().x(), 0);
180     slotUpdateTransitionProperties();
181 }
182
183 void Geometryval::slotAlignBottom()
184 {
185     int pos = m_ui.spinPos->value();
186     Mlt::GeometryItem item;
187     int error = m_geom->fetch(&item, pos);
188     if (error || item.key() == false) {
189         // no keyframe under cursor
190         return;
191     }
192     m_paramRect->setPos(m_paramRect->pos().x(), m_profile.height - m_paramRect->rect().height());
193     slotUpdateTransitionProperties();
194 }
195
196 void Geometryval::slotAlignLeft()
197 {
198     int pos = m_ui.spinPos->value();
199     Mlt::GeometryItem item;
200     int error = m_geom->fetch(&item, pos);
201     if (error || item.key() == false) {
202         // no keyframe under cursor
203         return;
204     }
205     m_paramRect->setPos(0, m_paramRect->pos().y());
206     slotUpdateTransitionProperties();
207 }
208
209 void Geometryval::slotAlignRight()
210 {
211     int pos = m_ui.spinPos->value();
212     Mlt::GeometryItem item;
213     int error = m_geom->fetch(&item, pos);
214     if (error || item.key() == false) {
215         // no keyframe under cursor
216         return;
217     }
218     m_paramRect->setPos(m_profile.width - m_paramRect->rect().width(), m_paramRect->pos().y());
219     slotUpdateTransitionProperties();
220 }
221
222 void Geometryval::slotResize50()
223 {
224     int pos = m_ui.spinPos->value();
225     Mlt::GeometryItem item;
226     int error = m_geom->fetch(&item, pos);
227     if (error || item.key() == false) {
228         // no keyframe under cursor
229         return;
230     }
231     m_paramRect->setRect(0, 0, m_profile.width / 2, m_profile.height / 2);
232     slotUpdateTransitionProperties();
233 }
234
235 void Geometryval::slotResize100()
236 {
237     int pos = m_ui.spinPos->value();
238     Mlt::GeometryItem item;
239     int error = m_geom->fetch(&item, pos);
240     if (error || item.key() == false) {
241         // no keyframe under cursor
242         return;
243     }
244     m_paramRect->setRect(0, 0, m_profile.width, m_profile.height);
245     slotUpdateTransitionProperties();
246 }
247
248 void Geometryval::slotResize200()
249 {
250     int pos = m_ui.spinPos->value();
251     Mlt::GeometryItem item;
252     int error = m_geom->fetch(&item, pos);
253     if (error || item.key() == false) {
254         // no keyframe under cursor
255         return;
256     }
257     m_paramRect->setRect(0, 0, m_profile.width * 2, m_profile.height * 2);
258     slotUpdateTransitionProperties();
259 }
260
261 void Geometryval::slotResizeOriginal()
262 {
263     if (m_frameSize.isNull()) slotResize100();
264     int pos = m_ui.spinPos->value();
265     Mlt::GeometryItem item;
266     int error = m_geom->fetch(&item, pos);
267     if (error || item.key() == false) {
268         // no keyframe under cursor
269         return;
270     }
271     m_paramRect->setRect(0, 0, m_frameSize.x(), m_frameSize.y());
272     slotUpdateTransitionProperties();
273 }
274
275 void Geometryval::slotResizeCustom()
276 {
277     int pos = m_ui.spinPos->value();
278     Mlt::GeometryItem item;
279     int error = m_geom->fetch(&item, pos);
280     if (error || item.key() == false) {
281         // no keyframe under cursor
282         return;
283     }
284     int scale = m_paramRect->rect().width() * 100 / m_profile.width;
285     bool ok;
286     scale =  QInputDialog::getInteger(this, i18n("Resize..."), i18n("Scale"), scale, 1, 2147483647, 10, &ok);
287     if (!ok) return;
288     m_paramRect->setRect(0, 0, m_profile.width * scale / 100, m_profile.height * scale / 100);
289     slotUpdateTransitionProperties();
290 }
291
292 void Geometryval::slotTransparencyChanged(int transp)
293 {
294     int pos = m_ui.spinPos->value();
295     Mlt::GeometryItem item;
296     int error = m_geom->fetch(&item, pos);
297     if (error || item.key() == false) {
298         // no keyframe under cursor
299         return;
300     }
301     item.mix(transp);
302     m_paramRect->setBrush(QColor(255, 0, 0, transp));
303     m_geom->insert(item);
304     emit parameterChanged();
305 }
306
307 void Geometryval::slotSyncCursor()
308 {
309     KdenliveSettings::setTransitionfollowcursor(m_syncAction->isChecked());
310 }
311
312 void Geometryval::slotPositionChanged(int pos, bool seek)
313 {
314     if (seek && KdenliveSettings::transitionfollowcursor()) emit seekToPos(pos);
315     m_ui.spinPos->setValue(pos);
316     m_helper->setValue(pos);
317     Mlt::GeometryItem item;
318     int error = m_geom->fetch(&item, pos);
319     if (error || item.key() == false) {
320         // no keyframe under cursor, adjust buttons
321         m_ui.buttonAdd->setEnabled(true);
322         m_ui.buttonDelete->setEnabled(false);
323         m_ui.widget->setEnabled(false);
324         m_ui.spinTransp->setEnabled(false);
325         m_scaleMenu->setEnabled(false);
326         m_alignMenu->setEnabled(false);
327     } else {
328         m_ui.buttonAdd->setEnabled(false);
329         m_ui.buttonDelete->setEnabled(true);
330         m_ui.widget->setEnabled(true);
331         m_ui.spinTransp->setEnabled(true);
332         m_scaleMenu->setEnabled(true);
333         m_alignMenu->setEnabled(true);
334     }
335     m_paramRect->setPos(item.x(), item.y());
336     m_paramRect->setRect(0, 0, item.w(), item.h());
337     m_ui.spinTransp->setValue(item.mix());
338     m_paramRect->setBrush(QColor(255, 0, 0, item.mix()));
339 }
340
341 void Geometryval::slotDeleteFrame()
342 {
343     // check there is more than one keyframe
344     Mlt::GeometryItem item;
345     const int pos = m_ui.spinPos->value();
346     int error = m_geom->next_key(&item, pos + 1);
347     if (error) {
348         error = m_geom->prev_key(&item, pos - 1);
349         if (error || item.frame() == pos) return;
350     }
351
352     m_geom->remove(m_ui.spinPos->value());
353     m_ui.buttonAdd->setEnabled(true);
354     m_ui.buttonDelete->setEnabled(false);
355     m_ui.widget->setEnabled(false);
356     m_ui.spinTransp->setEnabled(false);
357     m_scaleMenu->setEnabled(false);
358     m_alignMenu->setEnabled(false);
359     m_helper->update();
360     slotPositionChanged(pos, false);
361     updateTransitionPath();
362     emit parameterChanged();
363 }
364
365 void Geometryval::slotAddFrame()
366 {
367     int pos = m_ui.spinPos->value();
368     Mlt::GeometryItem item;
369     item.frame(pos);
370     item.x(m_paramRect->pos().x());
371     item.y(m_paramRect->pos().y());
372     item.w(m_paramRect->rect().width());
373     item.h(m_paramRect->rect().height());
374     item.mix(m_ui.spinTransp->value());
375     m_geom->insert(item);
376     m_ui.buttonAdd->setEnabled(false);
377     m_ui.buttonDelete->setEnabled(true);
378     m_ui.widget->setEnabled(true);
379     m_ui.spinTransp->setEnabled(true);
380     m_scaleMenu->setEnabled(true);
381     m_alignMenu->setEnabled(true);
382     m_helper->update();
383     emit parameterChanged();
384 }
385
386 void Geometryval::slotNextFrame()
387 {
388     Mlt::GeometryItem item;
389     int error = m_geom->next_key(&item, m_helper->value() + 1);
390     kDebug() << "// SEEK TO NEXT KFR: " << error;
391     if (error) {
392         // Go to end
393         m_ui.spinPos->setValue(m_ui.spinPos->maximum());
394         return;
395     }
396     int pos = item.frame();
397     m_ui.spinPos->setValue(pos);
398 }
399
400 void Geometryval::slotPreviousFrame()
401 {
402     Mlt::GeometryItem item;
403     int error = m_geom->prev_key(&item, m_helper->value() - 1);
404     kDebug() << "// SEEK TO NEXT KFR: " << error;
405     if (error) return;
406     int pos = item.frame();
407     m_ui.spinPos->setValue(pos);
408 }
409
410
411 QDomElement Geometryval::getParamDesc()
412 {
413     m_param.setAttribute("value", m_geom->serialise());
414     kDebug() << " / / UPDATING TRANSITION VALUE: " << m_param.attribute("value");
415     return m_param;
416 }
417
418 void Geometryval::setupParam(const QDomElement par, int minFrame, int maxFrame)
419 {
420     m_param = par;
421     QString val = par.attribute("value");
422     if (par.attribute("fixed") == "1") {
423         m_fixedMode = true;
424         m_ui.buttonPrevious->setHidden(true);
425         m_ui.buttonNext->setHidden(true);
426         m_ui.buttonDelete->setHidden(true);
427         m_ui.buttonAdd->setHidden(true);
428         m_ui.spinTransp->setMaximum(500);
429         m_ui.label_pos->setHidden(true);
430         m_helper->setHidden(true);
431         m_ui.spinPos->setHidden(true);
432     }
433     char *tmp = (char *) qstrdup(val.toUtf8().data());
434     if (m_geom) m_geom->parse(tmp, maxFrame - minFrame, m_profile.width, m_profile.height);
435     else m_geom = new Mlt::Geometry(tmp, maxFrame - minFrame, m_profile.width, m_profile.height);
436     delete[] tmp;
437
438     //kDebug() << " / / UPDATING TRANSITION VALUE: " << m_geom->serialise();
439     //read param her and set rect
440     if (!m_fixedMode) {
441         m_helper->setKeyGeometry(m_geom, maxFrame - minFrame - 1);
442         m_helper->update();
443         /*QDomDocument doc;
444         doc.appendChild(doc.importNode(par, true));
445         kDebug() << "IMPORTED TRANS: " << doc.toString();*/
446         m_ui.spinPos->setMaximum(maxFrame - minFrame - 1);
447         if (m_path == NULL) {
448             m_path = new QGraphicsPathItem();
449             m_path->setPen(QPen(Qt::red));
450             m_scene->addItem(m_path);
451         }
452         updateTransitionPath();
453     }
454     Mlt::GeometryItem item;
455
456     m_geom->fetch(&item, 0);
457     delete m_paramRect;
458     m_paramRect = new QGraphicsRectItem(QRectF(0, 0, item.w(), item.h()));
459     m_paramRect->setPos(item.x(), item.y());
460     m_paramRect->setZValue(0);
461
462     m_paramRect->setPen(QPen(QBrush(QColor(255, 0, 0, 255)), 1.0));
463     m_scene->addItem(m_paramRect);
464     slotPositionChanged(0, false);
465     if (!m_fixedMode) {
466         connect(m_ui.spinPos, SIGNAL(valueChanged(int)), this , SLOT(slotPositionChanged(int)));
467     }
468     connect(m_ui.spinTransp, SIGNAL(valueChanged(int)), this , SLOT(slotTransparencyChanged(int)));
469 }
470
471 void Geometryval::updateTransitionPath()
472 {
473     if (m_fixedMode) return;
474     Mlt::GeometryItem item;
475     int pos = 0;
476     int counter = 0;
477     QPainterPath path;
478     while (true) {
479         if (m_geom->next_key(&item, pos) == 1) break;
480         pos = item.frame();
481         if (counter == 0) {
482             path.moveTo(item.x() + item.w() / 2, item.y() + item.h() / 2);
483         } else {
484             path.lineTo(item.x() + item.w() / 2, item.y() + item.h() / 2);
485         }
486         counter++;
487         pos++;
488     }
489     m_path->setPath(path);
490 }
491
492 void Geometryval::slotUpdateTransitionProperties()
493 {
494     int pos = m_ui.spinPos->value();
495     Mlt::GeometryItem item;
496     int error = m_geom->next_key(&item, pos);
497     if (error || item.frame() != pos) {
498         // no keyframe under cursor
499         return;
500     }
501     QRectF r = m_paramRect->rect().normalized();
502     item.x(m_paramRect->pos().x());
503     item.y(m_paramRect->pos().y());
504     item.w(r.width());
505     item.h(r.height());
506     m_geom->insert(item);
507     updateTransitionPath();
508     emit parameterChanged();
509 }
510
511 void Geometryval::slotGeometry()
512 {
513     int pos = m_ui.spinPos->value();
514     Mlt::GeometryItem item;
515     int error = m_geom->fetch(&item, pos);
516     if (error || item.key() == false) {
517         // no keyframe under cursor
518         return;
519     }
520     QRectF r = m_paramRect->rect().normalized();
521
522     QDialog d(this);
523     m_view.setupUi(&d);
524     d.setWindowTitle(i18n("Frame Geometry"));
525     m_view.value_x->setMaximum(10000);
526     m_view.value_x->setMinimum(-10000);
527     m_view.value_y->setMaximum(10000);
528     m_view.value_y->setMinimum(-10000);
529     m_view.value_width->setMaximum(500000);
530     m_view.value_width->setMinimum(1);
531     m_view.value_height->setMaximum(500000);
532     m_view.value_height->setMinimum(1);
533
534     m_view.value_x->setValue(m_paramRect->pos().x());
535     m_view.value_y->setValue(m_paramRect->pos().y());
536     m_view.value_width->setValue(r.width());
537     m_view.value_height->setValue(r.height());
538     connect(m_view.button_reset , SIGNAL(clicked()) , this , SLOT(slotResetPosition()));
539
540     if (d.exec() == QDialog::Accepted) {
541         m_paramRect->setPos(m_view.value_x->value(), m_view.value_y->value());
542         m_paramRect->setRect(0, 0, m_view.value_width->value(), m_view.value_height->value());
543         slotUpdateTransitionProperties();
544     }
545 }
546
547 void Geometryval::slotResetPosition()
548 {
549     m_view.value_x->setValue(0);
550     m_view.value_y->setValue(0);
551
552     if (m_frameSize.isNull()) {
553         m_view.value_width->setValue(m_profile.width);
554         m_view.value_height->setValue(m_profile.height);
555     } else {
556         m_view.value_width->setValue(m_frameSize.x());
557         m_view.value_height->setValue(m_frameSize.y());
558     }
559 }
560
561 void Geometryval::setFrameSize(QPoint p)
562 {
563     m_frameSize = p;
564 }