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