]> git.sesse.net Git - kdenlive/blob - src/geometryval.cpp
Composite transition: add new "Resize to original size" option that will allow for...
[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     m_scaleMenu = new QMenu(i18n("Resize..."), this);
79     configMenu->addMenu(m_scaleMenu);
80     m_scaleMenu->addAction(i18n("50%"), this, SLOT(slotResize50()));
81     m_scaleMenu->addAction(i18n("100%"), this, SLOT(slotResize100()));
82     m_scaleMenu->addAction(i18n("200%"), this, SLOT(slotResize200()));
83     m_scaleMenu->addAction(i18n("Original size"), this, SLOT(slotResizeOriginal()));
84     m_scaleMenu->addAction(i18n("Custom"), this, SLOT(slotResizeCustom()));
85
86
87     m_alignMenu = new QMenu(i18n("Align..."), this);
88     configMenu->addMenu(m_alignMenu);
89     m_alignMenu->addAction(i18n("Center"), this, SLOT(slotAlignCenter()));
90     m_alignMenu->addAction(i18n("Hor. Center"), this, SLOT(slotAlignHCenter()));
91     m_alignMenu->addAction(i18n("Vert. Center"), this, SLOT(slotAlignVCenter()));
92     m_alignMenu->addAction(i18n("Right"), this, SLOT(slotAlignRight()));
93     m_alignMenu->addAction(i18n("Left"), this, SLOT(slotAlignLeft()));
94     m_alignMenu->addAction(i18n("Top"), this, SLOT(slotAlignTop()));
95     m_alignMenu->addAction(i18n("Bottom"), this, SLOT(slotAlignBottom()));
96
97
98     m_syncAction = configMenu->addAction(i18n("Sync timeline cursor"), this, SLOT(slotSyncCursor()));
99     m_syncAction->setCheckable(true);
100     m_syncAction->setChecked(KdenliveSettings::transitionfollowcursor());
101
102     //scene->setSceneRect(0, 0, profile.width * 2, profile.height * 2);
103     //view->fitInView(m_frameBorder, Qt::KeepAspectRatio);
104     const double sc = 100.0 / profile.height * 0.8;
105     QRectF srect = view->sceneRect();
106     view->setSceneRect(srect.x(), -srect.height() / 3 + 10, srect.width(), srect.height() + srect.height() / 3 * 2 - 10);
107     m_scene->setZoom(sc);
108     view->centerOn(m_frameBorder);
109     connect(m_ui.buttonNext , SIGNAL(clicked()) , this , SLOT(slotNextFrame()));
110     connect(m_ui.buttonPrevious , SIGNAL(clicked()) , this , SLOT(slotPreviousFrame()));
111     connect(m_ui.buttonDelete , SIGNAL(clicked()) , this , SLOT(slotDeleteFrame()));
112     connect(m_ui.buttonAdd , SIGNAL(clicked()) , this , SLOT(slotAddFrame()));
113     connect(m_scene, SIGNAL(actionFinished()), this, SLOT(slotUpdateTransitionProperties()));
114 }
115
116 void Geometryval::slotAlignCenter()
117 {
118     int pos = m_ui.spinPos->value();
119     Mlt::GeometryItem item;
120     int error = m_geom->fetch(&item, pos);
121     if (error || item.key() == false) {
122         // no keyframe under cursor
123         return;
124     }
125     m_paramRect->setPos((m_profile.width - m_paramRect->rect().width()) / 2, (m_profile.height - m_paramRect->rect().height()) / 2);
126     slotUpdateTransitionProperties();
127 }
128
129 void Geometryval::slotAlignHCenter()
130 {
131     int pos = m_ui.spinPos->value();
132     Mlt::GeometryItem item;
133     int error = m_geom->fetch(&item, pos);
134     if (error || item.key() == false) {
135         // no keyframe under cursor
136         return;
137     }
138     m_paramRect->setPos((m_profile.width - m_paramRect->rect().width()) / 2, m_paramRect->pos().y());
139     slotUpdateTransitionProperties();
140 }
141
142 void Geometryval::slotAlignVCenter()
143 {
144     int pos = m_ui.spinPos->value();
145     Mlt::GeometryItem item;
146     int error = m_geom->fetch(&item, pos);
147     if (error || item.key() == false) {
148         // no keyframe under cursor
149         return;
150     }
151     m_paramRect->setPos(m_paramRect->pos().x(), (m_profile.height - m_paramRect->rect().height()) / 2);
152     slotUpdateTransitionProperties();
153 }
154
155 void Geometryval::slotAlignTop()
156 {
157     int pos = m_ui.spinPos->value();
158     Mlt::GeometryItem item;
159     int error = m_geom->fetch(&item, pos);
160     if (error || item.key() == false) {
161         // no keyframe under cursor
162         return;
163     }
164     m_paramRect->setPos(m_paramRect->pos().x(), 0);
165     slotUpdateTransitionProperties();
166 }
167
168 void Geometryval::slotAlignBottom()
169 {
170     int pos = m_ui.spinPos->value();
171     Mlt::GeometryItem item;
172     int error = m_geom->fetch(&item, pos);
173     if (error || item.key() == false) {
174         // no keyframe under cursor
175         return;
176     }
177     m_paramRect->setPos(m_paramRect->pos().x(), m_profile.height - m_paramRect->rect().height());
178     slotUpdateTransitionProperties();
179 }
180
181 void Geometryval::slotAlignLeft()
182 {
183     int pos = m_ui.spinPos->value();
184     Mlt::GeometryItem item;
185     int error = m_geom->fetch(&item, pos);
186     if (error || item.key() == false) {
187         // no keyframe under cursor
188         return;
189     }
190     m_paramRect->setPos(0, m_paramRect->pos().y());
191     slotUpdateTransitionProperties();
192 }
193
194 void Geometryval::slotAlignRight()
195 {
196     int pos = m_ui.spinPos->value();
197     Mlt::GeometryItem item;
198     int error = m_geom->fetch(&item, pos);
199     if (error || item.key() == false) {
200         // no keyframe under cursor
201         return;
202     }
203     m_paramRect->setPos(m_profile.width - m_paramRect->rect().width(), m_paramRect->pos().y());
204     slotUpdateTransitionProperties();
205 }
206
207 void Geometryval::slotResize50()
208 {
209     int pos = m_ui.spinPos->value();
210     Mlt::GeometryItem item;
211     int error = m_geom->fetch(&item, pos);
212     if (error || item.key() == false) {
213         // no keyframe under cursor
214         return;
215     }
216     m_paramRect->setRect(0, 0, m_profile.width / 2, m_profile.height / 2);
217     slotUpdateTransitionProperties();
218 }
219
220 void Geometryval::slotResize100()
221 {
222     int pos = m_ui.spinPos->value();
223     Mlt::GeometryItem item;
224     int error = m_geom->fetch(&item, pos);
225     if (error || item.key() == false) {
226         // no keyframe under cursor
227         return;
228     }
229     m_paramRect->setRect(0, 0, m_profile.width, m_profile.height);
230     slotUpdateTransitionProperties();
231 }
232
233 void Geometryval::slotResize200()
234 {
235     int pos = m_ui.spinPos->value();
236     Mlt::GeometryItem item;
237     int error = m_geom->fetch(&item, pos);
238     if (error || item.key() == false) {
239         // no keyframe under cursor
240         return;
241     }
242     m_paramRect->setRect(0, 0, m_profile.width * 2, m_profile.height * 2);
243     slotUpdateTransitionProperties();
244 }
245
246 void Geometryval::slotResizeOriginal()
247 {
248     if (m_frameSize.isNull()) slotResize100();
249     int pos = m_ui.spinPos->value();
250     Mlt::GeometryItem item;
251     int error = m_geom->fetch(&item, pos);
252     if (error || item.key() == false) {
253         // no keyframe under cursor
254         return;
255     }
256     m_paramRect->setRect(0, 0, m_frameSize.x(), m_frameSize.y());
257     slotUpdateTransitionProperties();
258 }
259
260 void Geometryval::slotResizeCustom()
261 {
262     int pos = m_ui.spinPos->value();
263     Mlt::GeometryItem item;
264     int error = m_geom->fetch(&item, pos);
265     if (error || item.key() == false) {
266         // no keyframe under cursor
267         return;
268     }
269     int scale = m_paramRect->rect().width() * 100 / m_profile.width;
270     bool ok;
271     scale =  QInputDialog::getInteger(this, i18n("Resize..."), i18n("Scale"), scale, 1, 2147483647, 10, &ok);
272     if (!ok) return;
273     m_paramRect->setRect(0, 0, m_profile.width * scale / 100, m_profile.height * scale / 100);
274     slotUpdateTransitionProperties();
275 }
276
277 void Geometryval::slotTransparencyChanged(int transp)
278 {
279     int pos = m_ui.spinPos->value();
280     Mlt::GeometryItem item;
281     int error = m_geom->fetch(&item, pos);
282     if (error || item.key() == false) {
283         // no keyframe under cursor
284         return;
285     }
286     item.mix(transp);
287     m_paramRect->setBrush(QColor(255, 0, 0, transp));
288     m_geom->insert(item);
289     emit parameterChanged();
290 }
291
292 void Geometryval::slotSyncCursor()
293 {
294     KdenliveSettings::setTransitionfollowcursor(m_syncAction->isChecked());
295 }
296
297 void Geometryval::slotPositionChanged(int pos, bool seek)
298 {
299     if (seek && KdenliveSettings::transitionfollowcursor()) emit seekToPos(pos);
300     m_ui.spinPos->setValue(pos);
301     m_helper->setValue(pos);
302     Mlt::GeometryItem item;
303     int error = m_geom->fetch(&item, pos);
304     if (error || item.key() == false) {
305         // no keyframe under cursor, adjust buttons
306         m_ui.buttonAdd->setEnabled(true);
307         m_ui.buttonDelete->setEnabled(false);
308         m_ui.widget->setEnabled(false);
309         m_ui.spinTransp->setEnabled(false);
310         m_scaleMenu->setEnabled(false);
311         m_alignMenu->setEnabled(false);
312     } else {
313         m_ui.buttonAdd->setEnabled(false);
314         m_ui.buttonDelete->setEnabled(true);
315         m_ui.widget->setEnabled(true);
316         m_ui.spinTransp->setEnabled(true);
317         m_scaleMenu->setEnabled(true);
318         m_alignMenu->setEnabled(true);
319     }
320     m_paramRect->setPos(item.x(), item.y());
321     m_paramRect->setRect(0, 0, item.w(), item.h());
322     m_ui.spinTransp->setValue(item.mix());
323     m_paramRect->setBrush(QColor(255, 0, 0, item.mix()));
324 }
325
326 void Geometryval::slotDeleteFrame()
327 {
328     // check there is more than one keyframe
329     Mlt::GeometryItem item;
330     const int pos = m_ui.spinPos->value();
331     int error = m_geom->next_key(&item, pos + 1);
332     if (error) {
333         error = m_geom->prev_key(&item, pos - 1);
334         if (error || item.frame() == pos) return;
335     }
336
337     m_geom->remove(m_ui.spinPos->value());
338     m_ui.buttonAdd->setEnabled(true);
339     m_ui.buttonDelete->setEnabled(false);
340     m_ui.widget->setEnabled(false);
341     m_ui.spinTransp->setEnabled(false);
342     m_scaleMenu->setEnabled(false);
343     m_alignMenu->setEnabled(false);
344     m_helper->update();
345     slotPositionChanged(pos, false);
346     updateTransitionPath();
347     emit parameterChanged();
348 }
349
350 void Geometryval::slotAddFrame()
351 {
352     int pos = m_ui.spinPos->value();
353     Mlt::GeometryItem item;
354     item.frame(pos);
355     item.x(m_paramRect->pos().x());
356     item.y(m_paramRect->pos().y());
357     item.w(m_paramRect->rect().width());
358     item.h(m_paramRect->rect().height());
359     item.mix(m_ui.spinTransp->value());
360     m_geom->insert(item);
361     m_ui.buttonAdd->setEnabled(false);
362     m_ui.buttonDelete->setEnabled(true);
363     m_ui.widget->setEnabled(true);
364     m_ui.spinTransp->setEnabled(true);
365     m_scaleMenu->setEnabled(true);
366     m_alignMenu->setEnabled(true);
367     m_helper->update();
368     emit parameterChanged();
369 }
370
371 void Geometryval::slotNextFrame()
372 {
373     Mlt::GeometryItem item;
374     int error = m_geom->next_key(&item, m_helper->value() + 1);
375     kDebug() << "// SEEK TO NEXT KFR: " << error;
376     if (error) {
377         // Go to end
378         m_ui.spinPos->setValue(m_ui.spinPos->maximum());
379         return;
380     }
381     int pos = item.frame();
382     m_ui.spinPos->setValue(pos);
383 }
384
385 void Geometryval::slotPreviousFrame()
386 {
387     Mlt::GeometryItem item;
388     int error = m_geom->prev_key(&item, m_helper->value() - 1);
389     kDebug() << "// SEEK TO NEXT KFR: " << error;
390     if (error) return;
391     int pos = item.frame();
392     m_ui.spinPos->setValue(pos);
393 }
394
395
396 QDomElement Geometryval::getParamDesc()
397 {
398     m_param.setAttribute("value", m_geom->serialise());
399     kDebug() << " / / UPDATING TRANSITION VALUE: " << m_param.attribute("value");
400     return m_param;
401 }
402
403 void Geometryval::setupParam(const QDomElement& par, int minFrame, int maxFrame)
404 {
405     m_param = par;
406     QString val = par.attribute("value");
407     if (par.attribute("fixed") == "1") {
408         m_fixedMode = true;
409         m_ui.buttonPrevious->setHidden(true);
410         m_ui.buttonNext->setHidden(true);
411         m_ui.buttonDelete->setHidden(true);
412         m_ui.buttonAdd->setHidden(true);
413         m_ui.spinTransp->setMaximum(500);
414         m_ui.label_pos->setHidden(true);
415         m_helper->setHidden(true);
416         m_ui.spinPos->setHidden(true);
417     }
418     char *tmp = (char *) qstrdup(val.toUtf8().data());
419     if (m_geom) m_geom->parse(tmp, maxFrame - minFrame, m_profile.width, m_profile.height);
420     else m_geom = new Mlt::Geometry(tmp, maxFrame - minFrame, m_profile.width, m_profile.height);
421     delete[] tmp;
422
423     //kDebug() << " / / UPDATING TRANSITION VALUE: " << m_geom->serialise();
424     //read param her and set rect
425     if (!m_fixedMode) {
426         m_helper->setKeyGeometry(m_geom, maxFrame - minFrame - 1);
427         m_helper->update();
428         /*QDomDocument doc;
429         doc.appendChild(doc.importNode(par, true));
430         kDebug() << "IMPORTED TRANS: " << doc.toString();*/
431         m_ui.spinPos->setMaximum(maxFrame - minFrame - 1);
432         if (m_path == NULL) {
433             m_path = new QGraphicsPathItem();
434             m_path->setPen(QPen(Qt::red));
435             m_scene->addItem(m_path);
436         }
437         updateTransitionPath();
438     }
439     Mlt::GeometryItem item;
440
441     m_geom->fetch(&item, 0);
442     delete m_paramRect;
443     m_paramRect = new QGraphicsRectItem(QRectF(0, 0, item.w(), item.h()));
444     m_paramRect->setPos(item.x(), item.y());
445     m_paramRect->setZValue(0);
446
447     m_paramRect->setPen(QPen(QBrush(QColor(255, 0, 0, 255)), 1.0));
448     m_scene->addItem(m_paramRect);
449     slotPositionChanged(0, false);
450     if (!m_fixedMode) {
451         connect(m_ui.spinPos, SIGNAL(valueChanged(int)), this , SLOT(slotPositionChanged(int)));
452     }
453     connect(m_ui.spinTransp, SIGNAL(valueChanged(int)), this , SLOT(slotTransparencyChanged(int)));
454 }
455
456 void Geometryval::updateTransitionPath()
457 {
458     if (m_fixedMode) return;
459     Mlt::GeometryItem item;
460     int pos = 0;
461     int counter = 0;
462     QPainterPath path;
463     while (true) {
464         if (m_geom->next_key(&item, pos) == 1) break;
465         pos = item.frame();
466         if (counter == 0) {
467             path.moveTo(item.x() + item.w() / 2, item.y() + item.h() / 2);
468         } else {
469             path.lineTo(item.x() + item.w() / 2, item.y() + item.h() / 2);
470         }
471         counter++;
472         pos++;
473     }
474     m_path->setPath(path);
475 }
476
477 void Geometryval::slotUpdateTransitionProperties()
478 {
479     int pos = m_ui.spinPos->value();
480     Mlt::GeometryItem item;
481     int error = m_geom->next_key(&item, pos);
482     if (error || item.frame() != pos) {
483         // no keyframe under cursor
484         return;
485     }
486     QRectF r = m_paramRect->rect().normalized();
487     item.x(m_paramRect->pos().x());
488     item.y(m_paramRect->pos().y());
489     item.w(r.width());
490     item.h(r.height());
491     m_geom->insert(item);
492     updateTransitionPath();
493     emit parameterChanged();
494 }