]> git.sesse.net Git - kdenlive/blob - src/cornerswidget.cpp
on-monitor editing:
[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
49     m_item = new OnMonitorCornersItem();
50     m_scene->addItem(m_item);
51
52     // TODO: Better Icons
53     edit->addCustomButton(KIcon("transform-move"), i18n("Show additional controls"), this, SLOT(slotShowControls(bool)),
54                           true, KdenliveSettings::onmonitoreffects_cornersshowcontrols());
55     edit->addCustomButton(KIcon("insert-horizontal-rule"), i18n("Show/Hide the lines connecting the corners"), this, SLOT(slotShowLines(bool)),
56                           true, KdenliveSettings::onmonitoreffects_cornersshowlines());
57
58     connect(edit, SIGNAL(showEdit(bool)), this, SLOT(slotShowScene(bool)));
59     connect(m_monitor, SIGNAL(renderPosition(int)), this, SLOT(slotCheckMonitorPosition(int)));
60     connect(m_item, SIGNAL(changed()), this, SLOT(slotUpdateProperties()));
61     connect(m_scene, SIGNAL(addKeyframe()), this, SLOT(slotInsertKeyframe()));
62
63     connect(keyframe_list, SIGNAL(cellChanged(int, int)), this, SLOT(slotUpdateItem()));
64 }
65
66 CornersWidget::~CornersWidget()
67 {
68     m_scene->removeItem(m_item);
69     delete m_item;
70     if (m_monitor) {
71         MonitorEditWidget *edit = m_monitor->getEffectEdit();
72         edit->showVisibilityButton(false);
73         edit->removeCustomControls();
74         m_monitor->slotEffectScene(false);
75     }
76 }
77
78 void CornersWidget::addParameter(QDomElement e, int activeKeyframe)
79 {
80     KeyframeEdit::addParameter(e, activeKeyframe);
81
82     if (!m_item->polygon().count())
83         slotUpdateItem();
84 }
85
86 void CornersWidget::slotUpdateItem()
87 {
88     if (keyframe_list->columnCount() < 8)
89         return;
90
91     QTableWidgetItem *keyframe, *keyframeOld;
92     keyframe = keyframeOld = keyframe_list->item(0, 0);
93     for (int row = 0; row < keyframe_list->rowCount(); ++row) {
94         keyframeOld = keyframe;
95         keyframe = keyframe_list->item(row, 0);
96         if (getPos(row) >= m_pos)
97             break;
98     }
99
100     QList<QPointF> points, pointsPrev, pointsNext;
101     pointsPrev = getPoints(keyframeOld);
102     pointsNext = getPoints(keyframe);
103     if (pointsPrev.count() != 4 || pointsNext.count() != 4)
104         return;
105
106     qreal position = (m_pos - getPos(keyframeOld->row())) / (qreal)( getPos(keyframe->row()) - getPos(keyframeOld->row()) + 1 );
107
108     if (keyframeOld  == keyframe) {
109         points = pointsNext;
110     } else {
111         for (int i = 0; i < 4; ++i)
112             points.append(QLineF(pointsPrev.at(i), pointsNext.at(i)).pointAt(position));
113     }
114
115     m_scene->blockSignals(true);
116     m_item->setPolygon(QPolygonF() << points.at(0) << points.at(1) << points.at(2) << points.at(3));
117     m_scene->blockSignals(false);
118
119     bool enable = getPos(keyframe->row()) == m_pos || keyframe_list->rowCount() == 1;
120     m_item->setEnabled(enable);
121     m_scene->setEnabled(enable);
122 }
123
124 void CornersWidget::slotUpdateProperties()
125 {
126     if (keyframe_list->columnCount() < 8)
127         return;
128
129     QPolygonF pol = m_item->polygon();
130
131     QTableWidgetItem *item = keyframe_list->currentItem();
132     double val;
133     for (int col = 0; col < 8; col++) {
134         if (col % 2 == 0)
135             val = pol.at(col / 2).x() / (double)m_monitor->render->frameRenderWidth();
136         else
137             val = pol.at(col / 2).y() / (double)m_monitor->render->renderHeight();
138         val *= 2000;
139         val += 2000;
140         QTableWidgetItem *nitem = keyframe_list->item(item->row(), col);
141         if (nitem->text().toInt() != (int)val)
142             nitem->setText(QString::number((int)val));
143     }
144
145     slotAdjustKeyframeInfo(false);
146 }
147
148 QList<QPointF> CornersWidget::getPoints(QTableWidgetItem* keyframe)
149 {
150     QList<QPointF> points;
151
152     if (!keyframe)
153         return points;
154
155     double val;
156     for (int col = 0; col < 8; col++) {
157         if (!keyframe_list->item(keyframe->row(), col))
158             return QList<QPointF>();
159         val = (keyframe_list->item(keyframe->row(), col)->text().toInt() - 2000) / 2000.;
160         if (col % 2 == 0)
161             points << QPointF(val * m_monitor->render->frameRenderWidth(), 0);
162         else
163             points[col / 2].setY(val * m_monitor->render->renderHeight());
164     }
165     return points;
166 }
167
168 void CornersWidget::slotCheckMonitorPosition(int renderPos)
169 {
170     if (m_showScene)
171         emit checkMonitorPosition(renderPos);
172 }
173
174 void CornersWidget::slotShowScene(bool show)
175 {
176     m_showScene = show;
177     if (!m_showScene)
178         m_monitor->slotEffectScene(false);
179     else
180         slotCheckMonitorPosition(m_monitor->render->seekFramePosition());
181 }
182
183 void CornersWidget::slotShowLines(bool show)
184 {
185     KdenliveSettings::setOnmonitoreffects_cornersshowlines(show);
186     m_item->update();
187 }
188
189 void CornersWidget::slotShowControls(bool show)
190 {
191     KdenliveSettings::setOnmonitoreffects_cornersshowcontrols(show);
192     m_item->update();
193 }
194
195 void CornersWidget::slotSyncPosition(int relTimelinePos)
196 {
197     if (keyframe_list->rowCount()) {
198         relTimelinePos = qBound(0, relTimelinePos, m_max);
199         if (relTimelinePos != m_pos) {
200             m_pos = relTimelinePos;
201             slotUpdateItem();
202         }
203     }
204 }
205
206 void CornersWidget::slotInsertKeyframe()
207 {
208     keyframe_list->blockSignals(true);
209
210     int row;
211     QTableWidgetItem *keyframe, *keyframeOld;
212     keyframe = keyframeOld = keyframe_list->item(0, 0);
213     for (row = 0; row < keyframe_list->rowCount(); ++row) {
214         keyframeOld = keyframe;
215         keyframe = keyframe_list->item(row, 0);
216         if (getPos(row) >= m_pos)
217             break;
218     }
219
220
221     int pos2;
222     if (row == keyframe_list->rowCount()) {
223         pos2 = m_max;
224     } else {
225         pos2 = getPos(row);
226         if (pos2 == m_pos)
227             return;
228     }
229
230     int pos1 = 0;
231     if (row > 0)
232         pos1 = getPos(row - 1);
233
234     int col = keyframe_list->currentColumn();
235     double pos = (m_pos - pos1) / (double)(pos2 - pos1 + 1);
236
237     keyframe_list->insertRow(row);
238     keyframe_list->setVerticalHeaderItem(row, new QTableWidgetItem(getPosString(m_pos)));
239
240     QPolygonF pol = m_item->polygon();
241     double val;
242     for (int i = 0; i < keyframe_list->columnCount(); i++) {
243         if (i < 8) {
244             if (i % 2 == 0)
245                 val = pol.at(i / 2).x() / (double)m_monitor->render->frameRenderWidth();
246             else
247                 val = pol.at(i / 2).y() / (double)m_monitor->render->renderHeight();
248             val *= 2000;
249             val += 2000;
250             keyframe_list->setItem(row, i, new QTableWidgetItem(QString::number((int)val)));
251         } else {
252             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))));
253         }
254     }
255
256     keyframe_list->resizeRowsToContents();
257     slotAdjustKeyframeInfo();
258     keyframe_list->blockSignals(false);
259     generateAllParams();
260     button_delete->setEnabled(true);
261     keyframe_list->setCurrentCell(row, col);
262     keyframe_list->selectRow(row);
263 }
264
265 #include "cornerswidget.moc"