1 /***************************************************************************
2 * Copyright (C) 2007 by Jean-Baptiste Mardelle (jb@kdenlive.org) *
4 * This program is free software; you can redistribute it and/or modify *
5 * it under the terms of the GNU General Public License as published by *
6 * the Free Software Foundation; either version 2 of the License, or *
7 * (at your option) any later version. *
9 * This program is distributed in the hope that it will be useful, *
10 * but WITHOUT ANY WARRANTY; without even the implied warranty of *
11 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the *
12 * GNU General Public License for more details. *
14 * You should have received a copy of the GNU General Public License *
15 * along with this program; if not, write to the *
16 * Free Software Foundation, Inc., *
17 * 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA *
18 ***************************************************************************/
20 #include <QMouseEvent>
21 #include <QStylePainter>
22 #include <QGraphicsItem>
23 #include <QDomDocument>
30 #include "customtrackview.h"
32 #include "definitions.h"
33 #include "moveclipcommand.h"
34 #include "resizeclipcommand.h"
35 #include "addtimelineclipcommand.h"
37 CustomTrackView::CustomTrackView(KdenliveDoc *doc, QGraphicsScene * projectscene, QWidget *parent)
38 : QGraphicsView(projectscene, parent), m_tracksCount(0), m_cursorPos(0), m_dropItem(NULL), m_cursorLine(NULL), m_operationMode(NONE), m_startPos(QPointF()), m_dragItem(NULL), m_visualTip(NULL), m_moveOpMode(NONE), m_animation(NULL), m_projectDuration(0), m_scale(1.0), m_clickPoint(0), m_document(doc)
40 if (doc) m_commandStack = doc->commandStack();
41 else m_commandStack == NULL;
42 setMouseTracking(true);
44 m_animationTimer = new QTimeLine(800);
45 m_animationTimer->setFrameRange(0, 5);
46 m_animationTimer->setUpdateInterval(100);
47 m_animationTimer->setLoopCount(0);
48 m_tipColor = QColor(230, 50, 0, 150);
49 setContentsMargins(0, 0, 0, 0);
51 m_cursorLine = projectscene->addLine(0, 0, 0, 50);
52 m_cursorLine->setFlags(QGraphicsItem::ItemIsMovable | QGraphicsItem::ItemIgnoresTransformations);
53 m_cursorLine->setZValue(1000);
57 void CustomTrackView::initView()
63 void CustomTrackView::resizeEvent ( QResizeEvent * event )
65 QGraphicsView::resizeEvent(event);
69 void CustomTrackView::wheelEvent ( QWheelEvent * e )
71 if (e->modifiers() == Qt::ControlModifier) {
72 if (e->delta() > 0) emit zoomIn();
76 if (e->delta() > 0) horizontalScrollBar()->setValue (horizontalScrollBar()->value() + horizontalScrollBar()->singleStep ());
77 else horizontalScrollBar()->setValue (horizontalScrollBar()->value() - horizontalScrollBar()->singleStep ());
83 void CustomTrackView::mouseMoveEvent ( QMouseEvent * event )
86 emit mousePosition(mapToScene(event->pos()).x() / m_scale);
87 /*if (event->modifiers() == Qt::ControlModifier)
88 setDragMode(QGraphicsView::ScrollHandDrag);
89 else if (event->modifiers() == Qt::ShiftModifier)
90 setDragMode(QGraphicsView::RubberBandDrag);
93 if (m_dragItem) { //event->button() == Qt::LeftButton) {
94 // a button was pressed, delete visual tips
95 if (m_operationMode == MOVE) {
96 double snappedPos = getSnapPointForPos(mapToScene(event->pos()).x() - m_clickPoint);
97 double moveX = snappedPos; //mapToScene(event->pos()).x();
98 //kDebug()<<"/////// MOVE CLIP, EVENT Y: "<<event->scenePos().y()<<", SCENE HEIGHT: "<<scene()->sceneRect().height();
99 int moveTrack = (int) mapToScene(event->pos()).y() / 50;
100 int currentTrack = m_dragItem->track();
102 if (moveTrack > m_tracksCount - 1) moveTrack = m_tracksCount - 1;
103 else if (moveTrack < 0) moveTrack = 0;
105 int offset = moveTrack - currentTrack;
106 if (offset != 0) offset = 50 * offset;
107 m_dragItem->moveTo(moveX / m_scale, m_scale, offset, moveTrack);
109 else if (m_operationMode == RESIZESTART) {
110 int pos = mapToScene(event->pos()).x();
111 m_dragItem->resizeStart(pos / m_scale, m_scale);
113 else if (m_operationMode == RESIZEEND) {
114 int pos = mapToScene(event->pos()).x();
115 m_dragItem->resizeEnd(pos / m_scale, m_scale);
117 else if (m_operationMode == FADEIN) {
118 int pos = mapToScene(event->pos()).x() / m_scale;
119 m_dragItem->setFadeIn(pos - m_dragItem->startPos(), m_scale);
121 else if (m_operationMode == FADEOUT) {
122 int pos = mapToScene(event->pos()).x() / m_scale;
123 m_dragItem->setFadeOut(m_dragItem->endPos() - pos, m_scale);
126 if (m_animation) delete m_animation;
128 if (m_visualTip) delete m_visualTip;
130 QGraphicsView::mouseMoveEvent(event);
134 QList<QGraphicsItem *> itemList = items( event->pos());
136 QGraphicsItem *item = NULL;
137 for (int i = 0; i < itemList.count(); i++) {
138 if (itemList.at(i)->type() == 70000) {
139 item = itemList.at(i);
144 ClipItem *clip = (ClipItem*) item;
145 double size = mapToScene(QPoint(8, 0)).x();
146 OPERATIONTYPE opMode = clip->operationMode(mapToScene(event->pos()), m_scale);
147 if (opMode == m_moveOpMode) {
148 QGraphicsView::mouseMoveEvent(event);
153 if (m_animation) delete m_animation;
159 m_moveOpMode = opMode;
160 if (opMode == MOVE) {
161 setCursor(Qt::OpenHandCursor);
163 else if (opMode == RESIZESTART) {
164 kDebug()<<"******** RESIZE CLIP START; WIDTH: "<<size;
165 if (m_visualTip == NULL) {
166 m_visualTip = new QGraphicsRectItem(clip->rect().x(), clip->rect().y(), size, clip->rect().height());
167 ((QGraphicsRectItem*) m_visualTip)->setBrush(m_tipColor);
168 ((QGraphicsRectItem*) m_visualTip)->setPen(QPen(Qt::transparent));
169 m_visualTip->setZValue (100);
170 m_animation = new QGraphicsItemAnimation;
171 m_animation->setItem(m_visualTip);
172 m_animation->setTimeLine(m_animationTimer);
173 m_visualTip->setPos(0, 0);
175 m_animation->setScaleAt(.5, scale, 1);
176 m_animation->setPosAt(.5, QPointF(clip->rect().x() - clip->rect().x() * scale, 0));
178 m_animation->setScaleAt(1, scale, 1);
179 m_animation->setPosAt(1, QPointF(clip->rect().x() - clip->rect().x() * scale, 0));
180 scene()->addItem(m_visualTip);
181 m_animationTimer->start();
183 setCursor(Qt::SizeHorCursor);
185 else if (opMode == RESIZEEND) {
186 if (m_visualTip == NULL) {
187 m_visualTip = new QGraphicsRectItem(clip->rect().x() + clip->rect().width() - size, clip->rect().y(), size, clip->rect().height());
188 ((QGraphicsRectItem*) m_visualTip)->setBrush(m_tipColor);
189 ((QGraphicsRectItem*) m_visualTip)->setPen(QPen(Qt::transparent));
190 m_visualTip->setZValue (100);
191 m_animation = new QGraphicsItemAnimation;
192 m_animation->setItem(m_visualTip);
193 m_animation->setTimeLine(m_animationTimer);
194 m_visualTip->setPos(0, 0);
196 m_animation->setScaleAt(.5, scale, 1);
197 m_animation->setPosAt(.5, QPointF(clip->rect().x() - clip->rect().x() * scale - clip->rect().width(), 0));
199 m_animation->setScaleAt(1, scale, 1);
200 m_animation->setPosAt(1, QPointF(clip->rect().x() - clip->rect().x() * scale, 0));
201 scene()->addItem(m_visualTip);
202 m_animationTimer->start();
204 setCursor(Qt::SizeHorCursor);
206 else if (opMode == FADEIN) {
207 if (m_visualTip == NULL) {
208 m_visualTip = new QGraphicsEllipseItem(clip->rect().x() + clip->fadeIn() * m_scale - size, clip->rect().y() - 8, size * 2, 16);
209 ((QGraphicsEllipseItem*) m_visualTip)->setBrush(m_tipColor);
210 ((QGraphicsEllipseItem*) m_visualTip)->setPen(QPen(Qt::transparent));
211 m_visualTip->setZValue (100);
212 m_animation = new QGraphicsItemAnimation;
213 m_animation->setItem(m_visualTip);
214 m_animation->setTimeLine(m_animationTimer);
215 m_visualTip->setPos(0, 0);
217 m_animation->setScaleAt(.5, scale, scale);
218 m_animation->setPosAt(.5, QPointF(clip->rect().x() - clip->rect().x() * scale - clip->fadeIn() * m_scale, clip->rect().y() - clip->rect().y() * scale));
220 m_animation->setScaleAt(1, scale, scale);
221 m_animation->setPosAt(1, QPointF(clip->rect().x() - clip->rect().x() * scale, clip->rect().y() - clip->rect().y() * scale));
222 scene()->addItem(m_visualTip);
223 m_animationTimer->start();
225 setCursor(Qt::PointingHandCursor);
227 else if (opMode == FADEOUT) {
228 if (m_visualTip == NULL) {
229 m_visualTip = new QGraphicsEllipseItem(clip->rect().x() + clip->rect().width() - clip->fadeOut() * m_scale - size, clip->rect().y() - 8, size*2, 16);
230 ((QGraphicsEllipseItem*) m_visualTip)->setBrush(m_tipColor);
231 ((QGraphicsEllipseItem*) m_visualTip)->setPen(QPen(Qt::transparent));
232 m_visualTip->setZValue (100);
233 m_animation = new QGraphicsItemAnimation;
234 m_animation->setItem(m_visualTip);
235 m_animation->setTimeLine(m_animationTimer);
236 m_visualTip->setPos(0, 0);
238 m_animation->setScaleAt(.5, scale, scale);
239 m_animation->setPosAt(.5, QPointF(clip->rect().x() - clip->rect().x() * scale - clip->rect().width() + clip->fadeOut() * m_scale, clip->rect().y() - clip->rect().y() * scale));
241 m_animation->setScaleAt(1, scale, scale);
242 m_animation->setPosAt(1, QPointF(clip->rect().x() - clip->rect().x() * scale, clip->rect().y() - clip->rect().y() * scale));
243 scene()->addItem(m_visualTip);
244 m_animationTimer->start();
246 setCursor(Qt::PointingHandCursor);
252 if (m_animation) delete m_animation;
257 setCursor(Qt::ArrowCursor);
260 QGraphicsView::mouseMoveEvent(event);
264 void CustomTrackView::mousePressEvent ( QMouseEvent * event )
266 kDebug()<<"-- TIMELINE MSE PRESSED";
267 int pos = event->x();
268 if (event->modifiers() == Qt::ControlModifier)
269 setDragMode(QGraphicsView::ScrollHandDrag);
270 else if (event->modifiers() == Qt::ShiftModifier)
271 setDragMode(QGraphicsView::RubberBandDrag);
273 bool collision = false;
274 QList<QGraphicsItem *> collisionList = items(event->pos());
275 for (int i = 0; i < collisionList.size(); ++i) {
276 QGraphicsItem *item = collisionList.at(i);
277 if (item->type() == 70000) {
278 m_dragItem = (ClipItem *) item;
279 m_clickPoint = mapToScene(event->pos()).x() - m_dragItem->startPos() * m_scale;
280 m_operationMode = m_dragItem->operationMode(item->mapFromScene(mapToScene(event->pos())), m_scale);
281 if (m_operationMode == MOVE || m_operationMode == RESIZESTART)
282 m_startPos = QPointF(m_dragItem->startPos(), m_dragItem->track());
283 else if (m_operationMode == RESIZEEND)
284 m_startPos = QPointF(m_dragItem->endPos(), m_dragItem->track());
285 kDebug()<<"//////// ITEM CLICKED: "<<m_startPos;
291 kDebug()<<"//////// NO ITEM FOUND ON CLICK";
293 setCursor(Qt::ArrowCursor);
294 setCursorPos((int) mapToScene(event->x(), 0).x());
295 emit cursorMoved(cursorPos());
298 updateSnapPoints(m_dragItem);
300 QGraphicsView::mousePressEvent(event);
303 void CustomTrackView::dragEnterEvent ( QDragEnterEvent * event )
305 if (event->mimeData()->hasText()) {
306 QString clip = event->mimeData()->text();
307 addItem(clip, event->pos());
308 event->acceptProposedAction();
313 void CustomTrackView::addItem(QString producer, QPoint pos)
316 doc.setContent(producer);
317 QDomElement elem = doc.documentElement();
318 int in = elem.attribute("in", 0).toInt();
319 int out = elem.attribute("out", 0).toInt() - in;
320 if (out == 0) out = elem.attribute("duration", 0).toInt();
321 kDebug()<<"ADDING CLIP: "<<producer<<", OUT: "<<out<<", POS: "<<mapToScene(pos);
322 int trackTop = ((int) mapToScene(pos).y()/50) * 50 + 1;
323 m_dropItem = new ClipItem(elem, ((int) mapToScene(pos).y()/50), in, QRectF(mapToScene(pos).x() * m_scale, trackTop, out * m_scale, 49), out);
324 scene()->addItem(m_dropItem);
328 void CustomTrackView::dragMoveEvent(QDragMoveEvent * event) {
329 event->setDropAction(Qt::IgnoreAction);
331 int track = (int) mapToScene(event->pos()).y()/50; //) * (m_scale * 50) + m_scale;
332 kDebug()<<"+++++++++++++ DRAG MOVE, : "<<mapToScene(event->pos()).x()<<", SCAL: "<<m_scale;
333 m_dropItem->moveTo(mapToScene(event->pos()).x() / m_scale, m_scale, (track - m_dropItem->track()) * 50, track);
336 event->setDropAction(Qt::MoveAction);
337 if (event->mimeData()->hasText()) {
338 event->acceptProposedAction();
343 void CustomTrackView::dragLeaveEvent ( QDragLeaveEvent * event ) {
350 void CustomTrackView::dropEvent ( QDropEvent * event ) {
352 AddTimelineClipCommand *command = new AddTimelineClipCommand(this, m_dropItem->xml(), m_dropItem->track(), m_dropItem->startPos(), m_dropItem->rect(), m_dropItem->duration(), false);
353 m_commandStack->push(command);
354 kDebug()<<"IIIIIIIIIIIIIIIIIIIIIIII TRAX CNT: "<<m_tracksCount<<", DROP: "<<m_dropItem->track();
355 m_document->renderer()->mltInsertClip(m_tracksCount - m_dropItem->track(), GenTime(m_dropItem->startPos(), 25), m_dropItem->xml());
361 QStringList CustomTrackView::mimeTypes () const
363 QStringList qstrList;
364 // list of accepted mime types for drop
365 qstrList.append("text/plain");
369 Qt::DropActions CustomTrackView::supportedDropActions () const
371 // returns what actions are supported when dropping
372 return Qt::MoveAction;
375 void CustomTrackView::setDuration(int duration)
377 kDebug()<<"///////////// PRO DUR: "<<duration<<", height: "<<50 * m_tracksCount;
378 m_projectDuration = duration;
379 scene()->setSceneRect(0, 0, m_projectDuration + 500, scene()->sceneRect().height()); //50 * m_tracksCount);
383 void CustomTrackView::addTrack ()
386 m_cursorLine->setLine(m_cursorLine->line().x1(), 0, m_cursorLine->line().x1(), 50 * m_tracksCount);
387 //setSceneRect(0, 0, sceneRect().width(), 50 * m_tracksCount);
388 //verticalScrollBar()->setMaximum(50 * m_tracksCount);
389 //setFixedHeight(50 * m_tracksCount);
392 void CustomTrackView::removeTrack ()
395 m_cursorLine->setLine(m_cursorLine->line().x1(), 0, m_cursorLine->line().x1(), 50 * m_tracksCount);
398 void CustomTrackView::setCursorPos(int pos, bool seek)
401 m_cursorLine->setPos(pos, 0);
402 int frame = mapToScene(QPoint(pos, 0)).x() / m_scale;
403 if (seek) m_document->renderer()->seek(GenTime(frame, 25));
406 int CustomTrackView::cursorPos()
411 void CustomTrackView::mouseReleaseEvent ( QMouseEvent * event )
413 QGraphicsView::mouseReleaseEvent(event);
414 setDragMode(QGraphicsView::NoDrag);
415 if (m_dragItem == NULL) return;
416 //kDebug()<<"/// MOVING CLIP: "<<m_startPos<<", END: "<<QPoint(m_dragItem->rect().x(),m_dragItem->rect().y());
417 if (m_operationMode == MOVE) {
419 MoveClipCommand *command = new MoveClipCommand(this, m_startPos, QPointF(m_dragItem->startPos(), m_dragItem->track()), false);
420 m_commandStack->push(command);
421 m_document->renderer()->mltMoveClip(m_tracksCount - m_startPos.y(), m_tracksCount - m_dragItem->track(), m_startPos.x(), m_dragItem->startPos());
423 else if (m_operationMode == RESIZESTART) {
425 ResizeClipCommand *command = new ResizeClipCommand(this, m_startPos, QPointF(m_dragItem->startPos(), m_dragItem->track()), true, false);
426 m_commandStack->push(command);
428 else if (m_operationMode == RESIZEEND) {
430 ResizeClipCommand *command = new ResizeClipCommand(this, m_startPos, QPointF(m_dragItem->endPos(), m_dragItem->track()), false, false);
431 m_commandStack->push(command);
433 m_operationMode = NONE;
437 void CustomTrackView::deleteClip (int track, int startpos, const QRectF &rect )
439 ClipItem *item = (ClipItem *) scene()->itemAt(rect.x() + 1, rect.y() + 1);
441 kDebug()<<"---------------- ERROR, CANNOT find clip to move at: "<<rect.x();
445 m_document->renderer()->mltRemoveClip(m_tracksCount - track, GenTime(startpos, 25));
448 void CustomTrackView::addClip ( QDomElement xml, int track, int startpos, const QRectF &rect, int duration )
450 QRect r(startpos * m_scale, 50 * track, duration * m_scale, 49);
451 ClipItem *item = new ClipItem(xml, track, startpos, r, duration);
452 scene()->addItem(item);
453 m_document->renderer()->mltInsertClip(m_tracksCount - track, GenTime(startpos, 25), xml);
456 void CustomTrackView::moveClip ( const QPointF &startPos, const QPointF &endPos )
458 ClipItem *item = (ClipItem *) scene()->itemAt((startPos.x() + 1) * m_scale, startPos.y() * 50 + 25);
460 kDebug()<<"---------------- ERROR, CANNOT find clip to move at: "<<startPos.x() * m_scale * FRAME_SIZE + 1<<", "<<startPos.y() * 50 + 25;
463 kDebug()<<"---------------- Move CLIP FROM: "<<startPos.x()<<", END:"<<endPos.x();
464 item->moveTo(endPos.x(), m_scale, (endPos.y() - startPos.y()) * 50, endPos.y());
465 m_document->renderer()->mltMoveClip(m_tracksCount - startPos.y(), m_tracksCount - endPos.y(), startPos.x(), endPos.x());
468 void CustomTrackView::resizeClip ( const QPointF &startPos, const QPointF &endPos, bool resizeClipStart )
471 if (resizeClipStart) offset = 1;
473 ClipItem *item = (ClipItem *) scene()->itemAt((startPos.x() + offset) * m_scale, startPos.y() * 50 + 25);
475 kDebug()<<"---------------- ERROR, CANNOT find clip to resize at: "<<startPos;
478 qreal diff = endPos.x() - startPos.x();
479 if (resizeClipStart) {
480 item->resizeStart(endPos.x(), m_scale);
483 item->resizeEnd(endPos.x(), m_scale);
487 double CustomTrackView::getSnapPointForPos(double pos)
489 for (int i = 0; i < m_snapPoints.size(); ++i) {
490 //kDebug()<<"SNAP POINT: "<<m_snapPoints.at(i);
491 if (abs(pos - m_snapPoints.at(i) * m_scale) < 6 * m_scale) {
492 //kDebug()<<" FOUND SNAP POINT AT: "<<m_snapPoints.at(i)<<", current pos: "<<pos / m_scale;
493 return m_snapPoints.at(i) * m_scale + 0.5;
495 if (m_snapPoints.at(i) > pos) break;
500 void CustomTrackView::updateSnapPoints(ClipItem *selected)
502 m_snapPoints.clear();
504 if (selected) offset = selected->duration();
505 QList<QGraphicsItem *> itemList = items();
506 for (int i = 0; i < itemList.count(); i++) {
507 if (itemList.at(i)->type() == 70000 && itemList.at(i) != selected) {
508 ClipItem *item = (ClipItem *)itemList.at(i);
509 int start = item->startPos();
510 int fadein = item->fadeIn() + start;
511 int end = item->endPos();
512 int fadeout = end - item->fadeOut();
513 m_snapPoints.append(start);
514 if (fadein != start) m_snapPoints.append(fadein);
515 m_snapPoints.append(end);
516 if (fadeout != end) m_snapPoints.append(fadeout);
518 m_snapPoints.append(start - offset);
519 if (fadein != start) m_snapPoints.append(fadein - offset);
520 m_snapPoints.append(end - offset);
521 if (fadeout != end) m_snapPoints.append(fadeout - offset);
525 kDebug()<<" GOT SNAPPOINTS TOTAL: "<<m_snapPoints.count();
527 for (int i = 0; i < m_snapPoints.size(); ++i)
528 kDebug()<<"SNAP POINT: "<<m_snapPoints.at(i);
532 void CustomTrackView::setScale(double scaleFactor)
534 //scale(scaleFactor, scaleFactor);
535 m_scale = scaleFactor;
536 kDebug()<<" HHHHHHHH SCALING: "<<m_scale;
537 QList<QGraphicsItem *> itemList = items();
539 for (int i = 0; i < itemList.count(); i++) {
540 if (itemList.at(i)->type() == 70000) {
541 ClipItem *clip = (ClipItem *)itemList.at(i);
542 clip->setRect(clip->startPos() * m_scale, clip->rect().y(), clip->duration() * m_scale, clip->rect().height());
544 /*else if (itemList.at(i)->type() == 70001) {
545 LabelItem *label = (LabelItem *)itemList.at(i);
546 QGraphicsItem *parent = label->parentItem();
547 QRectF r = label->boundingRect();
548 QRectF p = parent->boundingRect();
549 label->setPos(p.x() + p.width() / 2 - r.width() / 2, p.y() + p.height() / 2 - r.height() / 2);
550 //label->setRect(clip->startPos() * m_scale, clip->rect().y(), clip->duration() * m_scale, clip->rect().height());
555 void CustomTrackView::drawBackground ( QPainter * painter, const QRectF & rect )
557 QColor base = palette().button().color();
558 //painter->setPen(base);
559 painter->setClipRect(rect);
560 painter->drawLine(0, 0, rect.width(), 0);
561 for (uint i = 0; i < m_tracksCount;i++)
563 painter->drawLine(0, 50 * (i+1), width(), 50 * (i+1));
564 painter->drawText(QRectF(10, 50 * i, 100, 50 * i + 49), Qt::AlignLeft, i18n(" Track ") + QString::number(i + 1));
566 int lowerLimit = 50 * m_tracksCount;
567 if (height() > lowerLimit)
568 painter->fillRect(QRectF(0, lowerLimit, rect.width(), height() - lowerLimit), QBrush(base));
571 void CustomTrackView::drawForeground ( QPainter * painter, const QRectF & rect )
573 //kDebug()<<"///// DRAWING FB: "<<rect.x()<<", width: "<<rect.width();
574 painter->fillRect(rect, QColor(50, rand() % 250,50,100));
575 painter->drawLine(m_cursorPos, rect.y(), m_cursorPos, rect.y() + rect.height());
578 #include "customtrackview.moc"