2012-10-02 09:12:39 +02:00
|
|
|
/****************************************************************************
|
2011-03-25 09:25:17 +01:00
|
|
|
**
|
2014-01-07 13:27:11 +01:00
|
|
|
** Copyright (C) 2014 Digia Plc and/or its subsidiary(-ies).
|
2012-10-02 09:12:39 +02:00
|
|
|
** Contact: http://www.qt-project.org/legal
|
2011-03-25 09:25:17 +01:00
|
|
|
**
|
2012-10-02 09:12:39 +02:00
|
|
|
** This file is part of Qt Creator.
|
2011-03-25 09:25:17 +01:00
|
|
|
**
|
2012-10-02 09:12:39 +02:00
|
|
|
** Commercial License Usage
|
|
|
|
|
** Licensees holding valid commercial Qt licenses may use this file in
|
|
|
|
|
** accordance with the commercial license agreement provided with the
|
|
|
|
|
** Software or, alternatively, in accordance with the terms contained in
|
|
|
|
|
** a written agreement between you and Digia. For licensing terms and
|
2014-10-01 13:21:18 +02:00
|
|
|
** conditions see http://www.qt.io/licensing. For further information
|
|
|
|
|
** use the contact form at http://www.qt.io/contact-us.
|
2011-03-25 09:25:17 +01:00
|
|
|
**
|
|
|
|
|
** GNU Lesser General Public License Usage
|
2012-10-02 09:12:39 +02:00
|
|
|
** Alternatively, this file may be used under the terms of the GNU Lesser
|
2014-10-01 13:21:18 +02:00
|
|
|
** General Public License version 2.1 or version 3 as published by the Free
|
|
|
|
|
** Software Foundation and appearing in the file LICENSE.LGPLv21 and
|
|
|
|
|
** LICENSE.LGPLv3 included in the packaging of this file. Please review the
|
|
|
|
|
** following information to ensure the GNU Lesser General Public License
|
|
|
|
|
** requirements will be met: https://www.gnu.org/licenses/lgpl.html and
|
|
|
|
|
** http://www.gnu.org/licenses/old-licenses/lgpl-2.1.html.
|
2012-10-02 09:12:39 +02:00
|
|
|
**
|
|
|
|
|
** In addition, as a special exception, Digia gives you certain additional
|
|
|
|
|
** rights. These rights are described in the Digia Qt LGPL Exception
|
2011-03-25 09:25:17 +01:00
|
|
|
** version 1.1, included in the file LGPL_EXCEPTION.txt in this package.
|
|
|
|
|
**
|
2012-10-02 09:12:39 +02:00
|
|
|
****************************************************************************/
|
2011-03-25 09:25:17 +01:00
|
|
|
|
2014-12-09 11:55:08 +01:00
|
|
|
#include "timelinerenderer_p.h"
|
2014-11-18 18:32:43 +01:00
|
|
|
#include "timelinerenderpass.h"
|
2014-10-27 17:41:22 +01:00
|
|
|
#include "qmlprofilernotesmodel.h"
|
2014-11-18 18:32:43 +01:00
|
|
|
#include "timelineitemsrenderpass.h"
|
|
|
|
|
#include "qmlprofilerbindingloopsrenderpass.h"
|
|
|
|
|
#include "timelineselectionrenderpass.h"
|
|
|
|
|
#include "timelinenotesrenderpass.h"
|
2011-03-11 12:22:57 +01:00
|
|
|
|
2014-11-18 18:32:43 +01:00
|
|
|
#include <QElapsedTimer>
|
2013-09-16 14:33:07 +02:00
|
|
|
#include <QQmlContext>
|
|
|
|
|
#include <QQmlProperty>
|
2012-02-15 10:42:41 +01:00
|
|
|
#include <QTimer>
|
|
|
|
|
#include <QPixmap>
|
|
|
|
|
#include <QGraphicsSceneMouseEvent>
|
2013-08-08 13:28:08 +02:00
|
|
|
#include <QVarLengthArray>
|
2014-11-18 18:32:43 +01:00
|
|
|
#include <QSGTransformNode>
|
|
|
|
|
#include <QSGSimpleRectNode>
|
2011-10-26 11:32:01 +02:00
|
|
|
|
|
|
|
|
#include <math.h>
|
2011-03-11 12:22:57 +01:00
|
|
|
|
2014-12-09 11:01:23 +01:00
|
|
|
namespace Timeline {
|
2011-03-25 09:21:00 +01:00
|
|
|
|
2014-12-09 11:55:08 +01:00
|
|
|
TimelineRenderer::TimelineRendererPrivate::TimelineRendererPrivate(TimelineRenderer *q) :
|
|
|
|
|
model(0), zoomer(0), notes(0), selectedItem(-1), selectionLocked(true), modelDirty(false),
|
|
|
|
|
rowHeightsDirty(false), rowCountsDirty(false), lastState(0), q_ptr(q)
|
|
|
|
|
{
|
|
|
|
|
resetCurrentSelection();
|
|
|
|
|
}
|
|
|
|
|
|
2014-11-18 18:32:43 +01:00
|
|
|
TimelineRenderer::TimelineRenderer(QQuickItem *parent) :
|
2014-12-09 11:55:08 +01:00
|
|
|
QQuickItem(parent), d_ptr(new TimelineRendererPrivate(this))
|
2011-03-11 12:22:57 +01:00
|
|
|
{
|
2014-11-18 18:32:43 +01:00
|
|
|
setFlag(QQuickItem::ItemHasContents);
|
2011-10-26 11:32:01 +02:00
|
|
|
setAcceptedMouseButtons(Qt::LeftButton);
|
|
|
|
|
setAcceptHoverEvents(true);
|
2013-08-08 13:28:08 +02:00
|
|
|
}
|
|
|
|
|
|
2014-12-09 11:55:08 +01:00
|
|
|
bool TimelineRenderer::selectionLocked() const
|
|
|
|
|
{
|
|
|
|
|
Q_D(const TimelineRenderer);
|
|
|
|
|
return d->selectionLocked;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
int TimelineRenderer::selectedItem() const
|
|
|
|
|
{
|
|
|
|
|
Q_D(const TimelineRenderer);
|
|
|
|
|
return d->selectedItem;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
TimelineModel *TimelineRenderer::model() const
|
|
|
|
|
{
|
|
|
|
|
Q_D(const TimelineRenderer);
|
|
|
|
|
return d->model;
|
|
|
|
|
}
|
|
|
|
|
|
2014-11-18 18:32:43 +01:00
|
|
|
void TimelineRenderer::setModel(TimelineModel *model)
|
2013-08-08 13:28:08 +02:00
|
|
|
{
|
2014-12-09 11:55:08 +01:00
|
|
|
Q_D(TimelineRenderer);
|
|
|
|
|
if (d->model == model)
|
2014-10-30 14:57:49 +01:00
|
|
|
return;
|
|
|
|
|
|
2014-12-09 11:55:08 +01:00
|
|
|
if (d->model) {
|
|
|
|
|
disconnect(d->model, SIGNAL(expandedChanged()), this, SLOT(update()));
|
|
|
|
|
disconnect(d->model, SIGNAL(hiddenChanged()), this, SLOT(update()));
|
|
|
|
|
disconnect(d->model, SIGNAL(expandedRowHeightChanged(int,int)),
|
2014-11-18 18:32:43 +01:00
|
|
|
this, SLOT(setRowHeightsDirty()));
|
2014-12-09 11:55:08 +01:00
|
|
|
disconnect(d->model, SIGNAL(emptyChanged()), this, SLOT(setModelDirty()));
|
|
|
|
|
disconnect(d->model, SIGNAL(expandedRowCountChanged()), this, SLOT(setRowCountsDirty()));
|
|
|
|
|
disconnect(d->model, SIGNAL(collapsedRowCountChanged()), this, SLOT(setRowCountsDirty()));
|
2014-06-18 15:47:20 +02:00
|
|
|
}
|
2014-10-30 14:57:49 +01:00
|
|
|
|
2014-12-09 11:55:08 +01:00
|
|
|
d->model = model;
|
|
|
|
|
if (d->model) {
|
|
|
|
|
connect(d->model, SIGNAL(expandedChanged()), this, SLOT(update()));
|
|
|
|
|
connect(d->model, SIGNAL(hiddenChanged()), this, SLOT(update()));
|
|
|
|
|
connect(d->model, SIGNAL(expandedRowHeightChanged(int,int)),
|
2014-11-18 18:32:43 +01:00
|
|
|
this, SLOT(setRowHeightsDirty()));
|
2014-12-09 11:55:08 +01:00
|
|
|
connect(d->model, SIGNAL(emptyChanged()), this, SLOT(setModelDirty()));
|
|
|
|
|
connect(d->model, SIGNAL(expandedRowCountChanged()), this, SLOT(setRowCountsDirty()));
|
|
|
|
|
connect(d->model, SIGNAL(collapsedRowCountChanged()), this, SLOT(setRowCountsDirty()));
|
|
|
|
|
d->renderPasses = d->model->supportedRenderPasses();
|
2014-06-18 15:47:20 +02:00
|
|
|
}
|
2014-10-30 14:57:49 +01:00
|
|
|
|
2014-11-18 18:32:43 +01:00
|
|
|
setModelDirty();
|
|
|
|
|
setRowHeightsDirty();
|
|
|
|
|
setRowCountsDirty();
|
2014-12-09 11:55:08 +01:00
|
|
|
emit modelChanged(d->model);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
TimelineZoomControl *TimelineRenderer::zoomer() const
|
|
|
|
|
{
|
|
|
|
|
Q_D(const TimelineRenderer);
|
|
|
|
|
return d->zoomer;
|
2011-03-11 12:22:57 +01:00
|
|
|
}
|
|
|
|
|
|
2014-10-30 14:57:49 +01:00
|
|
|
void TimelineRenderer::setZoomer(TimelineZoomControl *zoomer)
|
2014-10-14 16:44:45 +02:00
|
|
|
{
|
2014-12-09 11:55:08 +01:00
|
|
|
Q_D(TimelineRenderer);
|
|
|
|
|
if (zoomer != d->zoomer) {
|
|
|
|
|
if (d->zoomer != 0)
|
|
|
|
|
disconnect(d->zoomer, SIGNAL(windowChanged(qint64,qint64)), this, SLOT(update()));
|
|
|
|
|
d->zoomer = zoomer;
|
|
|
|
|
if (d->zoomer != 0)
|
|
|
|
|
connect(d->zoomer, SIGNAL(windowChanged(qint64,qint64)), this, SLOT(update()));
|
2014-10-14 16:44:45 +02:00
|
|
|
emit zoomerChanged(zoomer);
|
|
|
|
|
update();
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2014-12-09 11:55:08 +01:00
|
|
|
TimelineNotesModel *TimelineRenderer::notes() const
|
|
|
|
|
{
|
|
|
|
|
Q_D(const TimelineRenderer);
|
|
|
|
|
return d->notes;
|
|
|
|
|
}
|
|
|
|
|
|
2014-12-08 16:14:52 +01:00
|
|
|
void TimelineRenderer::setNotes(TimelineNotesModel *notes)
|
2011-03-11 12:22:57 +01:00
|
|
|
{
|
2014-12-09 11:55:08 +01:00
|
|
|
Q_D(TimelineRenderer);
|
|
|
|
|
if (d->notes == notes)
|
2014-10-30 14:57:49 +01:00
|
|
|
return;
|
2011-03-11 12:22:57 +01:00
|
|
|
|
2014-12-09 11:55:08 +01:00
|
|
|
if (d->notes)
|
|
|
|
|
disconnect(d->notes, &TimelineNotesModel::changed,
|
2014-11-18 18:32:43 +01:00
|
|
|
this, &TimelineRenderer::setNotesDirty);
|
2014-10-30 14:57:49 +01:00
|
|
|
|
2014-12-09 11:55:08 +01:00
|
|
|
d->notes = notes;
|
|
|
|
|
if (d->notes)
|
|
|
|
|
connect(d->notes, &TimelineNotesModel::changed,
|
2014-11-18 18:32:43 +01:00
|
|
|
this, &TimelineRenderer::setNotesDirty);
|
2014-10-30 14:57:49 +01:00
|
|
|
|
2014-12-09 11:55:08 +01:00
|
|
|
emit notesChanged(d->notes);
|
2011-10-26 11:32:01 +02:00
|
|
|
update();
|
2011-03-11 12:22:57 +01:00
|
|
|
}
|
|
|
|
|
|
2014-11-18 18:32:43 +01:00
|
|
|
bool TimelineRenderer::modelDirty() const
|
2014-08-27 12:11:26 +02:00
|
|
|
{
|
2014-12-09 11:55:08 +01:00
|
|
|
Q_D(const TimelineRenderer);
|
|
|
|
|
return d->modelDirty;
|
2014-08-27 12:11:26 +02:00
|
|
|
}
|
|
|
|
|
|
2014-11-18 18:32:43 +01:00
|
|
|
bool TimelineRenderer::notesDirty() const
|
2014-03-28 15:58:27 +01:00
|
|
|
{
|
2014-12-09 11:55:08 +01:00
|
|
|
Q_D(const TimelineRenderer);
|
|
|
|
|
return d->notesDirty;
|
2014-11-18 18:32:43 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
bool TimelineRenderer::rowHeightsDirty() const
|
|
|
|
|
{
|
2014-12-09 11:55:08 +01:00
|
|
|
Q_D(const TimelineRenderer);
|
|
|
|
|
return d->rowHeightsDirty;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
bool TimelineRenderer::rowCountsDirty() const
|
|
|
|
|
{
|
|
|
|
|
Q_D(const TimelineRenderer);
|
|
|
|
|
return d->rowCountsDirty;
|
2014-03-28 15:58:27 +01:00
|
|
|
}
|
|
|
|
|
|
2014-12-09 11:55:08 +01:00
|
|
|
void TimelineRenderer::TimelineRendererPrivate::resetCurrentSelection()
|
2014-07-09 17:43:10 +02:00
|
|
|
{
|
2014-12-09 11:55:08 +01:00
|
|
|
currentSelection.startTime = -1;
|
|
|
|
|
currentSelection.endTime = -1;
|
|
|
|
|
currentSelection.row = -1;
|
|
|
|
|
currentSelection.eventIndex = -1;
|
2014-07-09 17:43:10 +02:00
|
|
|
}
|
|
|
|
|
|
2014-12-09 11:55:08 +01:00
|
|
|
TimelineRenderState *TimelineRenderer::TimelineRendererPrivate::findRenderState()
|
2011-03-11 12:22:57 +01:00
|
|
|
{
|
2014-11-18 18:32:43 +01:00
|
|
|
int newLevel = 0;
|
|
|
|
|
int newOffset = 0;
|
|
|
|
|
int level;
|
|
|
|
|
int offset;
|
|
|
|
|
|
2014-12-09 11:55:08 +01:00
|
|
|
qint64 newStart = zoomer->traceStart();
|
|
|
|
|
qint64 newEnd = zoomer->traceEnd();
|
2014-11-18 18:32:43 +01:00
|
|
|
qint64 start;
|
|
|
|
|
qint64 end;
|
|
|
|
|
do {
|
|
|
|
|
level = newLevel;
|
|
|
|
|
offset = newOffset;
|
|
|
|
|
start = newStart;
|
|
|
|
|
end = newEnd;
|
|
|
|
|
|
|
|
|
|
newLevel = level + 1;
|
2014-12-09 11:55:08 +01:00
|
|
|
qint64 range = zoomer->traceDuration() >> newLevel;
|
|
|
|
|
newOffset = (zoomer->windowStart() - zoomer->traceStart() + range / 2) / range;
|
|
|
|
|
newStart = zoomer->traceStart() + newOffset * range - range / 2;
|
2014-11-18 18:32:43 +01:00
|
|
|
newEnd = newStart + range;
|
2014-12-09 11:55:08 +01:00
|
|
|
} while (newStart < zoomer->windowStart() && newEnd > zoomer->windowEnd());
|
2014-11-18 18:32:43 +01:00
|
|
|
|
|
|
|
|
|
2014-12-09 11:55:08 +01:00
|
|
|
if (renderStates.length() <= level)
|
|
|
|
|
renderStates.resize(level + 1);
|
|
|
|
|
if (renderStates[level].length() <= offset)
|
|
|
|
|
renderStates[level].resize(offset + 1);
|
|
|
|
|
TimelineRenderState *state = renderStates[level][offset];
|
2014-11-18 18:32:43 +01:00
|
|
|
if (state == 0) {
|
|
|
|
|
state = new TimelineRenderState(start, end, 1.0 / static_cast<qreal>(SafeFloatMax),
|
2014-12-09 11:55:08 +01:00
|
|
|
renderPasses.size());
|
|
|
|
|
renderStates[level][offset] = state;
|
2012-05-11 16:45:47 +02:00
|
|
|
}
|
2014-11-18 18:32:43 +01:00
|
|
|
return state;
|
2011-10-26 11:32:01 +02:00
|
|
|
}
|
2011-03-11 12:22:57 +01:00
|
|
|
|
2014-12-09 11:55:08 +01:00
|
|
|
QSGNode *TimelineRenderer::updatePaintNode(QSGNode *node, UpdatePaintNodeData *updatePaintNodeData)
|
2014-10-30 14:57:49 +01:00
|
|
|
{
|
2014-12-09 11:55:08 +01:00
|
|
|
Q_D(TimelineRenderer);
|
2014-11-18 18:32:43 +01:00
|
|
|
Q_UNUSED(updatePaintNodeData)
|
2014-10-30 14:57:49 +01:00
|
|
|
|
2014-12-09 11:55:08 +01:00
|
|
|
if (!d->model || d->model->hidden() || d->model->isEmpty() ||
|
|
|
|
|
d->zoomer->windowDuration() <= 0) {
|
2014-11-18 18:32:43 +01:00
|
|
|
delete node;
|
|
|
|
|
return 0;
|
|
|
|
|
} else if (node == 0) {
|
|
|
|
|
node = new QSGTransformNode;
|
|
|
|
|
}
|
2011-12-14 16:44:33 +01:00
|
|
|
|
2014-12-09 11:55:08 +01:00
|
|
|
qreal spacing = width() / d->zoomer->windowDuration();
|
2011-10-26 11:32:01 +02:00
|
|
|
|
2014-12-09 11:55:08 +01:00
|
|
|
if (d->modelDirty || d->rowCountsDirty) {
|
2014-11-18 18:32:43 +01:00
|
|
|
node->removeAllChildNodes();
|
2014-12-09 11:55:08 +01:00
|
|
|
foreach (QVector<TimelineRenderState *> stateVector, d->renderStates)
|
2014-11-18 18:32:43 +01:00
|
|
|
qDeleteAll(stateVector);
|
2014-12-09 11:55:08 +01:00
|
|
|
d->renderStates.clear();
|
|
|
|
|
d->lastState = 0;
|
2014-11-18 18:32:43 +01:00
|
|
|
}
|
2014-06-18 11:20:41 +02:00
|
|
|
|
2014-12-09 11:55:08 +01:00
|
|
|
TimelineRenderState *state = d->findRenderState();
|
2013-08-08 13:28:08 +02:00
|
|
|
|
2014-12-09 11:55:08 +01:00
|
|
|
int lastIndex = d->model->lastIndex(d->zoomer->windowEnd());
|
|
|
|
|
int firstIndex = d->model->firstIndex(d->zoomer->windowStart());
|
2014-03-28 15:58:27 +01:00
|
|
|
|
2014-12-09 11:55:08 +01:00
|
|
|
for (int i = 0; i < d->renderPasses.length(); ++i)
|
|
|
|
|
state->setPassState(i, d->renderPasses[i]->update(this, state, state->passState(i),
|
2014-11-18 18:32:43 +01:00
|
|
|
firstIndex, lastIndex + 1,
|
2014-12-09 11:55:08 +01:00
|
|
|
state != d->lastState, spacing));
|
2011-07-26 13:56:14 +02:00
|
|
|
|
2014-11-18 18:32:43 +01:00
|
|
|
if (state->isEmpty()) { // new state
|
2014-12-09 11:55:08 +01:00
|
|
|
for (int pass = 0; pass < d->renderPasses.length(); ++pass) {
|
2014-11-18 18:32:43 +01:00
|
|
|
const TimelineRenderPass::State *passState = state->passState(pass);
|
|
|
|
|
if (!passState)
|
|
|
|
|
continue;
|
|
|
|
|
if (passState->expandedOverlay)
|
|
|
|
|
state->expandedOverlayRoot()->appendChildNode(passState->expandedOverlay);
|
|
|
|
|
if (passState->collapsedOverlay)
|
|
|
|
|
state->collapsedOverlayRoot()->appendChildNode(passState->collapsedOverlay);
|
|
|
|
|
}
|
2011-07-26 13:56:14 +02:00
|
|
|
|
2014-11-18 18:32:43 +01:00
|
|
|
int row = 0;
|
2014-12-09 11:55:08 +01:00
|
|
|
for (int i = 0; i < d->model->expandedRowCount(); ++i) {
|
2014-11-18 18:32:43 +01:00
|
|
|
QSGTransformNode *rowNode = new QSGTransformNode;
|
2014-12-09 11:55:08 +01:00
|
|
|
for (int pass = 0; pass < d->renderPasses.length(); ++pass) {
|
2014-11-18 18:32:43 +01:00
|
|
|
const TimelineRenderPass::State *passState = state->passState(pass);
|
|
|
|
|
if (passState && passState->expandedRows.length() > row) {
|
|
|
|
|
QSGNode *rowChildNode = passState->expandedRows[row];
|
|
|
|
|
if (rowChildNode)
|
|
|
|
|
rowNode->appendChildNode(rowChildNode);
|
|
|
|
|
}
|
2014-11-04 11:53:50 +01:00
|
|
|
}
|
2014-11-18 18:32:43 +01:00
|
|
|
state->expandedRowRoot()->appendChildNode(rowNode);
|
|
|
|
|
++row;
|
2014-11-04 11:53:50 +01:00
|
|
|
}
|
|
|
|
|
|
2014-12-09 11:55:08 +01:00
|
|
|
for (int row = 0; row < d->model->collapsedRowCount(); ++row) {
|
2014-11-18 18:32:43 +01:00
|
|
|
QSGTransformNode *rowNode = new QSGTransformNode;
|
|
|
|
|
QMatrix4x4 matrix;
|
|
|
|
|
matrix.translate(0, row * TimelineModel::defaultRowHeight(), 0);
|
|
|
|
|
rowNode->setMatrix(matrix);
|
2014-12-09 11:55:08 +01:00
|
|
|
for (int pass = 0; pass < d->renderPasses.length(); ++pass) {
|
2014-11-18 18:32:43 +01:00
|
|
|
const TimelineRenderPass::State *passState = state->passState(pass);
|
|
|
|
|
if (passState && passState->collapsedRows.length() > row) {
|
|
|
|
|
QSGNode *rowChildNode = passState->collapsedRows[row];
|
|
|
|
|
if (rowChildNode)
|
|
|
|
|
rowNode->appendChildNode(rowChildNode);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
state->collapsedRowRoot()->appendChildNode(rowNode);
|
|
|
|
|
}
|
2011-10-26 11:32:01 +02:00
|
|
|
}
|
2013-08-08 13:28:08 +02:00
|
|
|
|
2014-12-09 11:55:08 +01:00
|
|
|
if (d->rowHeightsDirty || state != d->lastState) {
|
2014-11-18 18:32:43 +01:00
|
|
|
int row = 0;
|
|
|
|
|
qreal offset = 0;
|
|
|
|
|
for (QSGNode *rowNode = state->expandedRowRoot()->firstChild(); rowNode != 0;
|
|
|
|
|
rowNode = rowNode->nextSibling()) {
|
2014-12-09 11:55:08 +01:00
|
|
|
qreal rowHeight = d->model->expandedRowHeight(row++);
|
2014-11-18 18:32:43 +01:00
|
|
|
QMatrix4x4 matrix;
|
|
|
|
|
matrix.translate(0, offset, 0);
|
|
|
|
|
matrix.scale(1, rowHeight / TimelineModel::defaultRowHeight(), 1);
|
|
|
|
|
offset += rowHeight;
|
|
|
|
|
static_cast<QSGTransformNode *>(rowNode)->setMatrix(matrix);
|
|
|
|
|
}
|
|
|
|
|
}
|
2011-07-26 13:56:14 +02:00
|
|
|
|
2014-12-09 11:55:08 +01:00
|
|
|
d->modelDirty = false;
|
|
|
|
|
d->notesDirty = false;
|
|
|
|
|
d->rowCountsDirty = false;
|
|
|
|
|
d->rowHeightsDirty = false;
|
|
|
|
|
d->lastState = state;
|
2013-11-05 17:16:43 +01:00
|
|
|
|
2014-12-09 11:55:08 +01:00
|
|
|
QSGNode *rowNode = d->model->expanded() ? state->expandedRowRoot() : state->collapsedRowRoot();
|
|
|
|
|
QSGNode *overlayNode = d->model->expanded() ? state->expandedOverlayRoot() :
|
2014-11-18 18:32:43 +01:00
|
|
|
state->collapsedOverlayRoot();
|
2012-02-06 16:08:56 +01:00
|
|
|
|
2014-11-18 18:32:43 +01:00
|
|
|
QMatrix4x4 matrix;
|
2014-12-09 11:55:08 +01:00
|
|
|
matrix.translate((state->start() - d->zoomer->windowStart()) * spacing, 0, 0);
|
2014-11-18 18:32:43 +01:00
|
|
|
matrix.scale(spacing / state->scale(), 1, 1);
|
2012-02-06 16:08:56 +01:00
|
|
|
|
2014-11-18 18:32:43 +01:00
|
|
|
QSGTransformNode *transform = static_cast<QSGTransformNode *>(node);
|
|
|
|
|
transform->setMatrix(matrix);
|
|
|
|
|
|
|
|
|
|
if (node->firstChild() != rowNode || node->lastChild() != overlayNode) {
|
|
|
|
|
node->removeAllChildNodes();
|
|
|
|
|
node->appendChildNode(rowNode);
|
|
|
|
|
node->appendChildNode(overlayNode);
|
2012-02-06 16:08:56 +01:00
|
|
|
}
|
2014-11-18 18:32:43 +01:00
|
|
|
return node;
|
2012-02-06 16:08:56 +01:00
|
|
|
}
|
|
|
|
|
|
2014-11-18 18:32:43 +01:00
|
|
|
void TimelineRenderer::mousePressEvent(QMouseEvent *event)
|
2014-09-24 18:25:02 +02:00
|
|
|
{
|
2014-11-18 18:32:43 +01:00
|
|
|
Q_UNUSED(event);
|
2014-09-24 18:25:02 +02:00
|
|
|
}
|
|
|
|
|
|
2014-12-09 11:55:08 +01:00
|
|
|
int TimelineRenderer::TimelineRendererPrivate::rowFromPosition(int y) const
|
2014-06-18 11:20:41 +02:00
|
|
|
{
|
2014-12-09 11:55:08 +01:00
|
|
|
if (!model->expanded())
|
2014-12-04 15:06:17 +01:00
|
|
|
return y / TimelineModel::defaultRowHeight();
|
2014-06-18 11:20:41 +02:00
|
|
|
|
2014-12-04 15:06:17 +01:00
|
|
|
int ret = 0;
|
2014-12-09 11:55:08 +01:00
|
|
|
for (int row = 0; row < model->expandedRowCount(); ++row) {
|
|
|
|
|
y -= model->expandedRowHeight(row);
|
2014-12-04 15:06:17 +01:00
|
|
|
if (y <= 0) return ret;
|
2014-10-30 14:57:49 +01:00
|
|
|
++ret;
|
2013-08-08 13:28:08 +02:00
|
|
|
}
|
2011-10-26 11:32:01 +02:00
|
|
|
|
2014-10-30 14:57:49 +01:00
|
|
|
return ret;
|
2011-07-26 13:56:14 +02:00
|
|
|
}
|
|
|
|
|
|
2013-09-16 14:33:07 +02:00
|
|
|
void TimelineRenderer::mouseReleaseEvent(QMouseEvent *event)
|
2011-07-26 13:56:14 +02:00
|
|
|
{
|
2014-12-09 11:55:08 +01:00
|
|
|
Q_D(TimelineRenderer);
|
2011-10-26 11:32:01 +02:00
|
|
|
Q_UNUSED(event);
|
2014-12-09 11:55:08 +01:00
|
|
|
if (!d->model->isEmpty())
|
|
|
|
|
d->manageClicked();
|
2011-07-26 13:56:14 +02:00
|
|
|
}
|
|
|
|
|
|
2013-09-16 14:33:07 +02:00
|
|
|
void TimelineRenderer::mouseMoveEvent(QMouseEvent *event)
|
2011-07-26 13:56:14 +02:00
|
|
|
{
|
2011-10-26 11:32:01 +02:00
|
|
|
event->setAccepted(false);
|
|
|
|
|
}
|
2011-07-26 13:56:14 +02:00
|
|
|
|
2013-09-16 14:33:07 +02:00
|
|
|
void TimelineRenderer::hoverMoveEvent(QHoverEvent *event)
|
2011-10-26 11:32:01 +02:00
|
|
|
{
|
2014-12-09 11:55:08 +01:00
|
|
|
Q_D(TimelineRenderer);
|
2011-10-26 11:32:01 +02:00
|
|
|
Q_UNUSED(event);
|
2014-12-09 11:55:08 +01:00
|
|
|
d->manageHovered(event->pos().x(), event->pos().y());
|
|
|
|
|
if (d->currentSelection.eventIndex == -1)
|
2011-10-26 11:32:01 +02:00
|
|
|
event->setAccepted(false);
|
2011-07-26 13:56:14 +02:00
|
|
|
}
|
|
|
|
|
|
2014-12-09 11:55:08 +01:00
|
|
|
void TimelineRenderer::TimelineRendererPrivate::manageClicked()
|
2011-07-26 13:56:14 +02:00
|
|
|
{
|
2014-12-09 11:55:08 +01:00
|
|
|
Q_Q(TimelineRenderer);
|
|
|
|
|
if (currentSelection.eventIndex != -1) {
|
|
|
|
|
if (currentSelection.eventIndex == selectedItem)
|
|
|
|
|
q->setSelectionLocked(!selectionLocked);
|
2011-10-26 11:32:01 +02:00
|
|
|
else
|
2014-12-09 11:55:08 +01:00
|
|
|
q->setSelectionLocked(true);
|
2014-04-03 11:15:56 +02:00
|
|
|
|
|
|
|
|
// itemPressed() will trigger an update of the events and JavaScript views. Make sure the
|
|
|
|
|
// correct event is already selected when that happens, to prevent confusion.
|
2014-12-09 11:55:08 +01:00
|
|
|
q->setSelectedItem(currentSelection.eventIndex);
|
|
|
|
|
emit q->itemPressed(currentSelection.eventIndex);
|
2011-10-26 11:32:01 +02:00
|
|
|
} else {
|
2014-12-09 11:55:08 +01:00
|
|
|
q->setSelectionLocked(false);
|
|
|
|
|
q->setSelectedItem(-1);
|
|
|
|
|
emit q->itemPressed(-1);
|
2011-07-26 13:56:14 +02:00
|
|
|
}
|
2011-10-26 11:32:01 +02:00
|
|
|
}
|
2011-07-26 13:56:14 +02:00
|
|
|
|
2014-12-09 11:55:08 +01:00
|
|
|
void TimelineRenderer::TimelineRendererPrivate::manageHovered(int mouseX, int mouseY)
|
2011-10-26 11:32:01 +02:00
|
|
|
{
|
2014-12-09 11:55:08 +01:00
|
|
|
Q_Q(TimelineRenderer);
|
|
|
|
|
qint64 duration = zoomer->windowDuration();
|
2014-10-14 16:44:45 +02:00
|
|
|
if (duration <= 0)
|
2011-10-26 11:32:01 +02:00
|
|
|
return;
|
2011-07-26 13:56:14 +02:00
|
|
|
|
2014-03-11 12:16:37 +01:00
|
|
|
// Make the "selected" area 3 pixels wide by adding/subtracting 1 to catch very narrow events.
|
2014-12-09 11:55:08 +01:00
|
|
|
qint64 startTime = (mouseX - 1) * duration / q->width() + zoomer->windowStart();
|
|
|
|
|
qint64 endTime = (mouseX + 1) * duration / q->width() + zoomer->windowStart();
|
2014-12-04 13:24:49 +01:00
|
|
|
qint64 exactTime = (startTime + endTime) / 2;
|
2014-11-18 18:32:43 +01:00
|
|
|
int row = rowFromPosition(mouseY);
|
2011-07-26 13:56:14 +02:00
|
|
|
|
2014-12-09 11:55:08 +01:00
|
|
|
// already covered? Only recheck selectionLocked and make sure d->selectedItem is correct.
|
|
|
|
|
if (currentSelection.eventIndex != -1 &&
|
|
|
|
|
exactTime >= currentSelection.startTime &&
|
|
|
|
|
exactTime < currentSelection.endTime &&
|
|
|
|
|
row == currentSelection.row) {
|
|
|
|
|
if (!selectionLocked)
|
|
|
|
|
q->setSelectedItem(currentSelection.eventIndex);
|
2011-10-26 11:32:01 +02:00
|
|
|
return;
|
|
|
|
|
}
|
2011-07-26 13:56:14 +02:00
|
|
|
|
2011-10-26 11:32:01 +02:00
|
|
|
// find if there's items in the time range
|
2014-12-09 11:55:08 +01:00
|
|
|
int eventFrom = model->firstIndex(startTime);
|
|
|
|
|
int eventTo = model->lastIndex(endTime);
|
2014-12-04 13:24:49 +01:00
|
|
|
|
2014-12-09 11:55:08 +01:00
|
|
|
currentSelection.eventIndex = -1;
|
|
|
|
|
if (eventFrom == -1 || eventTo < eventFrom || eventTo >= model->count())
|
2011-07-26 13:56:14 +02:00
|
|
|
return;
|
|
|
|
|
|
2011-10-26 11:32:01 +02:00
|
|
|
// find if we are in the right column
|
2014-12-04 13:24:49 +01:00
|
|
|
qint64 bestOffset = std::numeric_limits<qint64>::max();
|
2011-10-26 11:32:01 +02:00
|
|
|
for (int i=eventTo; i>=eventFrom; --i) {
|
2014-12-09 11:55:08 +01:00
|
|
|
if (model->row(i) == row) {
|
2014-03-26 17:06:26 +01:00
|
|
|
// There can be small events that don't reach the cursor position after large events
|
|
|
|
|
// that do but are in a different row.
|
2014-12-09 11:55:08 +01:00
|
|
|
qint64 itemEnd = model->endTime(i);
|
2014-03-26 17:06:26 +01:00
|
|
|
if (itemEnd < startTime)
|
|
|
|
|
continue;
|
|
|
|
|
|
2014-12-09 11:55:08 +01:00
|
|
|
qint64 itemStart = model->startTime(i);
|
2014-12-04 13:24:49 +01:00
|
|
|
|
|
|
|
|
qint64 offset = qAbs(itemEnd - exactTime) + qAbs(itemStart - exactTime);
|
|
|
|
|
if (offset < bestOffset) {
|
|
|
|
|
// match
|
2014-12-09 11:55:08 +01:00
|
|
|
currentSelection.eventIndex = i;
|
|
|
|
|
currentSelection.startTime = itemStart;
|
|
|
|
|
currentSelection.endTime = itemEnd;
|
|
|
|
|
currentSelection.row = row;
|
2014-12-04 13:24:49 +01:00
|
|
|
bestOffset = offset;
|
|
|
|
|
}
|
2011-10-26 11:32:01 +02:00
|
|
|
}
|
2011-07-26 13:56:14 +02:00
|
|
|
}
|
2014-12-09 11:55:08 +01:00
|
|
|
if (!selectionLocked && currentSelection.eventIndex != -1)
|
|
|
|
|
q->setSelectedItem(currentSelection.eventIndex);
|
2011-10-26 11:32:01 +02:00
|
|
|
}
|
|
|
|
|
|
2012-02-24 10:47:17 +01:00
|
|
|
void TimelineRenderer::clearData()
|
2011-10-26 11:32:01 +02:00
|
|
|
{
|
2014-12-09 11:55:08 +01:00
|
|
|
Q_D(TimelineRenderer);
|
|
|
|
|
d->resetCurrentSelection();
|
2014-07-09 17:43:10 +02:00
|
|
|
setSelectedItem(-1);
|
|
|
|
|
setSelectionLocked(true);
|
2011-07-26 13:56:14 +02:00
|
|
|
}
|
|
|
|
|
|
2014-12-09 11:55:08 +01:00
|
|
|
void TimelineRenderer::setSelectedItem(int itemIndex)
|
|
|
|
|
{
|
|
|
|
|
Q_D(TimelineRenderer);
|
|
|
|
|
if (d->selectedItem != itemIndex) {
|
|
|
|
|
d->selectedItem = itemIndex;
|
|
|
|
|
update();
|
|
|
|
|
emit selectedItemChanged(itemIndex);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void TimelineRenderer::setSelectionLocked(bool locked)
|
|
|
|
|
{
|
|
|
|
|
Q_D(TimelineRenderer);
|
|
|
|
|
if (d->selectionLocked != locked) {
|
|
|
|
|
d->selectionLocked = locked;
|
|
|
|
|
update();
|
|
|
|
|
emit selectionLockedChanged(locked);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2014-10-30 14:57:49 +01:00
|
|
|
void TimelineRenderer::selectNextFromSelectionId(int selectionId)
|
2014-03-26 11:47:52 +01:00
|
|
|
{
|
2014-12-09 11:55:08 +01:00
|
|
|
Q_D(TimelineRenderer);
|
|
|
|
|
setSelectedItem(d->model->nextItemBySelectionId(selectionId, d->zoomer->rangeStart(),
|
|
|
|
|
d->selectedItem));
|
2014-03-26 11:47:52 +01:00
|
|
|
}
|
|
|
|
|
|
2014-10-30 14:57:49 +01:00
|
|
|
void TimelineRenderer::selectPrevFromSelectionId(int selectionId)
|
2011-10-26 11:32:01 +02:00
|
|
|
{
|
2014-12-09 11:55:08 +01:00
|
|
|
Q_D(TimelineRenderer);
|
|
|
|
|
setSelectedItem(d->model->prevItemBySelectionId(selectionId, d->zoomer->rangeStart(),
|
|
|
|
|
d->selectedItem));
|
2011-10-26 11:32:01 +02:00
|
|
|
}
|
2014-11-18 18:32:43 +01:00
|
|
|
|
|
|
|
|
void TimelineRenderer::setModelDirty()
|
|
|
|
|
{
|
2014-12-09 11:55:08 +01:00
|
|
|
Q_D(TimelineRenderer);
|
|
|
|
|
d->modelDirty = true;
|
2014-11-18 18:32:43 +01:00
|
|
|
update();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void TimelineRenderer::setRowHeightsDirty()
|
|
|
|
|
{
|
2014-12-09 11:55:08 +01:00
|
|
|
Q_D(TimelineRenderer);
|
|
|
|
|
d->rowHeightsDirty = true;
|
2014-11-18 18:32:43 +01:00
|
|
|
update();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void TimelineRenderer::setNotesDirty()
|
|
|
|
|
{
|
2014-12-09 11:55:08 +01:00
|
|
|
Q_D(TimelineRenderer);
|
|
|
|
|
d->notesDirty = true;
|
2014-11-18 18:32:43 +01:00
|
|
|
update();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void TimelineRenderer::setRowCountsDirty()
|
|
|
|
|
{
|
2014-12-09 11:55:08 +01:00
|
|
|
Q_D(TimelineRenderer);
|
|
|
|
|
d->rowCountsDirty = true;
|
|
|
|
|
update();
|
2014-11-18 18:32:43 +01:00
|
|
|
}
|
|
|
|
|
|
2014-12-09 11:01:23 +01:00
|
|
|
} // namespace Timeline
|