summaryrefslogtreecommitdiffstats
path: root/profile-widget/profilewidget2.cpp
diff options
context:
space:
mode:
authorGravatar Berthold Stoeger <bstoeger@mail.tuwien.ac.at>2018-07-15 17:56:18 +0200
committerGravatar Dirk Hohndel <dirk@hohndel.org>2018-07-23 15:58:55 -0700
commitb3feaa80e26510928c73e92049ec346841f6093a (patch)
tree1a1f002959ea09aa5a16a5cc04ace06326941377 /profile-widget/profilewidget2.cpp
parentc7428859840bfcf973a14f27c6376e39e915cb29 (diff)
downloadsubsurface-b3feaa80e26510928c73e92049ec346841f6093a.tar.gz
Dive video: paint duration-bar above thumbnail in profile plot
Paint a rectangle on top of thumbnails indicating the run-time of the video. Use the z=100.0-101.0 range for painting the thumbnails, whereby the z-value increases uniformly from first to last thumbnail (sorted by timestamp). The duration-bars are placed at z-values midway between those of the thumbnails. Signed-off-by: Berthold Stoeger <bstoeger@mail.tuwien.ac.at>
Diffstat (limited to 'profile-widget/profilewidget2.cpp')
-rw-r--r--profile-widget/profilewidget2.cpp102
1 files changed, 82 insertions, 20 deletions
diff --git a/profile-widget/profilewidget2.cpp b/profile-widget/profilewidget2.cpp
index 11dddaa41..79aa2e876 100644
--- a/profile-widget/profilewidget2.cpp
+++ b/profile-widget/profilewidget2.cpp
@@ -87,6 +87,10 @@ static struct _ItemPos {
_Axis heartBeatWithTankBar;
} itemPos;
+// Constant describing at which z-level the thumbnails are located.
+// We might add more constants here for easier customability.
+static const double thumbnailBaseZValue = 100.0;
+
ProfileWidget2::ProfileWidget2(QWidget *parent) : QGraphicsView(parent),
currentState(INVALID),
dataModel(new DivePlotDataModel(this)),
@@ -975,6 +979,21 @@ void ProfileWidget2::fixBackgroundPos()
background->setY(mapToScene(y, 20).y());
}
+void ProfileWidget2::scale(qreal sx, qreal sy)
+{
+ QGraphicsView::scale(sx, sy);
+
+#ifndef SUBSURFACE_MOBILE
+ // Since the zoom level changed, adjust the duration bars accordingly.
+ // We want to grow/shrink the length, but not the height and pen.
+ for (PictureEntry &p: pictures)
+ updateDurationLine(p);
+
+ // Since we created new duration lines, we have to update the order in which the thumbnails is painted.
+ updateThumbnailPaintOrder();
+#endif
+}
+
#ifndef SUBSURFACE_MOBILE
void ProfileWidget2::wheelEvent(QWheelEvent *event)
{
@@ -2064,9 +2083,37 @@ void ProfileWidget2::clearPictures()
pictures.clear();
}
+static const double unscaledDurationLineWidth = 2.5;
+static const double unscaledDurationLinePenWidth = 0.5;
+
+// Reset the duration line after an image was moved or we found a new duration
+void ProfileWidget2::updateDurationLine(PictureEntry &e)
+{
+ if (e.duration.seconds > 0) {
+ // We know the duration of this video, reset the line symbolizing its extent accordingly
+ double begin = timeAxis->posAtValue(e.offset.seconds);
+ double end = timeAxis->posAtValue(e.offset.seconds + e.duration.seconds);
+ double y = e.thumbnail->y();
+
+ // Undo scaling for pen-width and line-width. For this purpose, we use the scaling of the y-axis.
+ double scale = transform().m22();
+ double durationLineWidth = unscaledDurationLineWidth / scale;
+ double durationLinePenWidth = unscaledDurationLinePenWidth / scale;
+ e.durationLine.reset(new QGraphicsRectItem(begin, y - durationLineWidth - durationLinePenWidth, end - begin, durationLineWidth));
+ e.durationLine->setPen(QPen(getColor(GF_LINE, isGrayscale), durationLinePenWidth));
+ e.durationLine->setBrush(getColor(::BACKGROUND, isGrayscale));
+ e.durationLine->setVisible(prefs.show_pictures_in_profile);
+ scene()->addItem(e.durationLine.get());
+ } else {
+ // This is either a picture or a video with unknown duration.
+ // In case there was a line (how could that be?) remove it.
+ e.durationLine.reset();
+ }
+}
+
// This function is called asynchronously by the thumbnailer if a thumbnail
// was fetched from disk or freshly calculated.
-void ProfileWidget2::updateThumbnail(QString filename, QImage thumbnail)
+void ProfileWidget2::updateThumbnail(QString filename, QImage thumbnail, duration_t duration)
{
// Find the picture with the given filename
auto it = std::find_if(pictures.begin(), pictures.end(), [&filename](const PictureEntry &e)
@@ -2078,11 +2125,20 @@ void ProfileWidget2::updateThumbnail(QString filename, QImage thumbnail)
// Replace the pixmap of the thumbnail with the newly calculated one.
int size = Thumbnailer::defaultThumbnailSize();
it->thumbnail->setPixmap(QPixmap::fromImage(thumbnail.scaled(size, size, Qt::KeepAspectRatio)));
+
+ // If the duration changed, update the line
+ if (duration.seconds != it->duration.seconds) {
+ it->duration = duration;
+ updateDurationLine(*it);
+ // If we created / removed a duration line, we have to update the thumbnail paint order.
+ updateThumbnailPaintOrder();
+ }
}
}
// Create a PictureEntry object and add its thumbnail to the scene if profile pictures are shown.
ProfileWidget2::PictureEntry::PictureEntry(offset_t offsetIn, const QString &filenameIn, QGraphicsScene *scene) : offset(offsetIn),
+ duration(duration_t {0}),
filename(filenameIn),
thumbnail(new DivePictureItem)
{
@@ -2101,23 +2157,37 @@ bool ProfileWidget2::PictureEntry::operator< (const PictureEntry &e) const
return std::tie(offset.seconds, filename) < std::tie(e.offset.seconds, e.filename);
}
+// This function updates the paint order of the thumbnails and duration-lines, such that later
+// thumbnails are painted on top of previous thumbnails and duration-lines on top of the thumbnail
+// they belong to.
+void ProfileWidget2::updateThumbnailPaintOrder()
+{
+ if (!pictures.size())
+ return;
+ // To get the correct sort order, we place in thumbnails at equal z-distances
+ // between thumbnailBaseZValue and (thumbnailBaseZValue + 1.0).
+ // Duration-lines are placed between the thumbnails.
+ double z = thumbnailBaseZValue;
+ double step = 1.0 / (double)pictures.size();
+ for (PictureEntry &e: pictures) {
+ e.thumbnail->setBaseZValue(z);
+ if (e.durationLine)
+ e.durationLine->setZValue(z + step / 2.0);
+ z += step;
+ }
+}
+
// Calculate the y-coordinates of the thumbnails, which are supposed to be sorted by x-coordinate.
// This will also change the order in which the thumbnails are painted, to avoid weird effects,
// when items are added later to the scene. This is done using the QGraphicsItem::packBefore() function.
// We can't use the z-value, because that will be modified on hoverEnter and hoverExit events.
void ProfileWidget2::calculatePictureYPositions()
{
- // Quit early if there are no items. The last loop in this function assumes that the vector is not empty.
- if (pictures.empty())
- return;
-
double lastX = -1.0, lastY = 0.0;
- for (auto it = pictures.begin(); it != pictures.end(); ++it) {
- if (!it->thumbnail)
- continue;
+ for (PictureEntry &e: pictures) {
// let's put the picture at the correct time, but at a fixed "depth" on the profile
// not sure this is ideal, but it seems to look right.
- double x = it->thumbnail->x();
+ double x = e.thumbnail->x();
double y;
if (lastX >= 0.0 && fabs(x - lastX) < 3 && lastY <= (10 + 14 * 3))
y = lastY + 3;
@@ -2125,18 +2195,10 @@ void ProfileWidget2::calculatePictureYPositions()
y = 10;
lastX = x;
lastY = y;
- it->thumbnail->setY(y);
-
- // hoverEnter and hoverExit events modify the z-value. Objects with different z-values
- // are not considered in stackBefore() calls. Therefore, just to be sure, reset the
- // z-values of all picture entries.
- it->thumbnail->setZValue(0.0);
+ e.thumbnail->setY(y);
+ updateDurationLine(e); // If we changed the y-position, we also have to change the duration-line.
}
-
- // Plot the items in the correct order. Experience showed that this works only
- // if we rearrange the items starting from the back. Therefore, use rbegin() and rend().
- for (auto it = pictures.rbegin(); std::next(it) != pictures.rend(); ++it)
- std::next(it)->thumbnail->stackBefore(it->thumbnail.get());
+ updateThumbnailPaintOrder();
}
void ProfileWidget2::updateThumbnailXPos(PictureEntry &e)