1 /***************************************************************************
2 * Copyright (C) 2008 by Marco Gittler (g.marco@freenet.de) *
3 * Copyright (C) 2008 by Jean-Baptiste Mardelle (jb@kdenlive.org) *
5 * This program is free software; you can redistribute it and/or modify *
6 * it under the terms of the GNU General Public License as published by *
7 * the Free Software Foundation; either version 2 of the License, or *
8 * (at your option) any later version. *
10 * This program is distributed in the hope that it will be useful, *
11 * but WITHOUT ANY WARRANTY; without even the implied warranty of *
12 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the *
13 * GNU General Public License for more details. *
15 * You should have received a copy of the GNU General Public License *
16 * along with this program; if not, write to the *
17 * Free Software Foundation, Inc., *
18 * 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA *
19 ***************************************************************************/
21 #include <QGraphicsScene>
22 #include <QGraphicsView>
29 #include "abstractgroupitem.h"
30 #include "abstractclipitem.h"
31 #include "kdenlivesettings.h"
32 #include "customtrackscene.h"
34 AbstractGroupItem::AbstractGroupItem(double fps): QGraphicsItemGroup(), m_fps(fps) {
35 setFlags(QGraphicsItem::ItemClipsToShape | QGraphicsItem::ItemIsMovable | QGraphicsItem::ItemIsSelectable);
38 int AbstractGroupItem::type() const {
42 const int AbstractGroupItem::track() const {
43 return (int)(scenePos().y() / KdenliveSettings::trackheight());
46 CustomTrackScene* AbstractGroupItem::projectScene() {
47 if (scene()) return static_cast <CustomTrackScene*>(scene());
52 QPolygonF AbstractGroupItem::groupShape(QPointF offset) {
53 QList<QGraphicsItem *> children = childItems();
55 for (int i = 0; i < children.count(); i++) {
56 if (children.at(i)->type() == AVWIDGET) {
57 QPolygonF r = QPolygonF(children.at(i)->sceneBoundingRect());
58 path = path.united(r);
61 path.translate(offset);
65 void AbstractGroupItem::addItem(QGraphicsItem * item) {
70 void AbstractGroupItem::fixItemRect() {
71 QPointF start = boundingRect().topLeft();
72 if (start != QPointF(0, 0)) {
73 translate(0 - start.x(), 0 - start.y());
79 void AbstractGroupItem::paint(QPainter *p, const QStyleOptionGraphicsItem *, QWidget *) {
80 p->fillRect(boundingRect(), QColor(200, 100, 100, 100));
84 QVariant AbstractGroupItem::itemChange(GraphicsItemChange change, const QVariant &value) {
85 if (change == ItemPositionChange && scene()) {
86 // calculate new position.
87 QPointF newPos = value.toPointF();
88 int xpos = projectScene()->getSnapPointForPos((int) newPos.x(), KdenliveSettings::snaptopoints());
92 //kDebug()<<"// GRP MOVE: "<<pos().y()<<"->"<<newPos.y();
93 QPointF start = pos();//sceneBoundingRect().topLeft();
94 int posx = start.x() + newPos.x(); //projectScene()->getSnapPointForPos(start.x() + sc.x(), KdenliveSettings::snaptopoints());
96 int startTrack = (start.y() + KdenliveSettings::trackheight() / 2) / KdenliveSettings::trackheight();
97 int newTrack = (newPos.y()) / KdenliveSettings::trackheight();
98 //kDebug()<<"// GROUP NEW T:"<<newTrack<<",START T:"<<startTrack<<",MAX:"<<projectScene()->tracksCount() - 1;
99 newTrack = qMin(newTrack, projectScene()->tracksCount() - (int)(boundingRect().height() + 5) / KdenliveSettings::trackheight());
100 newTrack = qMax(newTrack, 0);
101 newPos.setY((int)((newTrack) * KdenliveSettings::trackheight()));
103 //kDebug() << "------------------------------------GRUOP MOVE";
105 if (start.x() + newPos.x() - pos().x() < 0) {
106 // If group goes below 0, adjust position to 0
107 return QPointF(pos().x() - start.x(), pos().y());
110 QPolygonF sceneShape = groupShape(newPos - pos());
111 QList<QGraphicsItem*> collindingItems = scene()->items(sceneShape, Qt::IntersectsItemShape);
112 QList<QGraphicsItem *> children = childItems();
113 for (int i = 0; i < children.count(); i++) {
114 collindingItems.removeAll(children.at(i));
116 if (collindingItems.isEmpty()) return newPos;
118 for (int i = 0; i < collindingItems.count(); i++) {
119 QGraphicsItem *collision = collindingItems.at(i);
120 if (collision->type() == AVWIDGET) {
123 //TODO: improve movement when collision happens
124 /*if (startTrack != newTrack) return pos();
125 if (collision->pos().x() > pos().x()) {
126 return QPointF(collision->sceneBoundingRect().x() - sceneBoundingRect().width() + pos().x() - start.x() - 1, newPos.y());
133 //else posx -= startx;
134 //posx = qMax(posx, 0);
136 //kDebug()<<"Y POS: "<<start.y() + newPos.y()<<"SCN MP: "<<sc;
137 /*int newTrack = (start.y() + newPos.y()) / KdenliveSettings::trackheight();
138 int oldTrack = (start.y() + pos().y()) / KdenliveSettings::trackheight();
139 newPos.setY((newTrack) * KdenliveSettings::trackheight() - start.y() + 1);*/
142 //if (start.y() + newPos.y() < 1) newTrack = oldTrack;
146 // Only one clip is moving
148 QList<QGraphicsItem*> items = scene()->items(sceneShape, Qt::IntersectsItemShape);
151 if (!items.isEmpty()) {
152 for (int i = 0; i < items.count(); i++) {
153 if (items.at(i)->type() == AVWIDGET) {
155 //kDebug()<<"/// COLLISION WITH ITEM: "<<items.at(i)->sceneBoundingRect();
157 QPointF otherPos = items.at(i)->pos();
158 if ((int) otherPos.y() != (int) pos().y()) return pos();
159 if (pos().x() < otherPos.x()) {
160 // move clip just before colliding clip
161 int npos = (static_cast < AbstractClipItem* >(items.at(i))->startPos()).frames(m_fps) - sceneBoundingRect().width();
164 // get pos just after colliding clip
165 int npos = static_cast < AbstractClipItem* >(items.at(i))->endPos().frames(m_fps);
174 return QGraphicsItem::itemChange(change, value);