]> git.sesse.net Git - kdenlive/blob - src/titlewidget.cpp
Merge branch 'buildsystem' into next
[kdenlive] / src / titlewidget.cpp
1 /***************************************************************************
2                           titlewidget.cpp  -  description
3                              -------------------
4     begin                : Feb 28 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 "titlewidget.h"
19 #include "kdenlivesettings.h"
20 #include "KoSliderCombo.h"
21 #include "kthumb.h"
22 #include <cmath>
23
24 #include <KDebug>
25 #include <KGlobalSettings>
26 #include <KFileDialog>
27 #include <KStandardDirs>
28 #include <KMessageBox>
29 #include <kio/netaccess.h>
30 #include <kdeversion.h>
31
32 #include <QDomDocument>
33 #include <QGraphicsItem>
34 #include <QGraphicsSvgItem>
35 #include <QTimer>
36 #include <QToolBar>
37 #include <QMenu>
38 #include <QSignalMapper>
39 #include <QTextBlockFormat>
40 #include <QTextCursor>
41 #include <QComboBox>
42 #include <QCryptographicHash>
43
44 #if QT_VERSION >= 0x040600
45 #include <QGraphicsEffect>
46 #include <QGraphicsBlurEffect>
47 #include <QGraphicsDropShadowEffect>
48 #endif
49
50 #include <iostream>
51
52 static QList<TitleTemplate> titletemplates;
53
54 // What exactly is this variable good for?
55 int settingUp = false;
56
57 const int IMAGEITEM = 7;
58 const int RECTITEM = 3;
59 const int TEXTITEM = 8;
60
61 const int NOEFFECT = 0;
62 const int BLUREFFECT = 1;
63 const int SHADOWEFFECT = 2;
64 const int TYPEWRITEREFFECT = 3;
65
66 TitleWidget::TitleWidget(KUrl url, Timecode tc, QString projectTitlePath, Render *render, QWidget *parent) :
67         QDialog(parent),
68         Ui::TitleWidget_UI(),
69         m_startViewport(NULL),
70         m_endViewport(NULL),
71         m_render(render),
72         m_count(0),
73         m_unicodeDialog(new UnicodeDialog(UnicodeDialog::InputHex)),
74         m_projectTitlePath(projectTitlePath),
75         m_tc(tc)
76 {
77     setupUi(this);
78     setFont(KGlobalSettings::toolBarFont());
79     frame_properties->setEnabled(false);
80     frame_properties->setFixedHeight(frame_toolbar->height());
81
82     // Set combo sliders values
83     textAlpha->setMinimum(0);
84     textAlpha->setMaximum(255);
85     textAlpha->setDecimals(0);
86     textAlpha->setValue(255);
87     textAlpha->setToolTip(i18n("Font color opacity"));
88
89     textOutlineAlpha->setMinimum(0);
90     textOutlineAlpha->setMaximum(255);
91     textOutlineAlpha->setDecimals(0);
92     textOutlineAlpha->setValue(255);
93     textOutlineAlpha->setToolTip(i18n("Outline color opacity"));
94
95     textOutline->setMinimum(0);
96     textOutline->setMaximum(200);
97     textOutline->setDecimals(0);
98     textOutline->setValue(0);
99     textOutline->setToolTip(i18n("Outline width"));
100
101     backgroundAlpha->setMinimum(0);
102     backgroundAlpha->setMaximum(255);
103     backgroundAlpha->setDecimals(0);
104     backgroundAlpha->setValue(0);
105     backgroundAlpha->setToolTip(i18n("Background color opacity"));
106
107     itemrotatex->setMinimum(-360);
108     itemrotatex->setMaximum(360);
109     itemrotatex->setDecimals(0);
110     itemrotatex->setValue(0);
111     itemrotatex->setToolTip(i18n("Rotation around the X axis"));
112
113     itemrotatey->setMinimum(-360);
114     itemrotatey->setMaximum(360);
115     itemrotatey->setDecimals(0);
116     itemrotatey->setValue(0);
117     itemrotatey->setToolTip(i18n("Rotation around the Y axis"));
118
119     itemrotatez->setMinimum(-360);
120     itemrotatez->setMaximum(360);
121     itemrotatez->setDecimals(0);
122     itemrotatez->setValue(0);
123     itemrotatez->setToolTip(i18n("Rotation around the Z axis"));
124
125     rectBAlpha->setMinimum(0);
126     rectBAlpha->setMaximum(255);
127     rectBAlpha->setDecimals(0);
128     rectBAlpha->setValue(255);
129     rectBAlpha->setToolTip(i18n("Color opacity"));
130
131     rectFAlpha->setMinimum(0);
132     rectFAlpha->setMaximum(255);
133     rectFAlpha->setDecimals(0);
134     rectFAlpha->setValue(255);
135     rectFAlpha->setToolTip(i18n("Border opacity"));
136
137     rectLineWidth->setMinimum(0);
138     rectLineWidth->setMaximum(100);
139     rectLineWidth->setDecimals(0);
140     rectLineWidth->setValue(0);
141     rectLineWidth->setToolTip(i18n("Border width"));
142
143     itemzoom->setSuffix(i18n("%"));
144     m_frameWidth = render->renderWidth();
145     m_frameHeight = render->renderHeight();
146     showToolbars(TITLE_SELECT);
147
148     splitter->setStretchFactor(0, 20);
149
150     //If project is drop frame, set the input mask as such.
151     title_duration->setInputMask("");
152     title_duration->setValidator(m_tc.validator());
153     title_duration->setText(m_tc.reformatSeparators(KdenliveSettings::title_duration()));
154
155     connect(backgroundColor, SIGNAL(clicked()), this, SLOT(slotChangeBackground())) ;
156     connect(backgroundAlpha, SIGNAL(valueChanged(qreal, bool)), this, SLOT(slotChangeBackground())) ;
157
158     connect(fontColorButton, SIGNAL(clicked()), this, SLOT(slotUpdateText())) ;
159     connect(textOutlineColor, SIGNAL(clicked()), this, SLOT(slotUpdateText())) ;
160     connect(font_family, SIGNAL(currentFontChanged(const QFont &)), this, SLOT(slotUpdateText())) ;
161     connect(font_size, SIGNAL(valueChanged(int)), this, SLOT(slotUpdateText())) ;
162     connect(textAlpha, SIGNAL(valueChanged(qreal, bool)), this, SLOT(slotUpdateText()));
163     connect(textOutline, SIGNAL(valueChanged(qreal, bool)), this, SLOT(slotUpdateText()));
164     connect(textOutlineAlpha, SIGNAL(valueChanged(qreal, bool)), this, SLOT(slotUpdateText()));
165     connect(font_weight_box, SIGNAL(currentIndexChanged(int)), this, SLOT(slotUpdateText()));
166
167     connect(font_family, SIGNAL(editTextChanged(const QString &)), this, SLOT(slotFontText(const QString&)));
168
169     connect(rectFAlpha, SIGNAL(valueChanged(qreal, bool)), this, SLOT(rectChanged()));
170     connect(rectBAlpha, SIGNAL(valueChanged(qreal, bool)), this, SLOT(rectChanged()));
171     connect(rectFColor, SIGNAL(clicked()), this, SLOT(rectChanged()));
172     connect(rectBColor, SIGNAL(clicked()), this, SLOT(rectChanged()));
173     connect(rectLineWidth, SIGNAL(valueChanged(qreal, bool)), this, SLOT(rectChanged()));
174
175     /*connect(startViewportX, SIGNAL(valueChanged(int)), this, SLOT(setupViewports()));
176     connect(startViewportY, SIGNAL(valueChanged(int)), this, SLOT(setupViewports()));
177     connect(startViewportSize, SIGNAL(valueChanged(int)), this, SLOT(setupViewports()));
178     connect(endViewportX, SIGNAL(valueChanged(int)), this, SLOT(setupViewports()));
179     connect(endViewportY, SIGNAL(valueChanged(int)), this, SLOT(setupViewports()));
180     connect(endViewportSize, SIGNAL(valueChanged(int)), this, SLOT(setupViewports()));*/
181
182     // Fill effects
183     effect_list->addItem(i18n("None"), NOEFFECT);
184     effect_list->addItem(i18n("Typewriter"), TYPEWRITEREFFECT);
185     effect_list->addItem(i18n("Blur"), BLUREFFECT);
186
187
188     connect(zValue, SIGNAL(valueChanged(int)), this, SLOT(zIndexChanged(int)));
189     connect(itemzoom, SIGNAL(valueChanged(int)), this, SLOT(itemScaled(int)));
190     connect(itemrotatex, SIGNAL(valueChanged(qreal, bool)), this, SLOT(itemRotateX(qreal)));
191     connect(itemrotatey, SIGNAL(valueChanged(qreal, bool)), this, SLOT(itemRotateY(qreal)));
192     connect(itemrotatez, SIGNAL(valueChanged(qreal, bool)), this, SLOT(itemRotateZ(qreal)));
193     connect(itemhcenter, SIGNAL(clicked()), this, SLOT(itemHCenter()));
194     connect(itemvcenter, SIGNAL(clicked()), this, SLOT(itemVCenter()));
195     connect(itemtop, SIGNAL(clicked()), this, SLOT(itemTop()));
196     connect(itembottom, SIGNAL(clicked()), this, SLOT(itemBottom()));
197     connect(itemleft, SIGNAL(clicked()), this, SLOT(itemLeft()));
198     connect(itemright, SIGNAL(clicked()), this, SLOT(itemRight()));
199     connect(effect_list, SIGNAL(currentIndexChanged(int)), this, SLOT(slotAddEffect(int)));
200     connect(typewriter_delay, SIGNAL(valueChanged(int)), this, SLOT(slotEditTypewriter(int)));
201     connect(typewriter_start, SIGNAL(valueChanged(int)), this, SLOT(slotEditTypewriter(int)));
202     connect(blur_radius, SIGNAL(valueChanged(int)), this, SLOT(slotEditBlur(int)));
203     connect(shadow_radius, SIGNAL(valueChanged(int)), this, SLOT(slotEditShadow()));
204     connect(shadow_x, SIGNAL(valueChanged(int)), this, SLOT(slotEditShadow()));
205     connect(shadow_y, SIGNAL(valueChanged(int)), this, SLOT(slotEditShadow()));
206     effect_stack->setHidden(true);
207     effect_frame->setEnabled(false);
208
209     connect(origin_x_left, SIGNAL(clicked()), this, SLOT(slotOriginXClicked()));
210     connect(origin_y_top, SIGNAL(clicked()), this, SLOT(slotOriginYClicked()));
211
212     // Position and size
213     m_signalMapper = new QSignalMapper(this);
214     m_signalMapper->setMapping(value_w, ValueWidth);
215     m_signalMapper->setMapping(value_h, ValueHeight);
216     m_signalMapper->setMapping(value_x, ValueX);
217     m_signalMapper->setMapping(value_y, ValueY);
218     connect(value_w, SIGNAL(valueChanged(int)), m_signalMapper, SLOT(map()));
219     connect(value_h, SIGNAL(valueChanged(int)), m_signalMapper, SLOT(map()));
220     connect(value_x, SIGNAL(valueChanged(int)), m_signalMapper, SLOT(map()));
221     connect(value_y, SIGNAL(valueChanged(int)), m_signalMapper, SLOT(map()));
222     connect(m_signalMapper, SIGNAL(mapped(int)), this, SLOT(slotValueChanged(int)));
223
224     connect(buttonFitZoom, SIGNAL(clicked()), this, SLOT(slotAdjustZoom()));
225     connect(buttonRealSize, SIGNAL(clicked()), this, SLOT(slotZoomOneToOne()));
226     connect(buttonItalic, SIGNAL(clicked()), this, SLOT(slotUpdateText()));
227     connect(buttonUnder, SIGNAL(clicked()), this, SLOT(slotUpdateText()));
228     connect(buttonAlignLeft, SIGNAL(clicked()), this, SLOT(slotUpdateText()));
229     connect(buttonAlignRight, SIGNAL(clicked()), this, SLOT(slotUpdateText()));
230     connect(buttonAlignCenter, SIGNAL(clicked()), this, SLOT(slotUpdateText()));
231     connect(buttonAlignNone, SIGNAL(clicked()), this, SLOT(slotUpdateText()));
232     connect(displayBg, SIGNAL(stateChanged(int)), this, SLOT(displayBackgroundFrame()));
233
234     connect(m_unicodeDialog, SIGNAL(charSelected(QString)), this, SLOT(slotInsertUnicodeString(QString)));
235
236     // mbd
237     connect(this, SIGNAL(accepted()), this, SLOT(slotAccepted()));
238
239     font_weight_box->blockSignals(true);
240     font_weight_box->addItem(i18nc("Font style", "Light"), QFont::Light);
241     font_weight_box->addItem(i18nc("Font style", "Normal"), QFont::Normal);
242     font_weight_box->addItem(i18nc("Font style", "Demi-Bold"), QFont::DemiBold);
243     font_weight_box->addItem(i18nc("Font style", "Bold"), QFont::Bold);
244     font_weight_box->addItem(i18nc("Font style", "Black"), QFont::Black);
245     font_weight_box->setToolTip(i18n("Font weight"));
246     font_weight_box->setCurrentIndex(1);
247     font_weight_box->blockSignals(false);
248
249     buttonFitZoom->setIcon(KIcon("zoom-fit-best"));
250     buttonRealSize->setIcon(KIcon("zoom-original"));
251     buttonItalic->setIcon(KIcon("format-text-italic"));
252     buttonUnder->setIcon(KIcon("format-text-underline"));
253     buttonAlignCenter->setIcon(KIcon("format-justify-center"));
254     buttonAlignLeft->setIcon(KIcon("format-justify-left"));
255     buttonAlignRight->setIcon(KIcon("format-justify-right"));
256     buttonAlignNone->setIcon(KIcon("kdenlive-align-none"));
257
258     buttonAlignNone->setToolTip(i18n("No alignment"));
259     buttonAlignRight->setToolTip(i18n("Align right"));
260     buttonAlignLeft->setToolTip(i18n("Align left"));
261     buttonAlignCenter->setToolTip(i18n("Align center"));
262
263     m_unicodeAction = new QAction(KIcon("kdenlive-insert-unicode"), QString(), this);
264     m_unicodeAction->setShortcut(Qt::SHIFT + Qt::CTRL + Qt::Key_U);
265     m_unicodeAction->setToolTip(getTooltipWithShortcut(i18n("Insert Unicode character"), m_unicodeAction));
266     connect(m_unicodeAction, SIGNAL(triggered()), this, SLOT(slotInsertUnicode()));
267     buttonInsertUnicode->setDefaultAction(m_unicodeAction);
268
269     m_zUp = new QAction(KIcon("kdenlive-zindex-up"), QString(), this);
270     m_zUp->setShortcut(Qt::Key_PageUp);
271     m_zUp->setToolTip(i18n("Raise object"));
272     connect(m_zUp, SIGNAL(triggered()), this, SLOT(slotZIndexUp()));
273     zUp->setDefaultAction(m_zUp);
274
275     m_zDown = new QAction(KIcon("kdenlive-zindex-down"), QString(), this);
276     m_zDown->setShortcut(Qt::Key_PageDown);
277     m_zDown->setToolTip(i18n("Lower object"));
278     connect(m_zDown, SIGNAL(triggered()), this, SLOT(slotZIndexDown()));
279     zDown->setDefaultAction(m_zDown);
280
281     m_zTop = new QAction(KIcon("kdenlive-zindex-top"), QString(), this);
282     // TODO mbt 1414: Shortcut should change z index only if
283     // cursor is NOT in a text field ...
284     //m_zTop->setShortcut(Qt::Key_Home);
285     m_zTop->setToolTip(i18n("Raise object to top"));
286     connect(m_zTop, SIGNAL(triggered()), this, SLOT(slotZIndexTop()));
287     zTop->setDefaultAction(m_zTop);
288
289     m_zBottom = new QAction(KIcon("kdenlive-zindex-bottom"), QString(), this);
290     // TODO mbt 1414
291     //m_zBottom->setShortcut(Qt::Key_End);
292     m_zBottom->setToolTip(i18n("Lower object to bottom"));
293     connect(m_zBottom, SIGNAL(triggered()), this, SLOT(slotZIndexBottom()));
294     zBottom->setDefaultAction(m_zBottom);
295
296     m_selectAll = new QAction(KIcon("kdenlive-select-all"), QString(), this);
297     m_selectAll->setShortcut(Qt::CTRL + Qt::Key_A);
298     connect(m_selectAll, SIGNAL(triggered()), this, SLOT(slotSelectAll()));
299     buttonSelectAll->setDefaultAction(m_selectAll);
300
301     m_selectText = new QAction(KIcon("kdenlive-select-texts"), QString(), this);
302     m_selectText->setShortcut(Qt::CTRL + Qt::Key_T);
303     connect(m_selectText, SIGNAL(triggered()), this, SLOT(slotSelectText()));
304     buttonSelectText->setDefaultAction(m_selectText);
305     buttonSelectText->setEnabled(false);
306
307     m_selectRects = new QAction(KIcon("kdenlive-select-rects"), QString(), this);
308     m_selectRects->setShortcut(Qt::CTRL + Qt::Key_R);
309     connect(m_selectRects, SIGNAL(triggered()), this, SLOT(slotSelectRects()));
310     buttonSelectRects->setDefaultAction(m_selectRects);
311     buttonSelectRects->setEnabled(false);
312
313     m_selectImages = new QAction(KIcon("kdenlive-select-images"), QString(), this);
314     m_selectImages->setShortcut(Qt::CTRL + Qt::Key_I);
315     connect(m_selectImages, SIGNAL(triggered()), this, SLOT(slotSelectImages()));
316     buttonSelectImages->setDefaultAction(m_selectImages);
317     buttonSelectImages->setEnabled(false);
318
319     m_unselectAll = new QAction(KIcon("kdenlive-unselect-all"), QString(), this);
320     m_unselectAll->setShortcut(Qt::SHIFT + Qt::CTRL + Qt::Key_A);
321     connect(m_unselectAll, SIGNAL(triggered()), this, SLOT(slotSelectNone()));
322     buttonUnselectAll->setDefaultAction(m_unselectAll);
323     buttonUnselectAll->setEnabled(false);
324
325     zDown->setIcon(KIcon("kdenlive-zindex-down"));
326     zTop->setIcon(KIcon("kdenlive-zindex-top"));
327     zBottom->setIcon(KIcon("kdenlive-zindex-bottom"));
328     connect(zDown, SIGNAL(clicked()), this, SLOT(slotZIndexDown()));
329     connect(zTop, SIGNAL(clicked()), this, SLOT(slotZIndexTop()));
330     connect(zBottom, SIGNAL(clicked()), this, SLOT(slotZIndexBottom()));
331
332     origin_x_left->setToolTip(i18n("Invert x axis and change 0 point"));
333     origin_y_top->setToolTip(i18n("Invert y axis and change 0 point"));
334     rectBColor->setToolTip(i18n("Select fill color"));
335     rectFColor->setToolTip(i18n("Select border color"));
336     rectBAlpha->setToolTip(i18n("Fill opacity"));
337     rectFAlpha->setToolTip(i18n("Border opacity"));
338     zoom_slider->setToolTip(i18n("Zoom"));
339     buttonRealSize->setToolTip(i18n("Original size (1:1)"));
340     buttonFitZoom->setToolTip(i18n("Fit zoom"));
341     backgroundColor->setToolTip(i18n("Select background color"));
342     backgroundAlpha->setToolTip(i18n("Background opacity"));
343     buttonSelectAll->setToolTip(getTooltipWithShortcut(i18n("Select all"), m_selectAll));
344     buttonSelectText->setToolTip(getTooltipWithShortcut(i18n("Select text items in current selection"), m_selectText));
345     buttonSelectRects->setToolTip(getTooltipWithShortcut(i18n("Select rect items in current selection"), m_selectRects));
346     buttonSelectImages->setToolTip(getTooltipWithShortcut(i18n("Select image items in current selection"), m_selectImages));
347     buttonUnselectAll->setToolTip(getTooltipWithShortcut(i18n("Unselect all"), m_unselectAll));
348
349     itemhcenter->setIcon(KIcon("kdenlive-align-hor"));
350     itemhcenter->setToolTip(i18n("Align item horizontally"));
351     itemvcenter->setIcon(KIcon("kdenlive-align-vert"));
352     itemvcenter->setToolTip(i18n("Align item vertically"));
353     itemtop->setIcon(KIcon("kdenlive-align-top"));
354     itemtop->setToolTip(i18n("Align item to top"));
355     itembottom->setIcon(KIcon("kdenlive-align-bottom"));
356     itembottom->setToolTip(i18n("Align item to bottom"));
357     itemright->setIcon(KIcon("kdenlive-align-right"));
358     itemright->setToolTip(i18n("Align item to right"));
359     itemleft->setIcon(KIcon("kdenlive-align-left"));
360     itemleft->setToolTip(i18n("Align item to left"));
361
362
363     QHBoxLayout *layout = new QHBoxLayout;
364     frame_toolbar->setLayout(layout);
365     layout->setContentsMargins(0, 0, 0, 0);
366     QToolBar *m_toolbar = new QToolBar("titleToolBar", this);
367     int s = style()->pixelMetric(QStyle::PM_SmallIconSize);
368     m_toolbar->setIconSize(QSize(s, s));
369
370     m_buttonCursor = m_toolbar->addAction(KIcon("transform-move"), QString());
371     m_buttonCursor->setCheckable(true);
372     m_buttonCursor->setShortcut(Qt::ALT + Qt::Key_S);
373     m_buttonCursor->setToolTip(i18n("Selection Tool") + ' ' + m_buttonCursor->shortcut().toString());
374     connect(m_buttonCursor, SIGNAL(triggered()), this, SLOT(slotSelectTool()));
375
376     m_buttonText = m_toolbar->addAction(KIcon("insert-text"), QString());
377     m_buttonText->setCheckable(true);
378     m_buttonText->setShortcut(Qt::ALT + Qt::Key_T);
379     m_buttonText->setToolTip(i18n("Add Text") + ' ' + m_buttonText->shortcut().toString());
380     connect(m_buttonText, SIGNAL(triggered()), this, SLOT(slotTextTool()));
381
382     m_buttonRect = m_toolbar->addAction(KIcon("kdenlive-insert-rect"), QString());
383     m_buttonRect->setCheckable(true);
384     m_buttonRect->setShortcut(Qt::ALT + Qt::Key_R);
385     m_buttonRect->setToolTip(i18n("Add Rectangle") + ' ' + m_buttonRect->shortcut().toString());
386     connect(m_buttonRect, SIGNAL(triggered()), this, SLOT(slotRectTool()));
387
388     m_buttonImage = m_toolbar->addAction(KIcon("insert-image"), QString());
389     m_buttonImage->setCheckable(false);
390     m_buttonImage->setShortcut(Qt::ALT + Qt::Key_I);
391     m_buttonImage->setToolTip(i18n("Add Image") + ' ' + m_buttonImage->shortcut().toString());
392     connect(m_buttonImage, SIGNAL(triggered()), this, SLOT(slotImageTool()));
393
394     m_toolbar->addSeparator();
395
396     m_buttonLoad = m_toolbar->addAction(KIcon("document-open"), i18n("Open Document"));
397     m_buttonLoad->setCheckable(false);
398     m_buttonLoad->setShortcut(Qt::CTRL + Qt::Key_O);
399     connect(m_buttonLoad, SIGNAL(triggered()), this, SLOT(loadTitle()));
400
401     m_buttonSave = m_toolbar->addAction(KIcon("document-save-as"), i18n("Save As"));
402     m_buttonSave->setCheckable(false);
403     m_buttonSave->setShortcut(Qt::CTRL + Qt::Key_S);
404     connect(m_buttonSave, SIGNAL(triggered()), this, SLOT(saveTitle()));
405
406     layout->addWidget(m_toolbar);
407
408     // initialize graphic scene
409     m_scene = new GraphicsSceneRectMove(this);
410     graphicsView->setScene(m_scene);
411     graphicsView->setMouseTracking(true);
412     graphicsView->setDragMode(QGraphicsView::RubberBandDrag);
413     graphicsView->setRubberBandSelectionMode(Qt::ContainsItemBoundingRect);
414     m_titledocument.setScene(m_scene, m_frameWidth, m_frameHeight);
415     connect(m_scene, SIGNAL(changed(QList<QRectF>)), this, SLOT(slotChanged()));
416     connect(font_size, SIGNAL(valueChanged(int)), m_scene, SLOT(slotUpdateFontSize(int)));
417
418     // a gradient background
419     /*QRadialGradient *gradient = new QRadialGradient(0, 0, 10);
420     gradient->setSpread(QGradient::ReflectSpread);
421     scene->setBackgroundBrush(*gradient);*/
422
423     QPen framepen(Qt::DotLine);
424     framepen.setColor(Qt::red);
425
426     m_frameBorder = new QGraphicsRectItem(QRectF(0, 0, m_frameWidth, m_frameHeight));
427     m_frameBorder->setPen(framepen);
428     m_frameBorder->setZValue(-1100);
429     m_frameBorder->setBrush(Qt::transparent);
430     m_frameBorder->setFlags(0);
431     graphicsView->scene()->addItem(m_frameBorder);
432     
433     m_frameImage = new QGraphicsPixmapItem();
434     QTransform qtrans;
435     qtrans.scale(2.0, 2.0);
436     m_frameImage->setTransform(qtrans);
437     m_frameImage->setZValue(-1200);
438     m_frameImage->setFlags(0);
439     displayBackgroundFrame();
440     graphicsView->scene()->addItem(m_frameImage);
441
442     connect(m_scene, SIGNAL(selectionChanged()), this , SLOT(selectionChanged()));
443     connect(m_scene, SIGNAL(itemMoved()), this , SLOT(selectionChanged()));
444     connect(m_scene, SIGNAL(sceneZoom(bool)), this , SLOT(slotZoom(bool)));
445     connect(m_scene, SIGNAL(actionFinished()), this , SLOT(slotSelectTool()));
446     //connect(m_scene, SIGNAL(actionFinished()), this , SLOT(selectionChanged()));
447     connect(m_scene, SIGNAL(newRect(QGraphicsRectItem *)), this , SLOT(slotNewRect(QGraphicsRectItem *)));
448     connect(m_scene, SIGNAL(newText(QGraphicsTextItem *)), this , SLOT(slotNewText(QGraphicsTextItem *)));
449     connect(zoom_slider, SIGNAL(valueChanged(int)), this , SLOT(slotUpdateZoom(int)));
450
451
452     // mbd: load saved settings
453     readChoices();
454
455     graphicsView->show();
456     //graphicsView->setRenderHint(QPainter::Antialiasing);
457     graphicsView->setInteractive(true);
458     //graphicsView->resize(400, 300);
459     kDebug() << "// TITLE WIDGWT: " << graphicsView->viewport()->width() << "x" << graphicsView->viewport()->height();
460     //toolBox->setItemEnabled(2, false);
461     m_startViewport = new QGraphicsRectItem(QRectF(0, 0, m_frameWidth, m_frameHeight));
462     m_endViewport = new QGraphicsRectItem(QRectF(0, 0, m_frameWidth, m_frameHeight));
463     m_startViewport->setData(0, m_frameWidth);
464     m_startViewport->setData(1, m_frameHeight);
465     m_endViewport->setData(0, m_frameWidth);
466     m_endViewport->setData(1, m_frameHeight);
467
468     // scale the view so that the title widget is not too big at startup
469     graphicsView->scale(.5, .5);
470     if (!url.isEmpty()) loadTitle(url);
471     else {
472         slotTextTool();
473         QTimer::singleShot(200, this, SLOT(slotAdjustZoom()));
474     }
475     initAnimation();
476     connect(anim_start, SIGNAL(toggled(bool)), this, SLOT(slotAnimStart(bool)));
477     connect(anim_end, SIGNAL(toggled(bool)), this, SLOT(slotAnimEnd(bool)));
478     connect(templateBox, SIGNAL(currentIndexChanged(int)), this, SLOT(templateIndexChanged(int)));
479
480     buttonBox->button(QDialogButtonBox::Ok)->setEnabled(KdenliveSettings::hastitleproducer());
481     refreshTitleTemplates();
482     //templateBox->setIconSize(QSize(60,60));
483     templateBox->clear();
484     templateBox->addItem("");
485     foreach(TitleTemplate t, titletemplates) {
486         templateBox->addItem(t.icon, t.name, t.file);
487     }
488     lastDocumentHash = QCryptographicHash::hash(xml().toString().toAscii(), QCryptographicHash::Md5).toHex();
489 }
490
491 TitleWidget::~TitleWidget()
492 {
493     delete m_buttonRect;
494     delete m_buttonText;
495     delete m_buttonImage;
496     delete m_buttonCursor;
497     delete m_buttonSave;
498     delete m_buttonLoad;
499     delete m_unicodeAction;
500     delete m_zUp;
501     delete m_zDown;
502     delete m_zTop;
503     delete m_zBottom;
504     delete m_selectAll;
505     delete m_selectText;
506     delete m_selectRects;
507     delete m_selectImages;
508     delete m_unselectAll;
509
510     delete m_unicodeDialog;
511     delete m_frameBorder;
512     delete m_frameImage;
513     delete m_startViewport;
514     delete m_endViewport;
515     delete m_scene;
516     delete m_signalMapper;
517 }
518
519 //static
520 QStringList TitleWidget::getFreeTitleInfo(const KUrl &projectUrl, bool isClone)
521 {
522     QStringList result;
523     QString titlePath = projectUrl.path(KUrl::AddTrailingSlash) + "titles/";
524     KStandardDirs::makeDir(titlePath);
525     titlePath.append((isClone == false) ? "title" : "clone");
526     int counter = 0;
527     QString path;
528     while (path.isEmpty() || QFile::exists(path)) {
529         counter++;
530         path = titlePath + QString::number(counter).rightJustified(3, '0', false) + ".png";
531     }
532     result.append(((isClone == false) ? i18n("Title") : i18n("Clone")) + ' ' + QString::number(counter).rightJustified(3, '0', false));
533     result.append(path);
534     return result;
535 }
536
537 // static
538 QString TitleWidget::getTitleResourceFromName(const KUrl &projectUrl, const QString &titleName)
539 {
540     QStringList result;
541     QString titlePath = projectUrl.path(KUrl::AddTrailingSlash) + "titles/";
542     KStandardDirs::makeDir(titlePath);
543     return titlePath + titleName + ".png";
544 }
545
546 // static
547 QStringList TitleWidget::extractImageList(QString xml)
548 {
549     QStringList result;
550     if (xml.isEmpty()) return result;
551     QDomDocument doc;
552     doc.setContent(xml);
553     QDomNodeList images = doc.elementsByTagName("content");
554     for (int i = 0; i < images.count(); i++) {
555         if (images.at(i).toElement().hasAttribute("url"))
556             result.append(images.at(i).toElement().attribute("url"));
557     }
558     return result;
559 }
560
561 // static
562 QStringList TitleWidget::extractFontList(QString xml)
563 {
564     QStringList result;
565     if (xml.isEmpty()) return result;
566     QDomDocument doc;
567     doc.setContent(xml);
568     QDomNodeList images = doc.elementsByTagName("content");
569     for (int i = 0; i < images.count(); i++) {
570         if (images.at(i).toElement().hasAttribute("font"))
571             result.append(images.at(i).toElement().attribute("font"));
572     }
573     return result;
574 }
575 //static
576 void TitleWidget::refreshTitleTemplates()
577 {
578     QStringList titlenamelist = QStringList() << i18n("None");
579     QStringList titlefiles = QStringList() << QString();
580     QStringList filters;
581     filters << "*.kdenlivetitle" ;
582     titletemplates.clear();
583     QStringList titleTemplates = KGlobal::dirs()->findDirs("appdata", "titles");
584     foreach(const QString & folder, titleTemplates) {
585         QStringList filesnames = QDir(folder).entryList(filters, QDir::Files);
586         foreach(const QString & fname, filesnames) {
587             //titlenamelist.append(fname);
588             //titlefiles.append(KUrl(folder).path(KUrl::AddTrailingSlash) + fname);
589             TitleTemplate t;
590             t.name = fname;
591             t.file = KUrl(folder).path(KUrl::AddTrailingSlash) + fname;
592             t.icon = QIcon(KThumb::getImage(t.file, 0, 60, 60));
593             titletemplates.append(t);
594         }
595     }
596     kDebug()  << titlenamelist << titlefiles;
597 }
598 void TitleWidget::templateIndexChanged(int index)
599 {
600     QString item = templateBox->itemData(index).toString();
601     if (item != "") {
602         if (lastDocumentHash != QCryptographicHash::hash(xml().toString().toAscii(), QCryptographicHash::Md5).toHex()) {
603             if (KMessageBox::questionYesNo(this, i18n("Do you really want to load a new template? Changes in this title will be lost!")) == KMessageBox::No) return;
604         }
605         loadTitle(item);
606
607         // mbt 1607: Add property to distinguish between unchanged template titles and user titles.
608         // Text of unchanged template titles should be selected when clicked.
609         QList<QGraphicsItem *> list = graphicsView->scene()->items();
610         foreach(QGraphicsItem * qgItem, list) {
611             if (qgItem->type() == TEXTITEM) {
612                 QGraphicsTextItem *i;
613                 i = static_cast<QGraphicsTextItem *>(qgItem);
614                 i->setProperty("isTemplate", "true");
615                 i->setProperty("templateText", i->toHtml());
616             }
617
618         }
619         lastDocumentHash = QCryptographicHash::hash(xml().toString().toAscii(), QCryptographicHash::Md5).toHex();
620     }
621 }
622 //virtual
623 void TitleWidget::resizeEvent(QResizeEvent * /*event*/)
624 {
625     //slotAdjustZoom();
626 }
627
628 void TitleWidget::slotTextTool()
629 {
630     m_scene->setTool(TITLE_TEXT);
631     showToolbars(TITLE_TEXT);
632     checkButton(TITLE_TEXT);
633 }
634
635 void TitleWidget::slotRectTool()
636 {
637     m_scene->setTool(TITLE_RECTANGLE);
638     showToolbars(TITLE_RECTANGLE);
639     checkButton(TITLE_RECTANGLE);
640
641     // Disable dragging mode, would make dragging a rect impossible otherwise ;)
642     graphicsView->setDragMode(QGraphicsView::NoDrag);
643 }
644
645 void TitleWidget::slotSelectTool()
646 {
647     m_scene->setTool(TITLE_SELECT);
648
649     // Enable rubberband selecting mode.
650     graphicsView->setDragMode(QGraphicsView::RubberBandDrag);
651
652     // Find out which toolbars need to be shown, depending on selected item
653     TITLETOOL t = TITLE_SELECT;
654     QList<QGraphicsItem *> l = graphicsView->scene()->selectedItems();
655     if (l.size() > 0) {
656         switch (l.at(0)->type()) {
657         case TEXTITEM:
658             t = TITLE_TEXT;
659             break;
660         case RECTITEM:
661             t = TITLE_RECTANGLE;
662             break;
663         case IMAGEITEM:
664             t = TITLE_IMAGE;
665             break;
666         }
667     }
668
669     enableToolbars(t);
670     if (t == TITLE_RECTANGLE && (l.at(0) == m_endViewport || l.at(0) == m_startViewport)) {
671         //graphicsView->centerOn(l.at(0));
672         t = TITLE_SELECT;
673     }
674     showToolbars(t);
675
676     if (l.size() > 0) {
677         updateCoordinates(l.at(0));
678         updateDimension(l.at(0));
679         updateRotZoom(l.at(0));
680     }
681
682     checkButton(TITLE_SELECT);
683 }
684
685 void TitleWidget::slotImageTool()
686 {
687     // TODO: find a way to get a list of all supported image types...
688     QString allExtensions = "image/gif image/jpeg image/png image/x-tga image/x-bmp image/svg+xml image/tiff image/x-xcf-gimp image/x-vnd.adobe.photoshop image/x-pcx image/x-exr";
689     KUrl url = KFileDialog::getOpenUrl(KUrl(), allExtensions, this, i18n("Load Image")); //"*.svg *.png *.jpg *.jpeg *.gif *.raw"
690     if (!url.isEmpty()) {
691         if (url.path().endsWith(".svg")) {
692             QGraphicsSvgItem *svg = new QGraphicsSvgItem(url.toLocalFile());
693             svg->setFlags(QGraphicsItem::ItemIsMovable | QGraphicsItem::ItemIsSelectable);
694             svg->setZValue(m_count++);
695             svg->setData(Qt::UserRole, url.path());
696             graphicsView->scene()->addItem(svg);
697         } else {
698             QPixmap pix(url.path());
699             QGraphicsPixmapItem *image = new QGraphicsPixmapItem(pix);
700             image->setShapeMode(QGraphicsPixmapItem::BoundingRectShape);
701             image->setFlags(QGraphicsItem::ItemIsMovable | QGraphicsItem::ItemIsSelectable);
702             image->setData(Qt::UserRole, url.path());
703             image->setZValue(m_count++);
704             graphicsView->scene()->addItem(image);
705         }
706     }
707     m_scene->setTool(TITLE_SELECT);
708     showToolbars(TITLE_SELECT);
709     checkButton(TITLE_SELECT);
710 }
711
712 void TitleWidget::showToolbars(TITLETOOL toolType)
713 {
714     switch (toolType) {
715     case TITLE_SELECT:
716         toolbar_stack->setCurrentIndex(0);
717         break;
718     case TITLE_IMAGE:
719         toolbar_stack->setCurrentIndex(3);
720         break;
721     case TITLE_RECTANGLE:
722         toolbar_stack->setCurrentIndex(1);
723         break;
724     case TITLE_TEXT:
725         toolbar_stack->setCurrentIndex(2);
726         break;
727     }
728 }
729
730 void TitleWidget::enableToolbars(TITLETOOL toolType)
731 {
732     // TITLETOOL is defined in graphicsscenerectmove.h
733     bool enable = false;
734     if (toolType == TITLE_RECTANGLE || toolType == TITLE_IMAGE) enable = true;
735     value_w->setEnabled(enable);
736     value_h->setEnabled(enable);
737 }
738
739 void TitleWidget::checkButton(TITLETOOL toolType)
740 {
741     bool bSelect = false;
742     bool bText = false;
743     bool bRect = false;
744     bool bImage = false;
745
746     switch (toolType) {
747     case TITLE_SELECT:
748         bSelect = true;
749         break;
750     case TITLE_TEXT:
751         bText = true;
752         break;
753     case TITLE_RECTANGLE:
754         bRect = true;
755         break;
756     case TITLE_IMAGE:
757         bImage = true;
758         break;
759     default:
760         break;
761     }
762
763     m_buttonCursor->setChecked(bSelect);
764     m_buttonText->setChecked(bText);
765     m_buttonRect->setChecked(bRect);
766     m_buttonImage->setChecked(bImage);
767 }
768
769 void TitleWidget::displayBackgroundFrame()
770 {
771     QRectF r = m_frameBorder->sceneBoundingRect();
772     if (!displayBg->isChecked()) {
773         QPixmap pattern(20, 20);
774         pattern.fill();
775         QColor bgcolor(210, 210, 210);
776         QPainter p(&pattern);
777         p.fillRect(QRect(0, 0, 10, 10), bgcolor);
778         p.fillRect(QRect(10, 10, 20, 20), bgcolor);
779         p.end();
780         QBrush br(pattern);
781         QPixmap bg((int) (r.width() / 2), (int) (r.height()/ 2));
782         QPainter p2(&bg);
783         p2.fillRect(bg.rect(), br);
784         p2.end();
785         m_frameImage->setPixmap(bg);
786     } else {
787         QImage img = m_render->extractFrame((int) m_render->seekPosition().frames(m_render->fps()), QString(), m_render->frameRenderWidth() / 2, m_render->renderHeight() / 2);
788         m_frameImage->setPixmap(QPixmap::fromImage(img.scaled(r.width() / 2, r.height() / 2)));
789     }
790 }
791
792 void TitleWidget::initAnimation()
793 {
794     align_box->setEnabled(false);
795     QPen startpen(Qt::DotLine);
796     QPen endpen(Qt::DashDotLine);
797     startpen.setColor(QColor(100, 200, 100, 140));
798     endpen.setColor(QColor(200, 100, 100, 140));
799
800     m_startViewport->setPen(startpen);
801     m_endViewport->setPen(endpen);
802
803     m_startViewport->setZValue(-1000);
804     m_endViewport->setZValue(-1000);
805
806     m_startViewport->setFlags(0);
807     m_endViewport->setFlags(0);
808
809     graphicsView->scene()->addItem(m_startViewport);
810     graphicsView->scene()->addItem(m_endViewport);
811
812     connect(keep_aspect, SIGNAL(toggled(bool)), this, SLOT(slotKeepAspect(bool)));
813     connect(resize50, SIGNAL(clicked()), this, SLOT(slotResize50()));
814     connect(resize100, SIGNAL(clicked()), this, SLOT(slotResize100()));
815     connect(resize200, SIGNAL(clicked()), this, SLOT(slotResize200()));
816 }
817
818 void TitleWidget::slotUpdateZoom(int pos)
819 {
820     m_scene->setZoom((double) pos / 100);
821     zoom_label->setText(QString::number(pos) + '%');
822 }
823
824 void TitleWidget::slotZoom(bool up)
825 {
826     int pos = zoom_slider->value();
827     if (up) pos++;
828     else pos--;
829     zoom_slider->setValue(pos);
830 }
831
832 void TitleWidget::slotAdjustZoom()
833 {
834     /*double scalex = graphicsView->width() / (double)(m_frameWidth * 1.2);
835     double scaley = graphicsView->height() / (double)(m_frameHeight * 1.2);
836     if (scalex > scaley) scalex = scaley;
837     int zoompos = (int)(scalex * 7 + 0.5);*/
838     graphicsView->fitInView(m_frameBorder, Qt::KeepAspectRatio);
839     int zoompos = graphicsView->matrix().m11() * 100;
840     zoom_slider->setValue(zoompos);
841     graphicsView->centerOn(m_frameBorder);
842 }
843
844 void TitleWidget::slotZoomOneToOne()
845 {
846     zoom_slider->setValue(100);
847     graphicsView->centerOn(m_frameBorder);
848 }
849
850 void TitleWidget::slotNewRect(QGraphicsRectItem * rect)
851 {
852     updateAxisButtons(rect); // back to default
853
854     QColor f = rectFColor->color();
855     f.setAlpha(rectFAlpha->value());
856     QPen penf(f);
857     penf.setWidth(rectLineWidth->value());
858     penf.setJoinStyle(Qt::RoundJoin);
859     rect->setPen(penf);
860     QColor b = rectBColor->color();
861     b.setAlpha(rectBAlpha->value());
862     rect->setBrush(QBrush(b));
863     rect->setZValue(m_count++);
864     rect->setData(ZOOMFACTOR, 100);
865     //setCurrentItem(rect);
866     //graphicsView->setFocus();
867 }
868
869 void TitleWidget::slotNewText(QGraphicsTextItem *tt)
870 {
871     updateAxisButtons(tt); // back to default
872
873     QFont font = font_family->currentFont();
874     font.setPixelSize(font_size->value());
875     // mbd: issue 551:
876     font.setWeight(font_weight_box->itemData(font_weight_box->currentIndex()).toInt());
877     font.setItalic(buttonItalic->isChecked());
878     font.setUnderline(buttonUnder->isChecked());
879
880     tt->setFont(font);
881     QColor color = fontColorButton->color();
882     color.setAlpha(textAlpha->value());
883     tt->setDefaultTextColor(color);
884
885     QTextCursor cur(tt->document());
886     cur.select(QTextCursor::Document);
887     QTextBlockFormat format = cur.blockFormat();
888     QTextCharFormat cformat = cur.charFormat();
889     QColor outlineColor = textOutlineColor->color();
890     outlineColor.setAlpha(textOutlineAlpha->value());
891     double outlineWidth = textOutline->value() / 10.0;
892
893     tt->setData(101, outlineWidth);
894     tt->setData(102, outlineColor);
895     if (outlineWidth > 0.0) cformat.setTextOutline(QPen(outlineColor, outlineWidth));
896
897     cformat.setForeground(QBrush(color));
898     cur.setCharFormat(cformat);
899     cur.setBlockFormat(format);
900     tt->setTextCursor(cur);
901     tt->setZValue(m_count++);
902     setCurrentItem(tt);
903 }
904
905 void TitleWidget::setFontBoxWeight(int weight)
906 {
907     int index = font_weight_box->findData(weight);
908     if (index < 0) {
909         index = font_weight_box->findData(QFont::Normal);
910     }
911     font_weight_box->setCurrentIndex(index);
912 }
913
914 void TitleWidget::setCurrentItem(QGraphicsItem *item)
915 {
916     m_scene->setSelectedItem(item);
917 }
918
919 void TitleWidget::zIndexChanged(int v)
920 {
921     QList<QGraphicsItem*> l = graphicsView->scene()->selectedItems();
922     for (int i = 0; i < l.size(); i++) {
923         l[i]->setZValue(v);
924     }
925 }
926
927 void TitleWidget::selectionChanged()
928 {
929     if (m_scene->tool() != TITLE_SELECT) return;
930
931     std::cout << "Number of selected items: " << graphicsView->scene()->selectedItems().length() << "\n";
932
933     QList<QGraphicsItem *> l;
934
935     // mbt 1607: One text item might have grabbed the keyboard.
936     // Ungrab it for all items that are not selected, otherwise
937     // text input would only work for the text item that grabbed
938     // the keyboard last.
939     l = graphicsView->scene()->items();
940     foreach(QGraphicsItem * item, l) {
941         if (item->type() == TEXTITEM && !item->isSelected()) {
942             QGraphicsTextItem *i = static_cast<QGraphicsTextItem *>(item);
943             i->ungrabKeyboard();
944         }
945     }
946
947     l = graphicsView->scene()->selectedItems();
948
949     if (l.size() > 0) {
950         buttonUnselectAll->setEnabled(true);
951         // Enable all z index buttons if items selected.
952         // We can selectively disable them later.
953         zUp->setEnabled(true);
954         zDown->setEnabled(true);
955         zTop->setEnabled(true);
956         zBottom->setEnabled(true);
957     } else {
958         buttonUnselectAll->setEnabled(false);
959     }
960     if (l.size() >= 2) {
961         buttonSelectText->setEnabled(true);
962         buttonSelectRects->setEnabled(true);
963         buttonSelectImages->setEnabled(true);
964     } else {
965         buttonSelectText->setEnabled(false);
966         buttonSelectRects->setEnabled(false);
967         buttonSelectImages->setEnabled(false);
968     }
969
970     if (l.size() == 0) {
971         prepareTools(NULL);
972     } else if (l.size() == 1) {
973         prepareTools(l.at(0));
974     } else {
975         /*
976         For multiple selected objects we need to decide which tools to show.
977         */
978         int firstType = l.at(0)->type();
979         bool allEqual = true;
980         for (int i = 0; i < l.size(); i++) {
981             if (l.at(i)->type() != firstType) {
982                 allEqual = false;
983                 break;
984             }
985         }
986         std::cout << "All equal? " << allEqual << ".\n";
987         if (allEqual) {
988             prepareTools(l.at(0));
989         } else {
990             // Get the default toolset, but enable the property frame (x,y,w,h)
991             prepareTools(NULL);
992             frame_properties->setEnabled(true);
993
994             // Enable x/y/w/h if it makes sense.
995             value_x->setEnabled(true);
996             value_y->setEnabled(true);
997             bool containsTextitem = false;
998             for (int i = 0; i < l.size(); i++) {
999                 if (l.at(i)->type() == TEXTITEM) {
1000                     containsTextitem = true;
1001                     break;
1002                 }
1003             }
1004             if (!containsTextitem) {
1005                 value_w->setEnabled(true);
1006                 value_h->setEnabled(true);
1007             }
1008         }
1009
1010         // Disable z index buttons if they don't make sense for the current selection
1011         int firstZindex = l.at(0)->zValue();
1012         allEqual = true;
1013         for (int i = 0; i < l.size(); i++) {
1014             if (l[i]->zValue() != firstZindex) {
1015                 allEqual = false;
1016                 break;
1017             }
1018         }
1019         if (!allEqual) {
1020             zUp->setEnabled(false);
1021             zDown->setEnabled(false);
1022         }
1023
1024
1025     }
1026
1027 }
1028
1029 void TitleWidget::slotValueChanged(int type)
1030 {
1031     /*
1032     type tells us which QSpinBox value has changed.
1033     */
1034
1035     QList<QGraphicsItem *> l = graphicsView->scene()->selectedItems();
1036     std::cout << l.size() << " items to be resized\n";
1037
1038     // Get the updated value here already to do less coding afterwards
1039     int val = 0;
1040     switch (type) {
1041     case ValueWidth:
1042         val = value_w->value();
1043         break;
1044     case ValueHeight:
1045         val = value_h->value();
1046         break;
1047     case ValueX:
1048         val = value_x->value();
1049         break;
1050     case ValueY:
1051         val = value_y->value();
1052         break;
1053     }
1054
1055     for (int k = 0; k < l.size(); k++) {
1056         std::cout << "Type of item " << k << ": " << l.at(k)->type() << "\n";
1057
1058         if (l.at(k)->type() == TEXTITEM) {
1059             // Just update the position. We don't allow setting width/height for text items yet.
1060             switch (type) {
1061             case ValueX:
1062                 updatePosition(l.at(k), val, l.at(k)->pos().y());
1063                 break;
1064             case ValueY:
1065                 updatePosition(l.at(k), l.at(k)->pos().x(), val);
1066                 break;
1067             }
1068
1069         } else if (l.at(k)->type() == RECTITEM) {
1070             QGraphicsRectItem *rec = static_cast <QGraphicsRectItem *>(l.at(k));
1071             switch (type) {
1072             case ValueX:
1073                 updatePosition(l.at(k), val, l.at(k)->pos().y());
1074                 break;
1075             case ValueY:
1076                 updatePosition(l.at(k), l.at(k)->pos().x(), val);
1077                 break;
1078             case ValueWidth:
1079                 rec->setRect(QRect(0, 0, val, rec->rect().height()));
1080                 break;
1081             case ValueHeight:
1082                 rec->setRect(QRect(0, 0, rec->rect().width(), val));
1083                 break;
1084             }
1085
1086         } else if (l.at(k)->type() == IMAGEITEM) {
1087
1088
1089             if (type == ValueX) {
1090                 updatePosition(l.at(k), val, l.at(k)->pos().y());
1091
1092             } else if (type == ValueY) {
1093                 updatePosition(l.at(k), l.at(k)->pos().x(), val);
1094
1095             } else {
1096                 // Width/height has changed. This is more complex.
1097
1098                 QGraphicsItem *i = l.at(k);
1099                 Transform t = m_transformations.value(i);
1100
1101                 // Ratio width:height
1102                 double phi = (double) i->boundingRect().width() / i->boundingRect().height();
1103                 // TODO: proper calculation for rotation around 3 axes
1104                 double alpha = (double) t.rotatez / 180.0 * M_PI;
1105
1106                 // New length
1107                 double length = val;
1108
1109                 // Scaling factor
1110                 double scale = 1;
1111
1112                 // We want to keep the aspect ratio of the image as the user does not yet have the possibility
1113                 // to restore the original ratio. You rarely want to change it anyway.
1114                 switch (type) {
1115                 case ValueWidth:
1116                     // Add 0.5 because otherwise incrementing by 1 might have no effect
1117                     length = val / (cos(alpha) + 1 / phi * sin(alpha)) + 0.5;
1118                     scale = length / i->boundingRect().width();
1119                     break;
1120                 case ValueHeight:
1121                     length = val / (phi * sin(alpha) + cos(alpha)) + 0.5;
1122                     scale = length / i->boundingRect().height();
1123                     break;
1124                 }
1125
1126                 t.scalex = scale;
1127                 t.scaley = scale;
1128                 QTransform qtrans;
1129                 qtrans.scale(scale, scale);
1130                 qtrans.rotate(t.rotatex, Qt::XAxis);
1131                 qtrans.rotate(t.rotatey, Qt::YAxis);
1132                 qtrans.rotate(t.rotatez, Qt::ZAxis);
1133                 i->setTransform(qtrans);
1134                 std::cout << "scale is: " << scale << "\n";
1135                 std::cout << i->boundingRect().width() << ": new width\n";
1136                 m_transformations[i] = t;
1137
1138                 if (l.size() == 1) {
1139                     // Only update the w/h values if the selection contains just one item.
1140                     // Otherwise, what should we do? ;)
1141                     // (Use the values of the first item? Of the second? Of the x-th?)
1142                     updateDimension(i);
1143                     // Update rotation/zoom values.
1144                     // These values are not yet able to handle multiple items!
1145                     updateRotZoom(i);
1146                 }
1147             }
1148
1149         }
1150     }
1151
1152
1153 }
1154
1155 void TitleWidget::updateDimension(QGraphicsItem *i)
1156 {
1157     bool wBlocked = value_w->signalsBlocked();
1158     bool hBlocked = value_h->signalsBlocked();
1159     bool zBlocked = zValue->signalsBlocked();
1160     value_w->blockSignals(true);
1161     value_h->blockSignals(true);
1162     zValue->blockSignals(true);
1163
1164     zValue->setValue((int) i->zValue());
1165     if (i->type() == IMAGEITEM) {
1166         // Get multipliers for rotation/scaling
1167
1168         /*Transform t = m_transformations.value(i);
1169         QRectF r = i->boundingRect();
1170         int width = (int) ( abs(r.width()*t.scalex * cos(t.rotate/180.0*M_PI))
1171                     + abs(r.height()*t.scaley * sin(t.rotate/180.0*M_PI)) );
1172         int height = (int) ( abs(r.height()*t.scaley * cos(t.rotate/180*M_PI))
1173                     + abs(r.width()*t.scalex * sin(t.rotate/180*M_PI)) );*/
1174
1175         value_w->setValue(i->sceneBoundingRect().width());
1176         value_h->setValue(i->sceneBoundingRect().height());
1177     } else if (i->type() == RECTITEM) {
1178         QGraphicsRectItem *r = static_cast <QGraphicsRectItem *>(i);
1179         std::cout << "Rect width is: " << r->rect().width() << ", was: " << value_w->value() << "\n";
1180         value_w->setValue((int) r->rect().width());
1181         value_h->setValue((int) r->rect().height());
1182     } else if (i->type() == TEXTITEM) {
1183         QGraphicsTextItem *t = static_cast <QGraphicsTextItem *>(i);
1184         value_w->setValue((int) t->boundingRect().width());
1185         value_h->setValue((int) t->boundingRect().height());
1186     }
1187
1188     zValue->blockSignals(zBlocked);
1189     value_w->blockSignals(wBlocked);
1190     value_h->blockSignals(hBlocked);
1191 }
1192
1193 void TitleWidget::updateCoordinates(QGraphicsItem *i)
1194 {
1195     // Block signals emitted by this method
1196     value_x->blockSignals(true);
1197     value_y->blockSignals(true);
1198
1199     if (i->type() == TEXTITEM) {
1200
1201         QGraphicsTextItem *rec = static_cast <QGraphicsTextItem *>(i);
1202
1203         // Set the correct x coordinate value
1204         if (origin_x_left->isChecked()) {
1205             // Origin (0 point) is at m_frameWidth, coordinate axis is inverted
1206             value_x->setValue((int)(m_frameWidth - rec->pos().x() - rec->boundingRect().width()));
1207         } else {
1208             // Origin is at 0 (default)
1209             value_x->setValue((int) rec->pos().x());
1210         }
1211
1212         // Same for y
1213         if (origin_y_top->isChecked()) {
1214             value_y->setValue((int)(m_frameHeight - rec->pos().y() - rec->boundingRect().height()));
1215         } else {
1216             value_y->setValue((int) rec->pos().y());
1217         }
1218
1219     } else if (i->type() == RECTITEM) {
1220
1221         QGraphicsRectItem *rec = static_cast <QGraphicsRectItem *>(i);
1222
1223         if (origin_x_left->isChecked()) {
1224             // Origin (0 point) is at m_frameWidth
1225             value_x->setValue((int)(m_frameWidth - rec->pos().x() - rec->rect().width()));
1226         } else {
1227             // Origin is at 0 (default)
1228             value_x->setValue((int) rec->pos().x());
1229         }
1230
1231         if (origin_y_top->isChecked()) {
1232             value_y->setValue((int)(m_frameHeight - rec->pos().y() - rec->rect().height()));
1233         } else {
1234             value_y->setValue((int) rec->pos().y());
1235         }
1236
1237     } else if (i->type() == IMAGEITEM) {
1238
1239         if (origin_x_left->isChecked()) {
1240             value_x->setValue((int)(m_frameWidth - i->pos().x() - i->sceneBoundingRect().width()));
1241         } else {
1242             value_x->setValue((int) i->pos().x());
1243         }
1244
1245         if (origin_y_top->isChecked()) {
1246             value_y->setValue((int)(m_frameHeight - i->pos().y() - i->sceneBoundingRect().height()));
1247         } else {
1248             value_y->setValue((int) i->pos().y());
1249         }
1250
1251     }
1252
1253     // Stop blocking signals now
1254     value_x->blockSignals(false);
1255     value_y->blockSignals(false);
1256 }
1257
1258 void TitleWidget::updateRotZoom(QGraphicsItem *i)
1259 {
1260     itemzoom->blockSignals(true);
1261     itemrotatex->blockSignals(true);
1262     itemrotatey->blockSignals(true);
1263     itemrotatez->blockSignals(true);
1264
1265     Transform t = m_transformations.value(i);
1266
1267     if (!i->data(ZOOMFACTOR).isNull()) itemzoom->setValue(i->data(ZOOMFACTOR).toInt());
1268     else itemzoom->setValue((int)(t.scalex * 100.0 + 0.5));
1269
1270     itemrotatex->setValue((int)(t.rotatex));
1271     itemrotatey->setValue((int)(t.rotatey));
1272     itemrotatez->setValue((int)(t.rotatez));
1273
1274     itemzoom->blockSignals(false);
1275     itemrotatex->blockSignals(false);
1276     itemrotatey->blockSignals(false);
1277     itemrotatez->blockSignals(false);
1278 }
1279
1280 void TitleWidget::updatePosition(QGraphicsItem *i)
1281 {
1282     updatePosition(i, value_x->value(), value_y->value());
1283 }
1284
1285 void TitleWidget::updatePosition(QGraphicsItem *i, int x, int y)
1286 {
1287     if (i->type() == TEXTITEM) {
1288         QGraphicsTextItem *rec = static_cast <QGraphicsTextItem *>(i);
1289
1290         int posX;
1291         if (origin_x_left->isChecked()) {
1292             /*
1293              * Origin of the X axis is at m_frameWidth, and distance from right
1294              * border of the item to the right border of the frame is taken. See
1295              * comment to slotOriginXClicked().
1296              */
1297             posX = m_frameWidth - x - rec->boundingRect().width();
1298         } else {
1299             posX = x;
1300         }
1301
1302         int posY;
1303         if (origin_y_top->isChecked()) {
1304             /* Same for y axis */
1305             posY = m_frameHeight - y - rec->boundingRect().height();
1306         } else {
1307             posY = y;
1308         }
1309
1310         rec->setPos(posX, posY);
1311
1312     } else if (i->type() == RECTITEM) {
1313
1314         QGraphicsRectItem *rec = static_cast <QGraphicsRectItem *>(i);
1315
1316         int posX;
1317         if (origin_x_left->isChecked()) {
1318             posX = m_frameWidth - x - rec->rect().width();
1319         } else {
1320             posX = x;
1321         }
1322
1323         int posY;
1324         if (origin_y_top->isChecked()) {
1325             posY = m_frameHeight - y - rec->rect().height();
1326         } else {
1327             posY = y;
1328         }
1329
1330         rec->setPos(posX, posY);
1331
1332     } else if (i->type() == IMAGEITEM) {
1333         int posX;
1334         if (origin_x_left->isChecked()) {
1335             // Use the sceneBoundingRect because this also regards transformations like zoom
1336             posX = m_frameWidth - x - i->sceneBoundingRect().width();
1337         } else {
1338             posX = x;
1339         }
1340
1341         int posY;
1342         if (origin_y_top->isChecked()) {
1343             posY = m_frameHeight - y - i->sceneBoundingRect().height();
1344         } else {
1345             posY = y;
1346         }
1347
1348         i->setPos(posX, posY);
1349
1350     }
1351
1352 }
1353
1354 void TitleWidget::updateTextOriginX()
1355 {
1356     if (origin_x_left->isChecked()) {
1357         origin_x_left->setText(i18n("\u2212X"));
1358     } else {
1359         origin_x_left->setText(i18n("+X"));
1360     }
1361 }
1362
1363 void TitleWidget::slotOriginXClicked()
1364 {
1365     // Update the text displayed on the button.
1366     updateTextOriginX();
1367
1368     QList<QGraphicsItem*> l = graphicsView->scene()->selectedItems();
1369     if (l.size() >= 1) {
1370         updateCoordinates(l.at(0));
1371
1372         // Remember x axis setting
1373         l.at(0)->setData(TitleDocument::OriginXLeft, origin_x_left->isChecked() ?
1374                          TitleDocument::AxisInverted : TitleDocument::AxisDefault);
1375     }
1376     graphicsView->setFocus();
1377 }
1378
1379 void TitleWidget::updateTextOriginY()
1380 {
1381     if (origin_y_top->isChecked()) {
1382         origin_y_top->setText(i18n("\u2212Y"));
1383     } else {
1384         origin_y_top->setText(i18n("+Y"));
1385     }
1386 }
1387
1388 void TitleWidget::slotOriginYClicked()
1389 {
1390     // Update the text displayed on the button.
1391     updateTextOriginY();
1392
1393     QList<QGraphicsItem*> l = graphicsView->scene()->selectedItems();
1394     if (l.size() >= 1) {
1395         updateCoordinates(l.at(0));
1396
1397         l.at(0)->setData(TitleDocument::OriginYTop, origin_y_top->isChecked() ?
1398                          TitleDocument::AxisInverted : TitleDocument::AxisDefault);
1399
1400     }
1401     graphicsView->setFocus();
1402 }
1403
1404 void TitleWidget::updateAxisButtons(QGraphicsItem *i)
1405 {
1406     int xAxis = i->data(TitleDocument::OriginXLeft).toInt();
1407     int yAxis = i->data(TitleDocument::OriginYTop).toInt();
1408     origin_x_left->blockSignals(true);
1409     origin_y_top->blockSignals(true);
1410
1411     if (xAxis == TitleDocument::AxisInverted) {
1412         origin_x_left->setChecked(true);
1413     } else {
1414         origin_x_left->setChecked(false);
1415     }
1416     updateTextOriginX();
1417
1418     if (yAxis == TitleDocument::AxisInverted) {
1419         origin_y_top->setChecked(true);
1420     } else {
1421         origin_y_top->setChecked(false);
1422     }
1423     updateTextOriginY();
1424
1425     origin_x_left->blockSignals(false);
1426     origin_y_top->blockSignals(false);
1427 }
1428
1429 void TitleWidget::slotChangeBackground()
1430 {
1431     QColor color = backgroundColor->color();
1432     color.setAlpha(backgroundAlpha->value());
1433     m_frameBorder->setBrush(QBrush(color));
1434 }
1435
1436 void TitleWidget::slotChanged()
1437 {
1438     QList<QGraphicsItem*> l = graphicsView->scene()->selectedItems();
1439     if (l.size() >= 1 && l.at(0)->type() == TEXTITEM) {
1440         textChanged(static_cast <QGraphicsTextItem *>(l.at(0)));
1441     }
1442 }
1443
1444 void TitleWidget::textChanged(QGraphicsTextItem *i)
1445 {
1446     /*
1447      * If the user has set origin_x_left (the same for y), we need to look
1448      * whether a text element has been selected. If yes, we need to ensure that
1449      * the right border of the text field remains fixed also when some text has
1450      * been entered.
1451      *
1452      * This is also known as right-justified, with the difference that it is not
1453      * valid for text but for its boundingRect. Text may still be
1454      * left-justified.
1455      */
1456     updateDimension(i);
1457
1458     if (origin_x_left->isChecked() || origin_y_top->isChecked()) {
1459         if (!i->toPlainText().isEmpty())
1460             updatePosition(i);
1461         else {
1462             /*
1463              * Don't do anything if the string is empty. If the position were
1464              * updated here, a newly created text field would be set to the
1465              * position of the last selected text field.
1466              */
1467         }
1468     }
1469
1470     // mbt 1607: Template text has changed; don't auto-select content anymore.
1471     if (i->property("isTemplate").isValid()) {
1472         if (i->property("templateText").isValid()) {
1473             if (i->property("templateText") == i->toHtml()) {
1474                 // Unchanged, do nothing.
1475             } else {
1476                 i->setProperty("isTemplate", QVariant::Invalid);
1477                 i->setProperty("templateText", QVariant::Invalid);
1478             }
1479         }
1480     }
1481 }
1482
1483 void TitleWidget::slotInsertUnicode()
1484 {
1485     m_unicodeDialog->exec();
1486 }
1487
1488 void TitleWidget::slotInsertUnicodeString(QString text)
1489 {
1490     QList<QGraphicsItem *> l = graphicsView->scene()->selectedItems();
1491     if (l.size() > 0) {
1492         if (l.at(0)->type() == TEXTITEM) {
1493             QGraphicsTextItem *t = static_cast <QGraphicsTextItem *>(l.at(0));
1494             t->textCursor().insertText(text);
1495         }
1496     }
1497 }
1498
1499 void TitleWidget::slotUpdateText()
1500 {
1501     QFont font = font_family->currentFont();
1502     font.setPixelSize(font_size->value());
1503     font.setItalic(buttonItalic->isChecked());
1504     font.setUnderline(buttonUnder->isChecked());
1505     font.setWeight(font_weight_box->itemData(font_weight_box->currentIndex()).toInt());
1506     QColor color = fontColorButton->color();
1507     color.setAlpha(textAlpha->value());
1508
1509     QColor outlineColor = textOutlineColor->color();
1510     outlineColor.setAlpha(textOutlineAlpha->value());
1511     double outlineWidth = textOutline->value() / 10.0;
1512
1513     int i;
1514     for (i = 0; i < graphicsView->scene()->selectedItems().length(); i++) {
1515         QGraphicsTextItem* item = NULL;
1516         QList<QGraphicsItem*> l = graphicsView->scene()->selectedItems();
1517         if (l.at(i)->type() == TEXTITEM) {
1518             item = static_cast <QGraphicsTextItem *>(l.at(i));
1519         }
1520         if (!item) {
1521             // No text item, try next one.
1522             continue;
1523         }
1524
1525         // Set alignment of all text in the text item
1526         QTextCursor cur(item->document());
1527         cur.select(QTextCursor::Document);
1528         QTextBlockFormat format = cur.blockFormat();
1529         if (buttonAlignLeft->isChecked() || buttonAlignCenter->isChecked() || buttonAlignRight->isChecked()) {
1530             item->setTextWidth(item->boundingRect().width());
1531             if (buttonAlignCenter->isChecked()) format.setAlignment(Qt::AlignHCenter);
1532             else if (buttonAlignRight->isChecked()) format.setAlignment(Qt::AlignRight);
1533             else if (buttonAlignLeft->isChecked()) format.setAlignment(Qt::AlignLeft);
1534         } else {
1535             format.setAlignment(Qt::AlignLeft);
1536             item->setTextWidth(-1);
1537         }
1538
1539         // Set font properties
1540         item->setFont(font);
1541         QTextCharFormat cformat = cur.charFormat();
1542
1543         item->setData(101, outlineWidth);
1544         item->setData(102, outlineColor);
1545         if (outlineWidth > 0.0) cformat.setTextOutline(QPen(outlineColor, outlineWidth, Qt::SolidLine, Qt::RoundCap, Qt::RoundJoin));
1546
1547         cformat.setForeground(QBrush(color));
1548         cur.setCharFormat(cformat);
1549         cur.setBlockFormat(format);
1550 //  item->setTextCursor(cur);
1551         cur.clearSelection();
1552         item->setTextCursor(cur);
1553         item->setDefaultTextColor(color);
1554     }
1555 }
1556
1557 void TitleWidget::rectChanged()
1558 {
1559     QList<QGraphicsItem*> l = graphicsView->scene()->selectedItems();
1560     int i;
1561     for (i = 0; i < l.length(); i++) {
1562         if (l.at(i)->type() == RECTITEM && !settingUp) {
1563             QGraphicsRectItem *rec = static_cast<QGraphicsRectItem *>(l.at(i));
1564             QColor f = rectFColor->color();
1565             f.setAlpha(rectFAlpha->value());
1566             QPen penf(f);
1567             penf.setWidth(rectLineWidth->value());
1568             penf.setJoinStyle(Qt::RoundJoin);
1569             rec->setPen(penf);
1570             QColor b = rectBColor->color();
1571             b.setAlpha(rectBAlpha->value());
1572             rec->setBrush(QBrush(b));
1573         }
1574     }
1575 }
1576
1577 void TitleWidget::itemScaled(int val)
1578 {
1579     QList<QGraphicsItem*> l = graphicsView->scene()->selectedItems();
1580     if (l.size() == 1) {
1581         Transform x = m_transformations.value(l.at(0));
1582         x.scalex = (double)val / 100.0;
1583         x.scaley = (double)val / 100.0;
1584         QTransform qtrans;
1585         qtrans.scale(x.scalex, x.scaley);
1586         qtrans.rotate(x.rotatex, Qt::XAxis);
1587         qtrans.rotate(x.rotatey, Qt::YAxis);
1588         qtrans.rotate(x.rotatez, Qt::ZAxis);
1589         l[0]->setTransform(qtrans);
1590         l[0]->setData(ZOOMFACTOR, val);
1591         m_transformations[l.at(0)] = x;
1592         updateDimension(l.at(0));
1593     }
1594 }
1595
1596 void TitleWidget::itemRotateX(qreal val)
1597 {
1598     itemRotate(val, 0);
1599 }
1600
1601 void TitleWidget::itemRotateY(qreal val)
1602 {
1603     itemRotate(val, 1);
1604 }
1605
1606 void TitleWidget::itemRotateZ(qreal val)
1607 {
1608     itemRotate(val, 2);
1609 }
1610
1611 void TitleWidget::itemRotate(qreal val, int axis)
1612 {
1613     QList<QGraphicsItem*> l = graphicsView->scene()->selectedItems();
1614     if (l.size() == 1) {
1615         Transform x = m_transformations[l.at(0)];
1616         switch (axis) {
1617         case 0:
1618             x.rotatex = val;
1619             break;
1620         case 1:
1621             x.rotatey = val;
1622             break;
1623         case 2:
1624             x.rotatez = val;
1625             break;
1626         }
1627
1628         l[0]->setData(ROTATEFACTOR, QList<QVariant>() << QVariant(x.rotatex) << QVariant(x.rotatey) << QVariant(x.rotatez));
1629
1630         QTransform qtrans;
1631         qtrans.scale(x.scalex, x.scaley);
1632         qtrans.rotate(x.rotatex, Qt::XAxis);
1633         qtrans.rotate(x.rotatey, Qt::YAxis);
1634         qtrans.rotate(x.rotatez, Qt::ZAxis);
1635         l[0]->setTransform(qtrans);
1636         m_transformations[l.at(0)] = x;
1637         if (l[0]->data(ZOOMFACTOR).isNull()) l[0]->setData(ZOOMFACTOR, 100);
1638         updateDimension(l.at(0));
1639     }
1640 }
1641
1642 void TitleWidget::itemHCenter()
1643 {
1644     QList<QGraphicsItem*> l = graphicsView->scene()->selectedItems();
1645     if (l.size() == 1) {
1646         QGraphicsItem *item = l.at(0);
1647         QRectF br = item->sceneBoundingRect();
1648         int width = (int)br.width();
1649         int newPos = (int)((m_frameWidth - width) / 2);
1650         newPos += item->pos().x() - br.left(); // Check item transformation
1651         item->setPos(newPos, item->pos().y());
1652         updateCoordinates(item);
1653     }
1654 }
1655
1656 void TitleWidget::itemVCenter()
1657 {
1658     QList<QGraphicsItem*> l = graphicsView->scene()->selectedItems();
1659     if (l.size() == 1) {
1660         QGraphicsItem *item = l.at(0);
1661         QRectF br = item->sceneBoundingRect();
1662         int height = (int)br.height();
1663         int newPos = (int)((m_frameHeight - height) / 2);
1664         newPos += item->pos().y() - br.top(); // Check item transformation
1665         item->setPos(item->pos().x(), newPos);
1666         updateCoordinates(item);
1667     }
1668 }
1669
1670 void TitleWidget::itemTop()
1671 {
1672     QList<QGraphicsItem*> l = graphicsView->scene()->selectedItems();
1673     if (l.size() == 1) {
1674         QGraphicsItem *item = l.at(0);
1675         QRectF br = item->sceneBoundingRect();
1676         double diff;
1677         if (br.top() > 0) diff = -br.top();
1678         else diff = -br.bottom();
1679         item->moveBy(0, diff);
1680         updateCoordinates(item);
1681     }
1682 }
1683
1684 void TitleWidget::itemBottom()
1685 {
1686     QList<QGraphicsItem*> l = graphicsView->scene()->selectedItems();
1687     if (l.size() == 1) {
1688         QGraphicsItem *item = l.at(0);
1689         QRectF br = item->sceneBoundingRect();
1690         double diff;
1691         if (br.bottom() > m_frameHeight) diff = m_frameHeight - br.top();
1692         else diff = m_frameHeight - br.bottom();
1693         item->moveBy(0, diff);
1694         updateCoordinates(item);
1695     }
1696 }
1697
1698 void TitleWidget::itemLeft()
1699 {
1700     QList<QGraphicsItem*> l = graphicsView->scene()->selectedItems();
1701     if (l.size() == 1) {
1702         QGraphicsItem *item = l.at(0);
1703         QRectF br = item->sceneBoundingRect();
1704         double diff;
1705         if (br.left() > 0) diff = -br.left();
1706         else diff = -br.right();
1707         item->moveBy(diff, 0);
1708         updateCoordinates(item);
1709     }
1710 }
1711
1712 void TitleWidget::itemRight()
1713 {
1714     QList<QGraphicsItem*> l = graphicsView->scene()->selectedItems();
1715     if (l.size() == 1) {
1716         QGraphicsItem *item = l.at(0);
1717         QRectF br = item->sceneBoundingRect();
1718         double diff;
1719         if (br.right() < m_frameWidth) diff = m_frameWidth - br.right();
1720         else diff = m_frameWidth - br.left();
1721         item->moveBy(diff, 0);
1722         updateCoordinates(item);
1723     }
1724 }
1725
1726 void TitleWidget::setupViewports()
1727 {
1728     //double aspect_ratio = 4.0 / 3.0;//read from project
1729     //better zoom centered, but render uses only the created rect, so no problem to change the zoom function
1730     /*QRectF sp(0, 0, startViewportSize->value() * m_frameWidth / 100.0 , startViewportSize->value()* m_frameHeight / 100.0);
1731     QRectF ep(0, 0, endViewportSize->value() * m_frameWidth / 100.0, endViewportSize->value() * m_frameHeight / 100.0);
1732     // use a polygon thiat uses 16:9 and 4:3 rects forpreview the size in all aspect ratios ?
1733     QPolygonF spoly(sp);
1734     QPolygonF epoly(ep);
1735     spoly.translate(startViewportX->value(), startViewportY->value());
1736     epoly.translate(endViewportX->value(), endViewportY->value());
1737     m_startViewport->setPolygon(spoly);
1738     m_endViewport->setPolygon(epoly);
1739     if (! insertingValues) {
1740         m_startViewport->setData(0, startViewportX->value());
1741         m_startViewport->setData(1, startViewportY->value());
1742         m_startViewport->setData(2, startViewportSize->value());
1743
1744         m_endViewport->setData(0, endViewportX->value());
1745         m_endViewport->setData(1, endViewportY->value());
1746         m_endViewport->setData(2, endViewportSize->value());
1747     }*/
1748 }
1749
1750 void TitleWidget::loadTitle(KUrl url)
1751 {
1752     if (url.isEmpty()) url = KFileDialog::getOpenUrl(KUrl(m_projectTitlePath), "application/x-kdenlivetitle", this, i18n("Load Title"));
1753     if (!url.isEmpty()) {
1754         QList<QGraphicsItem *> items = m_scene->items();
1755         for (int i = 0; i < items.size(); i++) {
1756             if (items.at(i)->zValue() > -1000) delete items.at(i);
1757         }
1758         m_scene->clearTextSelection();
1759         QDomDocument doc;
1760         QString tmpfile;
1761
1762         if (KIO::NetAccess::download(url, tmpfile, 0)) {
1763             QFile file(tmpfile);
1764             if (file.open(QIODevice::ReadOnly)) {
1765                 doc.setContent(&file, false);
1766                 file.close();
1767             } else return;
1768             KIO::NetAccess::removeTempFile(tmpfile);
1769         }
1770         setXml(doc);
1771
1772         /*int out;
1773         m_count = m_titledocument.loadDocument(url, m_startViewport, m_endViewport, &out) + 1;
1774         adjustFrameSize();
1775         title_duration->setText(m_tc.getTimecode(GenTime(out, m_render->fps())));
1776         insertingValues = true;
1777         startViewportX->setValue(m_startViewport->data(0).toInt());
1778         startViewportY->setValue(m_startViewport->data(1).toInt());
1779         startViewportSize->setValue(m_startViewport->data(2).toInt());
1780         endViewportX->setValue(m_endViewport->data(0).toInt());
1781         endViewportY->setValue(m_endViewport->data(1).toInt());
1782         endViewportSize->setValue(m_endViewport->data(2).toInt());
1783
1784         insertingValues = false;
1785         slotSelectTool();
1786         slotAdjustZoom();*/
1787     }
1788 }
1789
1790 void TitleWidget::saveTitle(KUrl url)
1791 {
1792     if (anim_start->isChecked()) slotAnimStart(false);
1793     if (anim_end->isChecked()) slotAnimEnd(false);
1794     bool embed_image=false;
1795     if (KMessageBox::questionYesNo(this, i18n("Do you want to embed Images into this TitleDocument?\nThis is most needed for sharing Titles.")) != KMessageBox::No)
1796     {
1797         embed_image=true;       
1798     }
1799     if (url.isEmpty()) {
1800         KFileDialog *fs = new KFileDialog(KUrl(m_projectTitlePath), "application/x-kdenlivetitle", this);
1801         fs->setOperationMode(KFileDialog::Saving);
1802         fs->setMode(KFile::File);
1803         fs->setConfirmOverwrite(true);
1804         fs->setKeepLocation(true);
1805         fs->exec();
1806         url = fs->selectedUrl();
1807         delete fs;
1808     }
1809     if (!url.isEmpty()) {
1810         if (m_titledocument.saveDocument(url, m_startViewport, m_endViewport, m_tc.getFrameCount(title_duration->text()), embed_image) == false)
1811             KMessageBox::error(this, i18n("Cannot write to file %1", url.path()));
1812     }
1813 }
1814
1815 QDomDocument TitleWidget::xml()
1816 {
1817     QDomDocument doc = m_titledocument.xml(m_startViewport, m_endViewport);
1818     doc.documentElement().setAttribute("out", m_tc.getFrameCount(title_duration->text()));
1819     return doc;
1820 }
1821
1822 int TitleWidget::outPoint() const
1823 {
1824     return m_tc.getFrameCount(title_duration->text());
1825 }
1826
1827 void TitleWidget::setXml(QDomDocument doc)
1828 {
1829     int out;
1830     m_count = m_titledocument.loadFromXml(doc, m_startViewport, m_endViewport, &out, m_projectTitlePath);
1831     adjustFrameSize();
1832     title_duration->setText(m_tc.getTimecode(GenTime(out, m_render->fps())));
1833     /*if (doc.documentElement().hasAttribute("out")) {
1834     GenTime duration = GenTime(doc.documentElement().attribute("out").toDouble() / 1000.0);
1835     title_duration->setText(m_tc.getTimecode(duration));
1836     }
1837     else title_duration->setText(m_tc.getTimecode(GenTime(5000)));*/
1838
1839     QDomElement e = doc.documentElement();
1840     m_transformations.clear();
1841     QList <QGraphicsItem *> items = graphicsView->scene()->items();
1842     const double PI = 4.0 * atan(1.0);
1843     for (int i = 0; i < items.count(); i++) {
1844         QTransform t = items.at(i)->transform();
1845         Transform x;
1846         x.scalex = t.m11();
1847         x.scaley = t.m22();
1848         if (!items.at(i)->data(ROTATEFACTOR).isNull()) {
1849             QList<QVariant> rotlist = items.at(i)->data(ROTATEFACTOR).toList();
1850             if (rotlist.count() >= 3) {
1851                 x.rotatex = rotlist[0].toDouble();
1852                 x.rotatey = rotlist[1].toDouble();
1853                 x.rotatez = rotlist[2].toDouble();
1854
1855                 // Try to adjust zoom
1856                 t.rotate(x.rotatex *(-1), Qt::XAxis);
1857                 t.rotate(x.rotatey *(-1), Qt::YAxis);
1858                 t.rotate(x.rotatez *(-1), Qt::ZAxis);
1859                 x.scalex = t.m11();
1860                 x.scaley = t.m22();
1861             } else {
1862                 x.rotatex = 0;
1863                 x.rotatey = 0;
1864                 x.rotatez = 0;
1865             }
1866         } else {
1867             x.rotatex = 0;
1868             x.rotatey = 0;
1869             x.rotatez = 180. / PI * atan2(-t.m21(), t.m11());
1870         }
1871         m_transformations[items.at(i)] = x;
1872     }
1873     // mbd: Update the GUI color selectors to match the stuff from the loaded document
1874     QColor background_color = m_titledocument.getBackgroundColor();
1875     backgroundAlpha->blockSignals(true);
1876     backgroundColor->blockSignals(true);
1877     backgroundAlpha->setValue(background_color.alpha());
1878     background_color.setAlpha(255);
1879     backgroundColor->setColor(background_color);
1880     backgroundAlpha->blockSignals(false);
1881     backgroundColor->blockSignals(false);
1882
1883     /*startViewportX->setValue(m_startViewport->data(0).toInt());
1884     startViewportY->setValue(m_startViewport->data(1).toInt());
1885     startViewportSize->setValue(m_startViewport->data(2).toInt());
1886     endViewportX->setValue(m_endViewport->data(0).toInt());
1887     endViewportY->setValue(m_endViewport->data(1).toInt());
1888     endViewportSize->setValue(m_endViewport->data(2).toInt());*/
1889
1890     QTimer::singleShot(200, this, SLOT(slotAdjustZoom()));
1891     slotSelectTool();
1892     selectionChanged();
1893 }
1894
1895 void TitleWidget::slotAccepted()
1896 {
1897     if (anim_start->isChecked()) slotAnimStart(false);
1898     if (anim_end->isChecked()) slotAnimEnd(false);
1899     writeChoices();
1900 }
1901
1902 void TitleWidget::writeChoices()
1903 {
1904     // Get a pointer to a shared configuration instance, then get the TitleWidget group.
1905     KSharedConfigPtr config = KGlobal::config();
1906     KConfigGroup titleConfig(config, "TitleWidget");
1907     // Write the entries
1908     titleConfig.writeEntry("font_family", font_family->currentFont());
1909     //titleConfig.writeEntry("font_size", font_size->value());
1910     titleConfig.writeEntry("font_pixel_size", font_size->value());
1911     titleConfig.writeEntry("font_color", fontColorButton->color());
1912     titleConfig.writeEntry("font_alpha", textAlpha->value());
1913     titleConfig.writeEntry("font_outline", textOutline->value());
1914     titleConfig.writeEntry("font_outline_color", textOutlineColor->color());
1915     titleConfig.writeEntry("font_outline_alpha", textOutlineAlpha->value());
1916     titleConfig.writeEntry("font_weight", font_weight_box->itemData(font_weight_box->currentIndex()).toInt());
1917     titleConfig.writeEntry("font_italic", buttonItalic->isChecked());
1918     titleConfig.writeEntry("font_underlined", buttonUnder->isChecked());
1919
1920     titleConfig.writeEntry("rect_foreground_color", rectFColor->color());
1921     titleConfig.writeEntry("rect_foreground_alpha", rectFAlpha->value());
1922     titleConfig.writeEntry("rect_background_color", rectBColor->color());
1923     titleConfig.writeEntry("rect_background_alpha", rectBAlpha->value());
1924     titleConfig.writeEntry("rect_line_width", rectLineWidth->value());
1925
1926     titleConfig.writeEntry("background_color", backgroundColor->color());
1927     titleConfig.writeEntry("background_alpha", backgroundAlpha->value());
1928
1929     //! \todo Not sure if I should sync - it is probably safe to do it
1930     config->sync();
1931
1932 }
1933
1934 void TitleWidget::readChoices()
1935 {
1936     // Get a pointer to a shared configuration instance, then get the TitleWidget group.
1937     KSharedConfigPtr config = KGlobal::config();
1938     KConfigGroup titleConfig(config, "TitleWidget");
1939     // read the entries
1940     font_family->setCurrentFont(titleConfig.readEntry("font_family", font_family->currentFont()));
1941     font_size->setValue(titleConfig.readEntry("font_pixel_size", font_size->value()));
1942     m_scene->slotUpdateFontSize(font_size->value());
1943     fontColorButton->setColor(titleConfig.readEntry("font_color", fontColorButton->color()));
1944     textAlpha->setValue(titleConfig.readEntry("font_alpha", textAlpha->value()));
1945
1946     textOutlineColor->setColor(titleConfig.readEntry("font_outline_color", textOutlineColor->color()));
1947     textOutlineAlpha->setValue(titleConfig.readEntry("font_outline_alpha", textOutlineAlpha->value()));
1948     textOutline->setValue(titleConfig.readEntry("font_outline", textOutline->value()));
1949
1950     int weight;
1951     if (titleConfig.readEntry("font_bold", false)) weight = QFont::Bold;
1952     else weight = titleConfig.readEntry("font_weight", font_weight_box->itemData(font_weight_box->currentIndex()).toInt());
1953     setFontBoxWeight(weight);
1954     buttonItalic->setChecked(titleConfig.readEntry("font_italic", buttonItalic->isChecked()));
1955     buttonUnder->setChecked(titleConfig.readEntry("font_underlined", buttonUnder->isChecked()));
1956
1957     rectFColor->setColor(titleConfig.readEntry("rect_foreground_color", rectFColor->color()));
1958     rectFAlpha->setValue(titleConfig.readEntry("rect_foreground_alpha", rectFAlpha->value()));
1959     rectBColor->setColor(titleConfig.readEntry("rect_background_color", rectBColor->color()));
1960     rectBAlpha->setValue(titleConfig.readEntry("rect_background_alpha", rectBAlpha->value()));
1961     rectLineWidth->setValue(titleConfig.readEntry("rect_line_width", rectLineWidth->value()));
1962
1963     backgroundColor->setColor(titleConfig.readEntry("background_color", backgroundColor->color()));
1964     backgroundAlpha->setValue(titleConfig.readEntry("background_alpha", backgroundAlpha->value()));
1965 }
1966
1967 void TitleWidget::adjustFrameSize()
1968 {
1969     m_frameWidth = m_titledocument.frameWidth();
1970     m_frameHeight = m_titledocument.frameHeight();
1971     m_frameBorder->setRect(0, 0, m_frameWidth, m_frameHeight);
1972     displayBackgroundFrame();
1973 }
1974
1975 void TitleWidget::slotAnimStart(bool anim)
1976 {
1977     if (anim && anim_end->isChecked()) {
1978         anim_end->setChecked(false);
1979         m_endViewport->setZValue(-1000);
1980         m_endViewport->setBrush(QBrush());
1981     }
1982     slotSelectTool();
1983     QList<QGraphicsItem *> list = m_scene->items();
1984     for (int i = 0; i < list.count(); i++) {
1985         if (list.at(i)->zValue() > -1000) {
1986             list.at(i)->setFlag(QGraphicsItem::ItemIsMovable, !anim);
1987             list.at(i)->setFlag(QGraphicsItem::ItemIsSelectable, !anim);
1988         }
1989     }
1990     align_box->setEnabled(anim);
1991     itemzoom->setEnabled(!anim);
1992     itemrotatex->setEnabled(!anim);
1993     itemrotatey->setEnabled(!anim);
1994     itemrotatez->setEnabled(!anim);
1995     frame_toolbar->setEnabled(!anim);
1996     toolbar_stack->setEnabled(!anim);
1997     if (anim) {
1998         keep_aspect->setChecked(!m_startViewport->data(0).isNull());
1999         m_startViewport->setZValue(1100);
2000         QColor col = m_startViewport->pen().color();
2001         col.setAlpha(100);
2002         m_startViewport->setBrush(col);
2003         m_startViewport->setFlags(QGraphicsItem::ItemIsMovable | QGraphicsItem::ItemIsSelectable);
2004         m_startViewport->setSelected(true);
2005         selectionChanged();
2006         slotSelectTool();
2007         if (m_startViewport->childItems().isEmpty()) addAnimInfoText();
2008     } else {
2009         m_startViewport->setZValue(-1000);
2010         m_startViewport->setBrush(QBrush());
2011         m_startViewport->setFlags(0);
2012         if (!anim_end->isChecked()) deleteAnimInfoText();
2013     }
2014
2015 }
2016
2017 void TitleWidget::slotAnimEnd(bool anim)
2018 {
2019     if (anim && anim_start->isChecked()) {
2020         anim_start->setChecked(false);
2021         m_startViewport->setZValue(-1000);
2022         m_startViewport->setBrush(QBrush());
2023     }
2024     slotSelectTool();
2025     QList<QGraphicsItem *> list = m_scene->items();
2026     for (int i = 0; i < list.count(); i++) {
2027         if (list.at(i)->zValue() > -1000) {
2028             list.at(i)->setFlag(QGraphicsItem::ItemIsMovable, !anim);
2029             list.at(i)->setFlag(QGraphicsItem::ItemIsSelectable, !anim);
2030         }
2031     }
2032     align_box->setEnabled(anim);
2033     itemzoom->setEnabled(!anim);
2034     itemrotatex->setEnabled(!anim);
2035     itemrotatey->setEnabled(!anim);
2036     itemrotatez->setEnabled(!anim);
2037     frame_toolbar->setEnabled(!anim);
2038     toolbar_stack->setEnabled(!anim);
2039     if (anim) {
2040         keep_aspect->setChecked(!m_endViewport->data(0).isNull());
2041         m_endViewport->setZValue(1100);
2042         QColor col = m_endViewport->pen().color();
2043         col.setAlpha(100);
2044         m_endViewport->setBrush(col);
2045         m_endViewport->setFlags(QGraphicsItem::ItemIsMovable | QGraphicsItem::ItemIsSelectable);
2046         m_endViewport->setSelected(true);
2047         selectionChanged();
2048         slotSelectTool();
2049         if (m_endViewport->childItems().isEmpty()) addAnimInfoText();
2050     } else {
2051         m_endViewport->setZValue(-1000);
2052         m_endViewport->setBrush(QBrush());
2053         m_endViewport->setFlags(0);
2054         if (!anim_start->isChecked()) deleteAnimInfoText();
2055     }
2056 }
2057
2058 void TitleWidget::addAnimInfoText()
2059 {
2060     // add text to anim viewport
2061     QGraphicsTextItem *t = new QGraphicsTextItem(i18n("Start"), m_startViewport);
2062     QGraphicsTextItem *t2 = new QGraphicsTextItem(i18n("End"), m_endViewport);
2063     QFont font = t->font();
2064     font.setPixelSize(m_startViewport->rect().width() / 10);
2065     QColor col = m_startViewport->pen().color();
2066     col.setAlpha(255);
2067     t->setDefaultTextColor(col);
2068     t->setFont(font);
2069     font.setPixelSize(m_endViewport->rect().width() / 10);
2070     col = m_endViewport->pen().color();
2071     col.setAlpha(255);
2072     t2->setDefaultTextColor(col);
2073     t2->setFont(font);
2074 }
2075
2076 void TitleWidget::updateInfoText()
2077 {
2078     // update info text font
2079     if (!m_startViewport->childItems().isEmpty()) {
2080         QGraphicsTextItem *item = static_cast <QGraphicsTextItem *>(m_startViewport->childItems().at(0));
2081         if (item) {
2082             QFont font = item->font();
2083             font.setPixelSize(m_startViewport->rect().width() / 10);
2084             item->setFont(font);
2085         }
2086     }
2087     if (!m_endViewport->childItems().isEmpty()) {
2088         QGraphicsTextItem *item = static_cast <QGraphicsTextItem *>(m_endViewport->childItems().at(0));
2089         if (item) {
2090             QFont font = item->font();
2091             font.setPixelSize(m_endViewport->rect().width() / 10);
2092             item->setFont(font);
2093         }
2094     }
2095 }
2096
2097 void TitleWidget::deleteAnimInfoText()
2098 {
2099     // end animation editing, remove info text
2100     while (!m_startViewport->childItems().isEmpty()) {
2101         QGraphicsItem *item = m_startViewport->childItems().at(0);
2102         m_scene->removeItem(item);
2103         delete item;
2104     }
2105     while (!m_endViewport->childItems().isEmpty()) {
2106         QGraphicsItem *item = m_endViewport->childItems().at(0);
2107         m_scene->removeItem(item);
2108         delete item;
2109     }
2110 }
2111
2112 void TitleWidget::slotKeepAspect(bool keep)
2113 {
2114     if (m_endViewport->zValue() == 1100) {
2115         m_endViewport->setData(0, keep == true ? m_frameWidth : QVariant());
2116         m_endViewport->setData(1, keep == true ? m_frameHeight : QVariant());
2117     } else {
2118         m_startViewport->setData(0, keep == true ? m_frameWidth : QVariant());
2119         m_startViewport->setData(1, keep == true ? m_frameHeight : QVariant());
2120     }
2121 }
2122
2123 void TitleWidget::slotResize50()
2124 {
2125     if (m_endViewport->zValue() == 1100) {
2126         m_endViewport->setRect(0, 0, m_frameWidth / 2, m_frameHeight / 2);
2127     } else m_startViewport->setRect(0, 0, m_frameWidth / 2, m_frameHeight / 2);
2128 }
2129
2130 void TitleWidget::slotResize100()
2131 {
2132     if (m_endViewport->zValue() == 1100) {
2133         m_endViewport->setRect(0, 0, m_frameWidth, m_frameHeight);
2134     } else m_startViewport->setRect(0, 0, m_frameWidth, m_frameHeight);
2135 }
2136
2137 void TitleWidget::slotResize200()
2138 {
2139     if (m_endViewport->zValue() == 1100) {
2140         m_endViewport->setRect(0, 0, m_frameWidth * 2, m_frameHeight * 2);
2141     } else m_startViewport->setRect(0, 0, m_frameWidth * 2, m_frameHeight * 2);
2142 }
2143
2144 void TitleWidget::slotAddEffect(int ix)
2145 {
2146     QList<QGraphicsItem *> list = graphicsView->scene()->selectedItems();
2147     int effect = effect_list->itemData(ix).toInt();
2148
2149     if (list.size() == 1) {
2150         if (effect == NOEFFECT)
2151             effect_stack->setHidden(true);
2152         else {
2153             effect_stack->setCurrentIndex(effect - 1);
2154             effect_stack->setHidden(false);
2155         }
2156     } else // Hide the effects stack when more than one element is selected.
2157         effect_stack->setHidden(true);
2158
2159     foreach(QGraphicsItem * item, list) {
2160         switch (effect) {
2161         case NOEFFECT:
2162             item->setData(100, QVariant());
2163 #if QT_VERSION >= 0x040600
2164             item->setGraphicsEffect(0);
2165 #endif
2166             break;
2167         case TYPEWRITEREFFECT:
2168             /*
2169              * Allow the user to set the typewriter effect to more than one
2170              * element, but do not add it to non-text elements.
2171              */
2172             if (item->type() == TEXTITEM) {
2173                 QStringList effdata = QStringList() << "typewriter" << QString::number(typewriter_delay->value()) + ";" + QString::number(typewriter_start->value());
2174                 item->setData(100, effdata);
2175             }
2176             break;
2177 #if QT_VERSION >= 0x040600
2178             // Do not remove the non-QGraphicsEffects.
2179         case BLUREFFECT:
2180             item->setGraphicsEffect(new QGraphicsBlurEffect());
2181             break;
2182         case SHADOWEFFECT:
2183             item->setGraphicsEffect(new QGraphicsDropShadowEffect());
2184             break;
2185 #endif
2186         }
2187     }
2188 }
2189
2190 void TitleWidget::slotFontText(const QString& s)
2191 {
2192     const QFont f(s);
2193     if (f.exactMatch()) {
2194         // Font really exists (could also just be a Â«d» if the user
2195         // starts typing Â«dejavu» for example).
2196         font_family->setCurrentFont(f);
2197     }
2198     // Note: Typing dejavu serif does not recognize the font (takes sans)
2199     // in older Qt versions. Case must match there (except for first letter)
2200 }
2201
2202 void TitleWidget::slotEditTypewriter(int /*ix*/)
2203 {
2204     QList<QGraphicsItem*> l = graphicsView->scene()->selectedItems();
2205     if (l.size() == 1) {
2206         QStringList effdata = QStringList() << "typewriter" << QString::number(typewriter_delay->value()) + ";" + QString::number(typewriter_start->value());
2207         l[0]->setData(100, effdata);
2208     }
2209 }
2210
2211 void TitleWidget::slotEditBlur(int ix)
2212 {
2213 #if QT_VERSION < 0x040600
2214     return;
2215 #else
2216     QList<QGraphicsItem*> l = graphicsView->scene()->selectedItems();
2217     if (l.size() == 1) {
2218         QGraphicsEffect *eff = l[0]->graphicsEffect();
2219         QGraphicsBlurEffect *blur = static_cast <QGraphicsBlurEffect *>(eff);
2220         if (blur) blur->setBlurRadius(ix);
2221     }
2222 #endif
2223 }
2224
2225 void TitleWidget::slotEditShadow()
2226 {
2227 #if QT_VERSION < 0x040600
2228     return;
2229 #else
2230     QList<QGraphicsItem*> l = graphicsView->scene()->selectedItems();
2231     if (l.size() == 1) {
2232         QGraphicsEffect *eff = l[0]->graphicsEffect();
2233         QGraphicsDropShadowEffect *shadow = static_cast <QGraphicsDropShadowEffect *>(eff);
2234         if (shadow) {
2235             shadow->setBlurRadius(shadow_radius->value());
2236             shadow->setOffset(shadow_x->value(), shadow_y->value());
2237         }
2238     }
2239 #endif
2240 }
2241
2242 qreal TitleWidget::zIndexBounds(bool maxBound, bool intersectingOnly)
2243 {
2244     qreal bound = maxBound ? -99 : 99;
2245     QList<QGraphicsItem*> l = graphicsView->scene()->selectedItems();
2246     if (l.size() > 0) {
2247         QList<QGraphicsItem*> lItems;
2248         // Get items (all or intersecting only)
2249         if (intersectingOnly) {
2250             lItems = graphicsView->scene()->items(l[0]->sceneBoundingRect(), Qt::IntersectsItemShape);
2251         } else {
2252             lItems = graphicsView->scene()->items();
2253         }
2254         if (lItems.size() > 0) {
2255             int n = lItems.size();
2256             qreal z;
2257             if (maxBound) {
2258                 for (int i = 0; i < n; i++) {
2259                     z = lItems[i]->zValue();
2260                     if (z > bound && !lItems[i]->isSelected()) {
2261                         bound = z;
2262                     } else if (z - 1 > bound) {
2263                         // To get the maximum index even if it is of an item of the current selection.
2264                         // Used when updating multiple items, to get all to the same level.
2265                         // Otherwise, the maximum would stay at -99 if the highest item is in the selection.
2266                         bound = z - 1;
2267                     }
2268                 }
2269             } else {
2270                 // Get minimum z index.
2271                 for (int i = 0; i < n; i++) {
2272                     z = lItems[i]->zValue();
2273                     if (z < bound && !lItems[i]->isSelected() && z > -999) {
2274                         // There are items at the very bottom (background e.g.) with z-index < -1000.
2275                         bound = z;
2276                     } else if (z + 1 < bound && z > -999) {
2277                         bound = z + 1;
2278                     }
2279                 }
2280             }
2281         }
2282     }
2283     return bound;
2284 }
2285
2286 void TitleWidget::slotZIndexUp()
2287 {
2288     QList<QGraphicsItem*> l = graphicsView->scene()->selectedItems();
2289     if (l.size() >= 1) {
2290         qreal currentZ = l[0]->zValue();
2291         qreal max = zIndexBounds(true, true);
2292         if (currentZ <= max) {
2293             l[0]->setZValue(currentZ + 1);
2294             updateDimension(l[0]);
2295         }
2296     }
2297 }
2298
2299 void TitleWidget::slotZIndexTop()
2300 {
2301     QList<QGraphicsItem*> l = graphicsView->scene()->selectedItems();
2302     qreal max = zIndexBounds(true, false);
2303     std::cout << "Max z-index is " << max << ".\n";
2304     for (int i = 0; i < l.size(); i++) {
2305         qreal currentZ = l[i]->zValue();
2306         if (currentZ <= max) {
2307             std::cout << "Updating item " << i << ", is " << currentZ << ".\n";
2308             l[i]->setZValue(max + 1);
2309         } else {
2310             std::cout << "Not updating " << i << ", is " << currentZ << ".\n";
2311         }
2312     }
2313     // Update the z index value in the GUI
2314     if (l.size() > 0) {
2315         updateDimension(l[0]);
2316     }
2317 }
2318
2319 void TitleWidget::slotZIndexDown()
2320 {
2321     QList<QGraphicsItem*> l = graphicsView->scene()->selectedItems();
2322     if (l.size() >= 1) {
2323         qreal currentZ = l[0]->zValue();
2324         qreal min = zIndexBounds(false, true);
2325         if (currentZ >= min) {
2326             l[0]->setZValue(currentZ - 1);
2327             updateDimension(l[0]);
2328         }
2329     }
2330 }
2331
2332 void TitleWidget::slotZIndexBottom()
2333 {
2334     QList<QGraphicsItem*> l = graphicsView->scene()->selectedItems();
2335     qreal min = zIndexBounds(false, false);
2336     for (int i = 0; i < l.size(); i++) {
2337         qreal currentZ = l[i]->zValue();
2338         if (currentZ >= min) {
2339             l[i]->setZValue(min - 1);
2340         }
2341     }
2342     // Update the z index value in the GUI
2343     if (l.size() > 0) {
2344         updateDimension(l[0]);
2345     }
2346 }
2347
2348 void TitleWidget::slotSelectAll()
2349 {
2350     QList<QGraphicsItem*> l = graphicsView->scene()->items();
2351     for (int i = 0; i < l.size(); i++) {
2352         l.at(i)->setSelected(true);
2353     }
2354 }
2355
2356 void TitleWidget::selectItems(int itemType)
2357 {
2358     QList<QGraphicsItem*> l;
2359     if (graphicsView->scene()->selectedItems().size() > 0) {
2360         l = graphicsView->scene()->selectedItems();
2361         for (int i = 0; i < l.size(); i++) {
2362             if (l.at(i)->type() != itemType) {
2363                 l.at(i)->setSelected(false);
2364             }
2365         }
2366     } else {
2367         l = graphicsView->scene()->items();
2368         for (int i = 0; i < l.size(); i++) {
2369             if (l.at(i)->type() == itemType) {
2370                 l.at(i)->setSelected(true);
2371             }
2372         }
2373     }
2374 }
2375
2376 void TitleWidget::slotSelectText()
2377 {
2378     selectItems(TEXTITEM);
2379 }
2380
2381 void TitleWidget::slotSelectRects()
2382 {
2383     selectItems(RECTITEM);
2384 }
2385
2386 void TitleWidget::slotSelectImages()
2387 {
2388     selectItems(IMAGEITEM);
2389 }
2390
2391 void TitleWidget::slotSelectNone()
2392 {
2393     graphicsView->blockSignals(true);
2394     QList<QGraphicsItem*> l = graphicsView->scene()->items();
2395     for (int i = 0; i < l.size(); i++) {
2396         l.at(i)->setSelected(false);
2397     }
2398     graphicsView->blockSignals(false);
2399     selectionChanged();
2400 }
2401
2402 QString TitleWidget::getTooltipWithShortcut(const QString &text, QAction *button)
2403 {
2404     return text + "  <b>" + button->shortcut().toString() + "</b>";
2405 }
2406
2407 void TitleWidget::prepareTools(QGraphicsItem *referenceItem)
2408 {
2409     // Let some GUI elements block signals. We may want to change their values without any sideeffects.
2410     // Additionally, store the previous blocking state to avoid side effects when this function is called from within another one.
2411     // Note: Disabling an element also blocks signals. So disabled elements don't need to be set to blocking too.
2412     bool blockOX = origin_x_left->signalsBlocked();
2413     bool blockOY = origin_y_top->signalsBlocked();
2414     bool blockEff = effect_list->signalsBlocked();
2415     bool blockRX = itemrotatex->signalsBlocked();
2416     bool blockRY = itemrotatey->signalsBlocked();
2417     bool blockRZ = itemrotatez->signalsBlocked();
2418     bool blockZoom = itemzoom->signalsBlocked();
2419     bool blockX = value_x->signalsBlocked();
2420     bool blockY = value_y->signalsBlocked();
2421     bool blockW = value_w->signalsBlocked();
2422     bool blockH = value_h->signalsBlocked();
2423     origin_x_left->blockSignals(true);
2424     origin_y_top->blockSignals(true);
2425     effect_list->blockSignals(true);
2426     itemrotatex->blockSignals(true);
2427     itemrotatey->blockSignals(true);
2428     itemrotatez->blockSignals(true);
2429     itemzoom->blockSignals(true);
2430     value_x->blockSignals(true);
2431     value_y->blockSignals(true);
2432     value_w->blockSignals(true);
2433     value_h->blockSignals(true);
2434
2435     if (referenceItem == NULL) {
2436         std::cout << "NULL item.\n";
2437         effect_stack->setHidden(true);
2438         effect_frame->setEnabled(false);
2439         effect_list->setCurrentIndex(0);
2440         origin_x_left->setChecked(false);
2441         origin_y_top->setChecked(false);
2442         updateTextOriginX();
2443         updateTextOriginY();
2444         enableToolbars(TITLE_SELECT);
2445         showToolbars(TITLE_SELECT);
2446
2447         itemzoom->setEnabled(false);
2448         itemrotatex->setEnabled(false);
2449         itemrotatey->setEnabled(false);
2450         itemrotatez->setEnabled(false);
2451         frame_properties->setEnabled(false);
2452     } else {
2453         effect_frame->setEnabled(true);
2454         frame_properties->setEnabled(true);
2455         if (referenceItem != m_startViewport && referenceItem != m_endViewport) {
2456             itemzoom->setEnabled(true);
2457             itemrotatex->setEnabled(true);
2458             itemrotatey->setEnabled(true);
2459             itemrotatez->setEnabled(true);
2460         } else {
2461             itemzoom->setEnabled(false);
2462             itemrotatex->setEnabled(false);
2463             itemrotatey->setEnabled(false);
2464             itemrotatez->setEnabled(false);
2465             updateInfoText();
2466         }
2467         if (referenceItem->type() == TEXTITEM) {
2468             showToolbars(TITLE_TEXT);
2469             QGraphicsTextItem* i = static_cast <QGraphicsTextItem *>(referenceItem);
2470             if (!i->data(100).isNull()) {
2471                 // Item has an effect
2472                 QStringList effdata = i->data(100).toStringList();
2473                 QString effectName = effdata.takeFirst();
2474                 if (effectName == "typewriter") {
2475                     QStringList params = effdata.at(0).split(';');
2476                     typewriter_delay->setValue(params.at(0).toInt());
2477                     typewriter_start->setValue(params.at(1).toInt());
2478                     effect_list->setCurrentIndex(effect_list->findData((int) TYPEWRITEREFFECT));
2479                     effect_stack->setHidden(false);
2480                 }
2481             } else {
2482 #if QT_VERSION >= 0x040600
2483                 if (i->graphicsEffect()) {
2484                     QGraphicsBlurEffect *blur = static_cast <QGraphicsBlurEffect *>(i->graphicsEffect());
2485                     if (blur) {
2486                         effect_list->setCurrentIndex(effect_list->findData((int) BLUREFFECT));
2487                         int rad = (int) blur->blurRadius();
2488                         blur_radius->setValue(rad);
2489                         effect_stack->setHidden(false);
2490                     } else {
2491                         QGraphicsDropShadowEffect *shad = static_cast <QGraphicsDropShadowEffect *>(i->graphicsEffect());
2492                         if (shad) {
2493                             effect_list->setCurrentIndex(effect_list->findData((int) SHADOWEFFECT));
2494                             shadow_radius->setValue(shad->blurRadius());
2495                             shadow_x->setValue(shad->xOffset());
2496                             shadow_y->setValue(shad->yOffset());
2497                             effect_stack->setHidden(false);
2498                         }
2499                     }
2500                 } else {
2501                     effect_list->setCurrentIndex(effect_list->findData((int) NOEFFECT));
2502                     effect_stack->setHidden(true);
2503                 }
2504 #else
2505                 effect_list->setCurrentIndex(effect_list->findData((int) NOEFFECT));
2506                 effect_stack->setHidden(true);
2507 #endif
2508             }
2509             font_size->blockSignals(true);
2510             font_family->blockSignals(true);
2511             font_weight_box->blockSignals(true);
2512             buttonItalic->blockSignals(true);
2513             buttonUnder->blockSignals(true);
2514             fontColorButton->blockSignals(true);
2515             textAlpha->blockSignals(true);
2516             buttonAlignLeft->blockSignals(true);
2517             buttonAlignRight->blockSignals(true);
2518             buttonAlignNone->blockSignals(true);
2519             buttonAlignCenter->blockSignals(true);
2520
2521             QFont font = i->font();
2522             font_family->setCurrentFont(font);
2523             font_size->setValue(font.pixelSize());
2524             m_scene->slotUpdateFontSize(font.pixelSize());
2525             buttonItalic->setChecked(font.italic());
2526             buttonUnder->setChecked(font.underline());
2527             setFontBoxWeight(font.weight());
2528
2529             QTextCursor cursor(i->document());
2530             cursor.select(QTextCursor::Document);
2531             QColor color = cursor.charFormat().foreground().color();
2532             textAlpha->setValue(color.alpha());
2533             color.setAlpha(255);
2534             fontColorButton->setColor(color);
2535
2536             if (!i->data(101).isNull()) {
2537                 textOutline->blockSignals(true);
2538                 textOutline->setValue(i->data(101).toDouble() * 10);
2539                 textOutline->blockSignals(false);
2540             }
2541             if (!i->data(102).isNull()) {
2542                 textOutlineColor->blockSignals(true);
2543                 textOutlineAlpha->blockSignals(true);
2544                 color = QColor(i->data(102).toString());
2545                 textOutlineAlpha->setValue(color.alpha());
2546                 color.setAlpha(255);
2547                 textOutlineColor->setColor(color);
2548                 textOutlineColor->blockSignals(false);
2549                 textOutlineAlpha->blockSignals(false);
2550             }
2551             QTextCursor cur = i->textCursor();
2552             QTextBlockFormat format = cur.blockFormat();
2553             if (i->textWidth() == -1) buttonAlignNone->setChecked(true);
2554             else if (format.alignment() == Qt::AlignHCenter) buttonAlignCenter->setChecked(true);
2555             else if (format.alignment() == Qt::AlignRight) buttonAlignRight->setChecked(true);
2556             else if (format.alignment() == Qt::AlignLeft) buttonAlignLeft->setChecked(true);
2557
2558             font_size->blockSignals(false);
2559             font_family->blockSignals(false);
2560             font_weight_box->blockSignals(false);
2561             buttonItalic->blockSignals(false);
2562             buttonUnder->blockSignals(false);
2563             fontColorButton->blockSignals(false);
2564             textAlpha->blockSignals(false);
2565             buttonAlignLeft->blockSignals(false);
2566             buttonAlignRight->blockSignals(false);
2567             buttonAlignNone->blockSignals(false);
2568             buttonAlignCenter->blockSignals(false);
2569
2570             // mbt 1607: Select text if the text item is an unchanged template item.
2571             if (i->property("isTemplate").isValid()) {
2572                 cur.setPosition(0, QTextCursor::MoveAnchor);
2573                 cur.select(QTextCursor::Document);
2574                 i->setTextCursor(cur);
2575                 // Make text editable now.
2576                 i->grabKeyboard();
2577                 i->setTextInteractionFlags(Qt::TextEditorInteraction);
2578             }
2579
2580             updateAxisButtons(i);
2581             updateCoordinates(i);
2582             updateDimension(i);
2583             enableToolbars(TITLE_TEXT);
2584
2585         } else if ((referenceItem)->type() == RECTITEM) {
2586             showToolbars(TITLE_RECTANGLE);
2587             settingUp = true;
2588             QGraphicsRectItem *rec = static_cast <QGraphicsRectItem *>(referenceItem);
2589             if (rec == m_startViewport || rec == m_endViewport) {
2590                 /*toolBox->setCurrentIndex(3);
2591                 toolBox->widget(0)->setEnabled(false);
2592                 toolBox->widget(1)->setEnabled(false);*/
2593                 enableToolbars(TITLE_SELECT);
2594             } else {
2595                 /*toolBox->widget(0)->setEnabled(true);
2596                 toolBox->widget(1)->setEnabled(true);
2597                 toolBox->setCurrentIndex(0);*/
2598                 //toolBox->setItemEnabled(3, true);
2599                 rectFAlpha->setValue(rec->pen().color().alpha());
2600                 rectBAlpha->setValue(rec->brush().color().alpha());
2601                 //kDebug() << rec->brush().color().alpha();
2602                 QColor fcol = rec->pen().color();
2603                 QColor bcol = rec->brush().color();
2604                 //fcol.setAlpha(255);
2605                 //bcol.setAlpha(255);
2606                 rectFColor->setColor(fcol);
2607                 rectBColor->setColor(bcol);
2608                 settingUp = false;
2609                 rectLineWidth->setValue(rec->pen().width());
2610                 enableToolbars(TITLE_RECTANGLE);
2611             }
2612
2613             updateAxisButtons(referenceItem);
2614             updateCoordinates(rec);
2615             updateDimension(rec);
2616
2617         } else if (referenceItem->type() == IMAGEITEM) {
2618             showToolbars(TITLE_IMAGE);
2619
2620             updateCoordinates(referenceItem);
2621             updateDimension(referenceItem);
2622
2623             enableToolbars(TITLE_IMAGE);
2624
2625         } else {
2626             //toolBox->setCurrentIndex(0);
2627             showToolbars(TITLE_SELECT);
2628             enableToolbars(TITLE_SELECT);
2629             frame_properties->setEnabled(false);
2630         }
2631         zValue->setValue((int)referenceItem->zValue());
2632         if (!referenceItem->data(ZOOMFACTOR).isNull()) itemzoom->setValue(referenceItem->data(ZOOMFACTOR).toInt());
2633         else itemzoom->setValue((int)(m_transformations.value(referenceItem).scalex * 100.0 + 0.5));
2634         itemrotatex->setValue((int)(m_transformations.value(referenceItem).rotatex));
2635         itemrotatey->setValue((int)(m_transformations.value(referenceItem).rotatey));
2636         itemrotatez->setValue((int)(m_transformations.value(referenceItem).rotatez));
2637     }
2638
2639
2640     effect_list->blockSignals(blockEff);
2641     itemrotatex->blockSignals(blockRX);
2642     itemrotatey->blockSignals(blockRY);
2643     itemrotatez->blockSignals(blockRZ);
2644     itemzoom->blockSignals(blockZoom);
2645     origin_x_left->blockSignals(blockOX);
2646     origin_y_top->blockSignals(blockOY);
2647     value_x->blockSignals(blockX);
2648     value_y->blockSignals(blockY);
2649     value_w->blockSignals(blockW);
2650     value_h->blockSignals(blockH);
2651 }