]> git.sesse.net Git - kdenlive/blob - src/cornerswidget.cpp
Cleanup effect stack (part 2) - show monitor scene when required
[kdenlive] / src / cornerswidget.cpp
1 /***************************************************************************
2  *   Copyright (C) 2010 by Till Theato (root@ttill.de)                     *
3  *                                                                         *
4  *   This program is free software; you can redistribute it and/or modify  *
5  *   it under the terms of the GNU General Public License as published by  *
6  *   the Free Software Foundation; either version 2 of the License, or     *
7  *   (at your option) any later version.                                   *
8  *                                                                         *
9  *   This program is distributed in the hope that it will be useful,       *
10  *   but WITHOUT ANY WARRANTY; without even the implied warranty of        *
11  *   MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the         *
12  *   GNU General Public License for more details.                          *
13  *                                                                         *
14  *   You should have received a copy of the GNU General Public License     *
15  *   along with this program; if not, write to the                         *
16  *   Free Software Foundation, Inc.,                                       *
17  *   51 Franklin Street, Fifth Floor, Boston, MA  02110-1301  USA          *
18  ***************************************************************************/
19
20 #include "cornerswidget.h"
21 #include "monitor.h"
22 #include "monitorscene.h"
23 #include "monitoreditwidget.h"
24 #include "onmonitoritems/onmonitorcornersitem.h"
25 #include "renderer.h"
26 #include "kdenlivesettings.h"
27
28 #include <QGraphicsView>
29 #include <QGridLayout>
30 #include <QToolButton>
31
32 #include <KIcon>
33
34 inline int lerp( const int a, const int b, double t )
35 {
36     return a + (b - a) * t;
37 }
38
39 CornersWidget::CornersWidget(Monitor *monitor, QDomElement e, int minFrame, int maxFrame, Timecode tc, int activeKeyframe, QWidget* parent) :
40         KeyframeEdit(e, minFrame, maxFrame, tc, activeKeyframe, parent),
41         m_monitor(monitor),
42         m_showScene(true),
43         m_pos(0)
44 {
45     MonitorEditWidget *edit = monitor->getEffectEdit();
46     edit->showVisibilityButton(true);
47     m_scene = edit->getScene();
48     m_scene->cleanup();
49
50     m_item = new OnMonitorCornersItem();
51     m_scene->addItem(m_item);
52
53     // TODO: Better Icons
54     edit->removeCustomControls();
55     edit->addCustomButton(KIcon("transform-move"), i18n("Show additional controls"), this, SLOT(slotShowControls(bool)),
56                           true, KdenliveSettings::onmonitoreffects_cornersshowcontrols());
57     edit->addCustomButton(KIcon("insert-horizontal-rule"), i18n("Show/Hide the lines connecting the corners"), this, SLOT(slotShowLines(bool)),
58                           true, KdenliveSettings::onmonitoreffects_cornersshowlines());
59
60     connect(edit, SIGNAL(showEdit(bool)), this, SLOT(slotShowScene(bool)));
61     connect(m_item, SIGNAL(changed()), this, SLOT(slotUpdateProperties()));
62     connect(m_scene, SIGNAL(addKeyframe()), this, SLOT(slotInsertKeyframe()));
63
64     connect(keyframe_list, SIGNAL(cellChanged(int, int)), this, SLOT(slotUpdateItem()));
65     m_scene->centerView();
66 }
67
68 CornersWidget::~CornersWidget()
69 {
70     m_scene->removeItem(m_item);
71     delete m_item;
72     if (m_monitor) {
73         MonitorEditWidget *edit = m_monitor->getEffectEdit();
74         edit->showVisibilityButton(false);
75         edit->removeCustomControls();
76     }
77 }
78
79 void CornersWidget::addParameter(QDomElement e, int activeKeyframe)
80 {
81     KeyframeEdit::addParameter(e, activeKeyframe);
82
83     if (!m_item->polygon().count())
84         slotUpdateItem();
85 }
86
87 void CornersWidget::slotUpdateItem()
88 {
89     if (keyframe_list->columnCount() < 8)
90         return;
91
92     QTableWidgetItem *keyframe, *keyframeOld;
93     keyframe = keyframeOld = keyframe_list->item(0, 0);
94     for (int row = 0; row < keyframe_list->rowCount(); ++row) {
95         keyframeOld = keyframe;
96         keyframe = keyframe_list->item(row, 0);
97         if (getPos(row) >= m_pos)
98             break;
99     }
100
101     QList<QPointF> points, pointsPrev, pointsNext;
102     pointsPrev = getPoints(keyframeOld);
103     pointsNext = getPoints(keyframe);
104     if (pointsPrev.count() != 4 || pointsNext.count() != 4)
105         return;
106
107     qreal position = (m_pos - getPos(keyframeOld->row())) / (qreal)( getPos(keyframe->row()) - getPos(keyframeOld->row()) + 1 );
108
109     if (keyframeOld  == keyframe) {
110         points = pointsNext;
111     } else {
112         for (int i = 0; i < 4; ++i)
113             points.append(QLineF(pointsPrev.at(i), pointsNext.at(i)).pointAt(position));
114     }
115
116     m_scene->blockSignals(true);
117     m_item->setPolygon(QPolygonF() << points.at(0) << points.at(1) << points.at(2) << points.at(3));
118     m_scene->blockSignals(false);
119
120     bool enable = getPos(keyframe->row()) == m_pos || keyframe_list->rowCount() == 1;
121     m_item->setEnabled(enable);
122     m_scene->setEnabled(enable);
123 }
124
125 void CornersWidget::slotUpdateProperties()
126 {
127     if (keyframe_list->columnCount() < 8)
128         return;
129
130     QPolygonF pol = m_item->polygon();
131
132     QTableWidgetItem *item = keyframe_list->currentItem();
133     double val;
134     for (int col = 0; col < 8; col++) {
135         if (col % 2 == 0)
136             val = pol.at(col / 2).x() / (double)m_monitor->render->frameRenderWidth();
137         else
138             val = pol.at(col / 2).y() / (double)m_monitor->render->renderHeight();
139         val *= 2000;
140         val += 2000;
141         QTableWidgetItem *nitem = keyframe_list->item(item->row(), col);
142         if (nitem->text().toInt() != (int)val)
143             nitem->setText(QString::number((int)val));
144     }
145
146     slotAdjustKeyframeInfo(false);
147 }
148
149 QList<QPointF> CornersWidget::getPoints(QTableWidgetItem* keyframe)
150 {
151     QList<QPointF> points;
152
153     if (!keyframe)
154         return points;
155
156     double val;
157     for (int col = 0; col < 8; col++) {
158         if (!keyframe_list->item(keyframe->row(), col))
159             return QList<QPointF>();
160         val = (keyframe_list->item(keyframe->row(), col)->text().toInt() - 2000) / 2000.;
161         if (col % 2 == 0)
162             points << QPointF(val * m_monitor->render->frameRenderWidth(), 0);
163         else
164             points[col / 2].setY(val * m_monitor->render->renderHeight());
165     }
166     return points;
167 }
168
169 void CornersWidget::slotCheckMonitorPosition(int renderPos)
170 {
171     if (m_showScene)
172         emit checkMonitorPosition(renderPos);
173 }
174
175 void CornersWidget::slotShowScene(bool show)
176 {
177     m_showScene = show;
178     if (!m_showScene)
179         m_monitor->slotShowEffectScene(false);
180     else
181         slotCheckMonitorPosition(m_monitor->render->seekFramePosition());
182 }
183
184 void CornersWidget::slotShowLines(bool show)
185 {
186     KdenliveSettings::setOnmonitoreffects_cornersshowlines(show);
187     m_item->update();
188 }
189
190 void CornersWidget::slotShowControls(bool show)
191 {
192     KdenliveSettings::setOnmonitoreffects_cornersshowcontrols(show);
193     m_item->update();
194 }
195
196 void CornersWidget::slotSyncPosition(int relTimelinePos)
197 {
198     if (keyframe_list->rowCount()) {
199         relTimelinePos = qBound(0, relTimelinePos, m_max);
200         if (relTimelinePos != m_pos - m_min) {
201             m_pos = relTimelinePos + m_min;
202             slotUpdateItem();
203         }
204     }
205 }
206
207 void CornersWidget::slotInsertKeyframe()
208 {
209     keyframe_list->blockSignals(true);
210
211     int row;
212     QTableWidgetItem *keyframe, *keyframeOld;
213     keyframe = keyframeOld = keyframe_list->item(0, 0);
214     for (row = 0; row < keyframe_list->rowCount(); ++row) {
215         keyframeOld = keyframe;
216         keyframe = keyframe_list->item(row, 0);
217         if (getPos(row) >= m_pos)
218             break;
219     }
220
221
222     int pos2;
223     if (row == keyframe_list->rowCount()) {
224         pos2 = m_max;
225     } else {
226         pos2 = getPos(row);
227         if (pos2 == m_pos)
228             return;
229     }
230
231     int pos1 = 0;
232     if (row > 0)
233         pos1 = getPos(row - 1);
234
235     int col = keyframe_list->currentColumn();
236     double pos = (m_pos - pos1) / (double)(pos2 - pos1 + 1);
237
238     keyframe_list->insertRow(row);
239     keyframe_list->setVerticalHeaderItem(row, new QTableWidgetItem(getPosString(m_pos)));
240
241     QPolygonF pol = m_item->polygon();
242     double val;
243     for (int i = 0; i < keyframe_list->columnCount(); i++) {
244         if (i < 8) {
245             if (i % 2 == 0)
246                 val = pol.at(i / 2).x() / (double)m_monitor->render->frameRenderWidth();
247             else
248                 val = pol.at(i / 2).y() / (double)m_monitor->render->renderHeight();
249             val *= 2000;
250             val += 2000;
251             keyframe_list->setItem(row, i, new QTableWidgetItem(QString::number((int)val)));
252         } else {
253             keyframe_list->setItem(row, i, new QTableWidgetItem(QString::number(lerp(keyframe_list->item(keyframeOld->row(), i)->text().toInt(), keyframe_list->item(keyframe->row(), i)->text().toInt(), pos))));
254         }
255     }
256
257     keyframe_list->resizeRowsToContents();
258     slotAdjustKeyframeInfo();
259     keyframe_list->blockSignals(false);
260     generateAllParams();
261     button_delete->setEnabled(true);
262     keyframe_list->setCurrentCell(row, col);
263     keyframe_list->selectRow(row);
264 }
265
266 #include "cornerswidget.moc"