mirror of
https://github.com/subsurface/subsurface.git
synced 2024-12-11 03:21:29 +00:00
f0a89759bf
The tooltip's plotInfo was not cleared when clearing the profile. With the new cylinder code, this lead to crashes, because the displayed_dive's cylinder array is now cleared. The old code would happily read stale data from the fixed-size cylinders array. Clear the plotInfo explicitly. Signed-off-by: Berthold Stoeger <bstoeger@mail.tuwien.ac.at>
294 lines
7.7 KiB
C++
294 lines
7.7 KiB
C++
// SPDX-License-Identifier: GPL-2.0
|
|
#include "profile-widget/divetooltipitem.h"
|
|
#include "profile-widget/divecartesianaxis.h"
|
|
#include "core/profile.h"
|
|
#include "core/membuffer.h"
|
|
#include "core/metrics.h"
|
|
#include "core/settings/qPrefDisplay.h"
|
|
#include <QPropertyAnimation>
|
|
#include <QGraphicsView>
|
|
#include <QStyleOptionGraphicsItem>
|
|
#include "core/qthelper.h"
|
|
|
|
void ToolTipItem::addToolTip(const QString &toolTip, const QIcon &icon, const QPixmap& pixmap)
|
|
{
|
|
const IconMetrics& iconMetrics = defaultIconMetrics();
|
|
|
|
QGraphicsPixmapItem *iconItem = 0;
|
|
double yValue = title->boundingRect().height() + iconMetrics.spacing;
|
|
Q_FOREACH (ToolTip t, toolTips) {
|
|
yValue += t.second->boundingRect().height();
|
|
}
|
|
if (entryToolTip.second) {
|
|
yValue += entryToolTip.second->boundingRect().height();
|
|
}
|
|
iconItem = new QGraphicsPixmapItem(this);
|
|
if (!icon.isNull()) {
|
|
iconItem->setPixmap(icon.pixmap(iconMetrics.sz_small, iconMetrics.sz_small));
|
|
} else if (!pixmap.isNull()) {
|
|
iconItem->setPixmap(pixmap);
|
|
}
|
|
const int sp2 = iconMetrics.spacing * 2;
|
|
iconItem->setPos(sp2, yValue);
|
|
|
|
QGraphicsSimpleTextItem *textItem = new QGraphicsSimpleTextItem(toolTip, this);
|
|
textItem->setPos(sp2 + iconMetrics.sz_small + sp2, yValue);
|
|
textItem->setBrush(QBrush(Qt::white));
|
|
textItem->setFlag(ItemIgnoresTransformations);
|
|
toolTips.push_back(qMakePair(iconItem, textItem));
|
|
}
|
|
|
|
void ToolTipItem::clear()
|
|
{
|
|
Q_FOREACH (ToolTip t, toolTips) {
|
|
delete t.first;
|
|
delete t.second;
|
|
}
|
|
toolTips.clear();
|
|
}
|
|
|
|
void ToolTipItem::setRect(const QRectF &r)
|
|
{
|
|
if( r == rect() ) {
|
|
return;
|
|
}
|
|
|
|
QGraphicsRectItem::setRect(r);
|
|
updateTitlePosition();
|
|
}
|
|
|
|
void ToolTipItem::collapse()
|
|
{
|
|
int dim = defaultIconMetrics().sz_small;
|
|
|
|
if (qPrefDisplay::animation_speed()) {
|
|
QPropertyAnimation *animation = new QPropertyAnimation(this, "rect");
|
|
animation->setDuration(100);
|
|
animation->setStartValue(nextRectangle);
|
|
animation->setEndValue(QRect(0, 0, dim, dim));
|
|
animation->start(QAbstractAnimation::DeleteWhenStopped);
|
|
} else {
|
|
setRect(nextRectangle);
|
|
}
|
|
clear();
|
|
|
|
status = COLLAPSED;
|
|
}
|
|
|
|
void ToolTipItem::expand()
|
|
{
|
|
if (!title)
|
|
return;
|
|
|
|
const IconMetrics& iconMetrics = defaultIconMetrics();
|
|
|
|
double width = 0, height = title->boundingRect().height() + iconMetrics.spacing;
|
|
Q_FOREACH (const ToolTip& t, toolTips) {
|
|
QRectF sRect = t.second->boundingRect();
|
|
if (sRect.width() > width)
|
|
width = sRect.width();
|
|
height += sRect.height();
|
|
}
|
|
|
|
if (entryToolTip.first) {
|
|
QRectF sRect = entryToolTip.second->boundingRect();
|
|
if (sRect.width() > width)
|
|
width = sRect.width();
|
|
height += sRect.height();
|
|
}
|
|
|
|
const int sp2 = iconMetrics.spacing * 2;
|
|
// pixmap left padding, icon, pixmap right padding, right padding */
|
|
width += sp2 + iconMetrics.sz_small + sp2 + sp2 * 2;
|
|
// bottom padding
|
|
height += sp2;
|
|
|
|
// clip the tooltip width
|
|
if (width < title->boundingRect().width() + sp2)
|
|
width = title->boundingRect().width() + sp2;
|
|
// clip the height
|
|
if (entryToolTip.first) {
|
|
const int minH = lrint(entryToolTip.first->y() + entryToolTip.first->pixmap().height() + sp2);
|
|
if (height < minH)
|
|
height = minH;
|
|
} else if (height < iconMetrics.sz_small) {
|
|
height = iconMetrics.sz_small;
|
|
}
|
|
|
|
nextRectangle.setWidth(width);
|
|
nextRectangle.setHeight(height);
|
|
|
|
if (nextRectangle != rect()) {
|
|
if (qPrefDisplay::animation_speed()) {
|
|
QPropertyAnimation *animation = new QPropertyAnimation(this, "rect", this);
|
|
animation->setDuration(prefs.animation_speed);
|
|
animation->setStartValue(rect());
|
|
animation->setEndValue(nextRectangle);
|
|
animation->start(QAbstractAnimation::DeleteWhenStopped);
|
|
} else {
|
|
setRect(nextRectangle);
|
|
}
|
|
}
|
|
|
|
status = EXPANDED;
|
|
}
|
|
|
|
ToolTipItem::ToolTipItem(QGraphicsItem *parent) : QGraphicsRectItem(parent),
|
|
title(new QGraphicsSimpleTextItem(tr("Information"), this)),
|
|
status(COLLAPSED),
|
|
timeAxis(0),
|
|
lastTime(-1)
|
|
{
|
|
clearPlotInfo();
|
|
entryToolTip.first = NULL;
|
|
entryToolTip.second = NULL;
|
|
setFlags(ItemIgnoresTransformations | ItemIsMovable | ItemClipsChildrenToShape);
|
|
|
|
QColor c = QColor(Qt::black);
|
|
c.setAlpha(155);
|
|
setBrush(c);
|
|
|
|
setZValue(99);
|
|
|
|
addToolTip(QString(), QIcon(), QPixmap(16,60));
|
|
entryToolTip = toolTips.first();
|
|
toolTips.clear();
|
|
|
|
title->setFlag(ItemIgnoresTransformations);
|
|
title->setPen(QPen(Qt::white, 1));
|
|
title->setBrush(Qt::white);
|
|
|
|
setPen(QPen(Qt::white, 2));
|
|
refreshTime.start();
|
|
}
|
|
|
|
ToolTipItem::~ToolTipItem()
|
|
{
|
|
clear();
|
|
}
|
|
|
|
void ToolTipItem::updateTitlePosition()
|
|
{
|
|
const IconMetrics& iconMetrics = defaultIconMetrics();
|
|
if (rect().width() < title->boundingRect().width() + iconMetrics.spacing * 4) {
|
|
QRectF newRect = rect();
|
|
newRect.setWidth(title->boundingRect().width() + iconMetrics.spacing * 4);
|
|
newRect.setHeight((newRect.height() && isExpanded()) ? newRect.height() : iconMetrics.sz_small);
|
|
setRect(newRect);
|
|
}
|
|
|
|
title->setPos(rect().width() / 2 - title->boundingRect().width() / 2 - 1, 0);
|
|
}
|
|
|
|
bool ToolTipItem::isExpanded() const
|
|
{
|
|
return status == EXPANDED;
|
|
}
|
|
|
|
void ToolTipItem::mouseReleaseEvent(QGraphicsSceneMouseEvent *event)
|
|
{
|
|
persistPos();
|
|
QGraphicsRectItem::mouseReleaseEvent(event);
|
|
Q_FOREACH (QGraphicsItem *item, oldSelection) {
|
|
item->setSelected(true);
|
|
}
|
|
}
|
|
|
|
void ToolTipItem::paint(QPainter *painter, const QStyleOptionGraphicsItem *option, QWidget*)
|
|
{
|
|
painter->save();
|
|
painter->setClipRect(option->rect);
|
|
painter->setPen(pen());
|
|
painter->setBrush(brush());
|
|
painter->drawRoundedRect(rect(), 8, 8, Qt::AbsoluteSize);
|
|
painter->restore();
|
|
}
|
|
|
|
void ToolTipItem::persistPos()
|
|
{
|
|
qPrefDisplay::set_tooltip_position(pos());
|
|
}
|
|
|
|
void ToolTipItem::readPos()
|
|
{
|
|
QPointF value = qPrefDisplay::tooltip_position();
|
|
if (!scene()->sceneRect().contains(value)) {
|
|
value = QPointF(0, 0);
|
|
}
|
|
setPos(value);
|
|
}
|
|
|
|
void ToolTipItem::setPlotInfo(const plot_info &plot)
|
|
{
|
|
pInfo = plot;
|
|
}
|
|
|
|
void ToolTipItem::clearPlotInfo()
|
|
{
|
|
memset(&pInfo, 0, sizeof(pInfo));
|
|
}
|
|
|
|
void ToolTipItem::setTimeAxis(DiveCartesianAxis *axis)
|
|
{
|
|
timeAxis = axis;
|
|
}
|
|
|
|
void ToolTipItem::refresh(const QPointF &pos)
|
|
{
|
|
static QPixmap tissues(16,60);
|
|
static QPainter painter(&tissues);
|
|
static struct membuffer mb = {};
|
|
|
|
if(refreshTime.elapsed() < 40)
|
|
return;
|
|
refreshTime.start();
|
|
|
|
int time = lrint(timeAxis->valueAt(pos));
|
|
|
|
lastTime = time;
|
|
clear();
|
|
|
|
mb.len = 0;
|
|
int idx = get_plot_details_new(&pInfo, time, &mb);
|
|
|
|
tissues.fill();
|
|
painter.setPen(QColor(0, 0, 0, 0));
|
|
painter.setBrush(QColor(LIMENADE1));
|
|
painter.drawRect(0, 10 + (100 - AMB_PERCENTAGE) / 2, 16, AMB_PERCENTAGE / 2);
|
|
painter.setBrush(QColor(SPRINGWOOD1));
|
|
painter.drawRect(0, 10, 16, (100 - AMB_PERCENTAGE) / 2);
|
|
painter.setBrush(QColor(Qt::red));
|
|
painter.drawRect(0,0,16,10);
|
|
if (idx) {
|
|
ProfileWidget2 *view = qobject_cast<ProfileWidget2*>(scene()->views().first());
|
|
Q_ASSERT(view);
|
|
|
|
const struct plot_data *entry = &pInfo.entry[idx];
|
|
painter.setPen(QColor(0, 0, 0, 255));
|
|
if (decoMode() == BUEHLMANN)
|
|
painter.drawLine(0, lrint(60 - entry->gfline / 2), 16, lrint(60 - entry->gfline / 2));
|
|
painter.drawLine(0, lrint(60 - AMB_PERCENTAGE * (entry->pressures.n2 + entry->pressures.he) / entry->ambpressure / 2),
|
|
16, lrint(60 - AMB_PERCENTAGE * (entry->pressures.n2 + entry->pressures.he) / entry->ambpressure /2));
|
|
painter.setPen(QColor(0, 0, 0, 127));
|
|
for (int i=0; i<16; i++) {
|
|
painter.drawLine(i, 60, i, 60 - entry->percentages[i] / 2);
|
|
}
|
|
entryToolTip.second->setText(QString::fromUtf8(mb.buffer, mb.len));
|
|
}
|
|
entryToolTip.first->setPixmap(tissues);
|
|
|
|
const auto l = scene()->items(pos, Qt::IntersectsItemBoundingRect, Qt::DescendingOrder,
|
|
scene()->views().first()->transform());
|
|
for (QGraphicsItem *item: l) {
|
|
if (!item->toolTip().isEmpty())
|
|
addToolTip(item->toolTip());
|
|
}
|
|
expand();
|
|
}
|
|
|
|
void ToolTipItem::mousePressEvent(QGraphicsSceneMouseEvent *event)
|
|
{
|
|
oldSelection = scene()->selectedItems();
|
|
scene()->clearSelection();
|
|
QGraphicsItem::mousePressEvent(event);
|
|
}
|