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
|
|
|
|
2012-02-24 10:47:17 +01:00
|
|
|
#include "timelinerenderer.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-10-30 14:57:49 +01:00
|
|
|
using namespace QmlProfiler;
|
2011-03-25 09:21:00 +01:00
|
|
|
using namespace QmlProfiler::Internal;
|
|
|
|
|
|
2014-11-18 18:32:43 +01:00
|
|
|
TimelineRenderer::TimelineRenderer(QQuickItem *parent) :
|
|
|
|
|
QQuickItem(parent), m_model(0), m_zoomer(0), m_notes(0),
|
|
|
|
|
m_selectedItem(-1), m_selectionLocked(true), m_modelDirty(false),
|
|
|
|
|
m_rowHeightsDirty(false), m_rowCountsDirty(false), m_lastState(0)
|
2011-03-11 12:22:57 +01:00
|
|
|
{
|
2014-11-18 18:32:43 +01:00
|
|
|
setFlag(QQuickItem::ItemHasContents);
|
2014-07-09 17:43:10 +02:00
|
|
|
resetCurrentSelection();
|
2011-10-26 11:32:01 +02:00
|
|
|
setAcceptedMouseButtons(Qt::LeftButton);
|
|
|
|
|
setAcceptHoverEvents(true);
|
2013-08-08 13:28:08 +02:00
|
|
|
}
|
|
|
|
|
|
2014-11-18 18:32:43 +01:00
|
|
|
void TimelineRenderer::setModel(TimelineModel *model)
|
2013-08-08 13:28:08 +02:00
|
|
|
{
|
2014-10-30 14:57:49 +01:00
|
|
|
if (m_model == model)
|
|
|
|
|
return;
|
|
|
|
|
|
|
|
|
|
if (m_model) {
|
2014-11-18 18:32:43 +01:00
|
|
|
disconnect(m_model, SIGNAL(expandedChanged()), this, SLOT(update()));
|
|
|
|
|
disconnect(m_model, SIGNAL(hiddenChanged()), this, SLOT(update()));
|
|
|
|
|
disconnect(m_model, SIGNAL(expandedRowHeightChanged(int,int)),
|
|
|
|
|
this, SLOT(setRowHeightsDirty()));
|
|
|
|
|
disconnect(m_model, SIGNAL(emptyChanged()), this, SLOT(setModelDirty()));
|
|
|
|
|
disconnect(m_model, SIGNAL(expandedRowCountChanged()), this, SLOT(setRowCountsDirty()));
|
|
|
|
|
disconnect(m_model, SIGNAL(collapsedRowCountChanged()), this, SLOT(setRowCountsDirty()));
|
2014-06-18 15:47:20 +02:00
|
|
|
}
|
2014-10-30 14:57:49 +01:00
|
|
|
|
|
|
|
|
m_model = model;
|
|
|
|
|
if (m_model) {
|
2014-11-18 18:32:43 +01:00
|
|
|
connect(m_model, SIGNAL(expandedChanged()), this, SLOT(update()));
|
|
|
|
|
connect(m_model, SIGNAL(hiddenChanged()), this, SLOT(update()));
|
|
|
|
|
connect(m_model, SIGNAL(expandedRowHeightChanged(int,int)),
|
|
|
|
|
this, SLOT(setRowHeightsDirty()));
|
|
|
|
|
connect(m_model, SIGNAL(emptyChanged()), this, SLOT(setModelDirty()));
|
|
|
|
|
connect(m_model, SIGNAL(expandedRowCountChanged()), this, SLOT(setRowCountsDirty()));
|
|
|
|
|
connect(m_model, SIGNAL(collapsedRowCountChanged()), this, SLOT(setRowCountsDirty()));
|
|
|
|
|
m_renderPasses = 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-10-30 14:57:49 +01:00
|
|
|
emit modelChanged(m_model);
|
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
|
|
|
{
|
|
|
|
|
if (zoomer != m_zoomer) {
|
|
|
|
|
if (m_zoomer != 0)
|
2014-11-18 18:32:43 +01:00
|
|
|
disconnect(m_zoomer, SIGNAL(windowChanged(qint64,qint64)), this, SLOT(update()));
|
2014-10-14 16:44:45 +02:00
|
|
|
m_zoomer = zoomer;
|
|
|
|
|
if (m_zoomer != 0)
|
2014-11-18 18:32:43 +01:00
|
|
|
connect(m_zoomer, SIGNAL(windowChanged(qint64,qint64)), this, SLOT(update()));
|
2014-10-14 16:44:45 +02:00
|
|
|
emit zoomerChanged(zoomer);
|
|
|
|
|
update();
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2014-12-08 16:14:52 +01:00
|
|
|
void TimelineRenderer::setNotes(TimelineNotesModel *notes)
|
2011-03-11 12:22:57 +01:00
|
|
|
{
|
2014-10-30 14:57:49 +01:00
|
|
|
if (m_notes == notes)
|
|
|
|
|
return;
|
2011-03-11 12:22:57 +01:00
|
|
|
|
2014-10-30 14:57:49 +01:00
|
|
|
if (m_notes)
|
2014-12-08 16:14:52 +01:00
|
|
|
disconnect(m_notes, &TimelineNotesModel::changed,
|
2014-11-18 18:32:43 +01:00
|
|
|
this, &TimelineRenderer::setNotesDirty);
|
2014-10-30 14:57:49 +01:00
|
|
|
|
|
|
|
|
m_notes = notes;
|
|
|
|
|
if (m_notes)
|
2014-12-08 16:14:52 +01:00
|
|
|
connect(m_notes, &TimelineNotesModel::changed,
|
2014-11-18 18:32:43 +01:00
|
|
|
this, &TimelineRenderer::setNotesDirty);
|
2014-10-30 14:57:49 +01:00
|
|
|
|
|
|
|
|
emit notesChanged(m_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-11-18 18:32:43 +01:00
|
|
|
return m_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-11-18 18:32:43 +01:00
|
|
|
return m_notesDirty;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
bool TimelineRenderer::rowHeightsDirty() const
|
|
|
|
|
{
|
|
|
|
|
return m_rowHeightsDirty;
|
2014-03-28 15:58:27 +01:00
|
|
|
}
|
|
|
|
|
|
2014-07-09 17:43:10 +02:00
|
|
|
void TimelineRenderer::resetCurrentSelection()
|
|
|
|
|
{
|
|
|
|
|
m_currentSelection.startTime = -1;
|
|
|
|
|
m_currentSelection.endTime = -1;
|
|
|
|
|
m_currentSelection.row = -1;
|
|
|
|
|
m_currentSelection.eventIndex = -1;
|
|
|
|
|
}
|
|
|
|
|
|
2014-11-18 18:32:43 +01:00
|
|
|
TimelineRenderState *TimelineRenderer::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;
|
|
|
|
|
|
|
|
|
|
qint64 newStart = m_zoomer->traceStart();
|
|
|
|
|
qint64 newEnd = m_zoomer->traceEnd();
|
|
|
|
|
qint64 start;
|
|
|
|
|
qint64 end;
|
|
|
|
|
do {
|
|
|
|
|
level = newLevel;
|
|
|
|
|
offset = newOffset;
|
|
|
|
|
start = newStart;
|
|
|
|
|
end = newEnd;
|
|
|
|
|
|
|
|
|
|
newLevel = level + 1;
|
|
|
|
|
qint64 range = m_zoomer->traceDuration() >> newLevel;
|
|
|
|
|
newOffset = (m_zoomer->windowStart() - m_zoomer->traceStart() + range / 2) / range;
|
|
|
|
|
newStart = m_zoomer->traceStart() + newOffset * range - range / 2;
|
|
|
|
|
newEnd = newStart + range;
|
|
|
|
|
} while (newStart < m_zoomer->windowStart() && newEnd > m_zoomer->windowEnd());
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
if (m_renderStates.length() <= level)
|
|
|
|
|
m_renderStates.resize(level + 1);
|
|
|
|
|
if (m_renderStates[level].length() <= offset)
|
|
|
|
|
m_renderStates[level].resize(offset + 1);
|
|
|
|
|
TimelineRenderState *state = m_renderStates[level][offset];
|
|
|
|
|
if (state == 0) {
|
|
|
|
|
state = new TimelineRenderState(start, end, 1.0 / static_cast<qreal>(SafeFloatMax),
|
|
|
|
|
m_renderPasses.size());
|
|
|
|
|
m_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-11-18 18:32:43 +01:00
|
|
|
QSGNode *TimelineRenderer::updatePaintNode(QSGNode *node,
|
|
|
|
|
UpdatePaintNodeData *updatePaintNodeData)
|
2014-10-30 14:57:49 +01:00
|
|
|
{
|
2014-11-18 18:32:43 +01:00
|
|
|
Q_UNUSED(updatePaintNodeData)
|
2014-10-30 14:57:49 +01:00
|
|
|
|
2014-11-18 18:32:43 +01:00
|
|
|
if (!m_model || m_model->hidden() || m_model->isEmpty() || m_zoomer->windowDuration() <= 0) {
|
|
|
|
|
delete node;
|
|
|
|
|
return 0;
|
|
|
|
|
} else if (node == 0) {
|
|
|
|
|
node = new QSGTransformNode;
|
|
|
|
|
}
|
2011-12-14 16:44:33 +01:00
|
|
|
|
2014-11-18 18:32:43 +01:00
|
|
|
qreal spacing = width() / m_zoomer->windowDuration();
|
2011-10-26 11:32:01 +02:00
|
|
|
|
2014-11-18 18:32:43 +01:00
|
|
|
if (m_modelDirty || m_rowCountsDirty) {
|
|
|
|
|
node->removeAllChildNodes();
|
|
|
|
|
foreach (QVector<TimelineRenderState *> stateVector, m_renderStates)
|
|
|
|
|
qDeleteAll(stateVector);
|
|
|
|
|
m_renderStates.clear();
|
|
|
|
|
m_lastState = 0;
|
|
|
|
|
}
|
2014-06-18 11:20:41 +02:00
|
|
|
|
2014-11-18 18:32:43 +01:00
|
|
|
TimelineRenderState *state = findRenderState();
|
2013-08-08 13:28:08 +02:00
|
|
|
|
2014-11-18 18:32:43 +01:00
|
|
|
int lastIndex = m_model->lastIndex(m_zoomer->windowEnd());
|
|
|
|
|
int firstIndex = m_model->firstIndex(m_zoomer->windowStart());
|
2014-03-28 15:58:27 +01:00
|
|
|
|
2014-11-18 18:32:43 +01:00
|
|
|
for (int i = 0; i < m_renderPasses.length(); ++i)
|
|
|
|
|
state->setPassState(i, m_renderPasses[i]->update(this, state, state->passState(i),
|
|
|
|
|
firstIndex, lastIndex + 1,
|
|
|
|
|
state != m_lastState, spacing));
|
2011-07-26 13:56:14 +02:00
|
|
|
|
2014-11-18 18:32:43 +01:00
|
|
|
if (state->isEmpty()) { // new state
|
|
|
|
|
for (int pass = 0; pass < m_renderPasses.length(); ++pass) {
|
|
|
|
|
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;
|
|
|
|
|
for (int i = 0; i < m_model->expandedRowCount(); ++i) {
|
|
|
|
|
QSGTransformNode *rowNode = new QSGTransformNode;
|
|
|
|
|
for (int pass = 0; pass < m_renderPasses.length(); ++pass) {
|
|
|
|
|
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-11-18 18:32:43 +01:00
|
|
|
for (int row = 0; row < m_model->collapsedRowCount(); ++row) {
|
|
|
|
|
QSGTransformNode *rowNode = new QSGTransformNode;
|
|
|
|
|
QMatrix4x4 matrix;
|
|
|
|
|
matrix.translate(0, row * TimelineModel::defaultRowHeight(), 0);
|
|
|
|
|
rowNode->setMatrix(matrix);
|
|
|
|
|
for (int pass = 0; pass < m_renderPasses.length(); ++pass) {
|
|
|
|
|
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-11-18 18:32:43 +01:00
|
|
|
if (m_rowHeightsDirty || state != m_lastState) {
|
|
|
|
|
int row = 0;
|
|
|
|
|
qreal offset = 0;
|
|
|
|
|
for (QSGNode *rowNode = state->expandedRowRoot()->firstChild(); rowNode != 0;
|
|
|
|
|
rowNode = rowNode->nextSibling()) {
|
|
|
|
|
qreal rowHeight = m_model->expandedRowHeight(row++);
|
|
|
|
|
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-11-18 18:32:43 +01:00
|
|
|
m_modelDirty = false;
|
|
|
|
|
m_notesDirty = false;
|
|
|
|
|
m_rowCountsDirty = false;
|
|
|
|
|
m_rowHeightsDirty = false;
|
|
|
|
|
m_lastState = state;
|
2013-11-05 17:16:43 +01:00
|
|
|
|
2014-11-18 18:32:43 +01:00
|
|
|
QSGNode *rowNode = m_model->expanded() ? state->expandedRowRoot() : state->collapsedRowRoot();
|
|
|
|
|
QSGNode *overlayNode = m_model->expanded() ? state->expandedOverlayRoot() :
|
|
|
|
|
state->collapsedOverlayRoot();
|
2012-02-06 16:08:56 +01:00
|
|
|
|
2014-11-18 18:32:43 +01:00
|
|
|
QMatrix4x4 matrix;
|
|
|
|
|
matrix.translate((state->start() - m_zoomer->windowStart()) * spacing, 0, 0);
|
|
|
|
|
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-06-18 11:20:41 +02:00
|
|
|
int TimelineRenderer::rowFromPosition(int y)
|
|
|
|
|
{
|
2014-12-04 15:06:17 +01:00
|
|
|
if (!m_model->expanded())
|
|
|
|
|
return y / TimelineModel::defaultRowHeight();
|
2014-06-18 11:20:41 +02:00
|
|
|
|
2014-12-04 15:06:17 +01:00
|
|
|
int ret = 0;
|
|
|
|
|
for (int row = 0; row < m_model->expandedRowCount(); ++row) {
|
|
|
|
|
y -= m_model->expandedRowHeight(row);
|
|
|
|
|
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
|
|
|
{
|
2011-10-26 11:32:01 +02:00
|
|
|
Q_UNUSED(event);
|
2014-10-30 14:57:49 +01:00
|
|
|
if (!m_model->isEmpty())
|
2014-07-09 17:43:10 +02:00
|
|
|
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
|
|
|
|
2011-10-26 11:32:01 +02:00
|
|
|
|
2013-09-16 14:33:07 +02:00
|
|
|
void TimelineRenderer::hoverMoveEvent(QHoverEvent *event)
|
2011-10-26 11:32:01 +02:00
|
|
|
{
|
|
|
|
|
Q_UNUSED(event);
|
|
|
|
|
manageHovered(event->pos().x(), event->pos().y());
|
|
|
|
|
if (m_currentSelection.eventIndex == -1)
|
|
|
|
|
event->setAccepted(false);
|
2011-07-26 13:56:14 +02:00
|
|
|
}
|
|
|
|
|
|
2012-02-24 10:47:17 +01:00
|
|
|
void TimelineRenderer::manageClicked()
|
2011-07-26 13:56:14 +02:00
|
|
|
{
|
2011-10-26 11:32:01 +02:00
|
|
|
if (m_currentSelection.eventIndex != -1) {
|
2014-10-30 14:57:49 +01:00
|
|
|
if (m_currentSelection.eventIndex == m_selectedItem)
|
2011-10-26 11:32:01 +02:00
|
|
|
setSelectionLocked(!m_selectionLocked);
|
|
|
|
|
else
|
|
|
|
|
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-10-30 14:57:49 +01:00
|
|
|
setSelectedItem(m_currentSelection.eventIndex);
|
|
|
|
|
emit itemPressed(m_currentSelection.eventIndex);
|
2011-10-26 11:32:01 +02:00
|
|
|
} else {
|
2013-08-08 13:28:08 +02:00
|
|
|
setSelectionLocked(false);
|
2014-10-30 14:57:49 +01:00
|
|
|
setSelectedItem(-1);
|
|
|
|
|
emit 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
|
|
|
|
2013-11-11 17:50:46 +01:00
|
|
|
void TimelineRenderer::manageHovered(int mouseX, int mouseY)
|
2011-10-26 11:32:01 +02:00
|
|
|
{
|
2014-11-18 18:32:43 +01:00
|
|
|
qint64 duration = m_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-11-18 18:32:43 +01:00
|
|
|
qint64 startTime = (mouseX - 1) * duration / width() + m_zoomer->windowStart();
|
|
|
|
|
qint64 endTime = (mouseX + 1) * duration / width() + m_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-10-30 14:57:49 +01:00
|
|
|
// already covered? Only recheck selectionLocked and make sure m_selectedItem is correct.
|
2012-02-24 10:47:17 +01:00
|
|
|
if (m_currentSelection.eventIndex != -1 &&
|
2014-12-04 13:24:49 +01:00
|
|
|
exactTime >= m_currentSelection.startTime &&
|
|
|
|
|
exactTime < m_currentSelection.endTime &&
|
2012-02-24 10:47:17 +01:00
|
|
|
row == m_currentSelection.row) {
|
2014-10-30 14:57:49 +01:00
|
|
|
if (!m_selectionLocked)
|
|
|
|
|
setSelectedItem(m_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-10-30 14:57:49 +01:00
|
|
|
int eventFrom = m_model->firstIndex(startTime);
|
|
|
|
|
int eventTo = m_model->lastIndex(endTime);
|
2014-12-04 13:24:49 +01:00
|
|
|
|
|
|
|
|
m_currentSelection.eventIndex = -1;
|
|
|
|
|
if (eventFrom == -1 || eventTo < eventFrom || eventTo >= m_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-10-30 14:57:49 +01:00
|
|
|
if ( m_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-10-30 14:57:49 +01:00
|
|
|
qint64 itemEnd = m_model->endTime(i);
|
2014-03-26 17:06:26 +01:00
|
|
|
if (itemEnd < startTime)
|
|
|
|
|
continue;
|
|
|
|
|
|
2014-12-04 13:24:49 +01:00
|
|
|
qint64 itemStart = m_model->startTime(i);
|
|
|
|
|
|
|
|
|
|
qint64 offset = qAbs(itemEnd - exactTime) + qAbs(itemStart - exactTime);
|
|
|
|
|
if (offset < bestOffset) {
|
|
|
|
|
// match
|
|
|
|
|
m_currentSelection.eventIndex = i;
|
|
|
|
|
m_currentSelection.startTime = itemStart;
|
|
|
|
|
m_currentSelection.endTime = itemEnd;
|
|
|
|
|
m_currentSelection.row = row;
|
|
|
|
|
bestOffset = offset;
|
|
|
|
|
}
|
2011-10-26 11:32:01 +02:00
|
|
|
}
|
2011-07-26 13:56:14 +02:00
|
|
|
}
|
2014-12-04 13:24:49 +01:00
|
|
|
if (!m_selectionLocked && m_currentSelection.eventIndex != -1)
|
|
|
|
|
setSelectedItem(m_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-07-09 17:43:10 +02:00
|
|
|
resetCurrentSelection();
|
|
|
|
|
setSelectedItem(-1);
|
|
|
|
|
setSelectionLocked(true);
|
2011-07-26 13:56:14 +02:00
|
|
|
}
|
|
|
|
|
|
2014-10-30 14:57:49 +01:00
|
|
|
void TimelineRenderer::selectNextFromSelectionId(int selectionId)
|
2014-03-26 11:47:52 +01:00
|
|
|
{
|
2014-10-30 14:57:49 +01:00
|
|
|
setSelectedItem(m_model->nextItemBySelectionId(selectionId, m_zoomer->rangeStart(),
|
|
|
|
|
m_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-10-30 14:57:49 +01:00
|
|
|
setSelectedItem(m_model->prevItemBySelectionId(selectionId, m_zoomer->rangeStart(),
|
|
|
|
|
m_selectedItem));
|
2011-10-26 11:32:01 +02:00
|
|
|
}
|
2014-11-18 18:32:43 +01:00
|
|
|
|
|
|
|
|
void TimelineRenderer::setModelDirty()
|
|
|
|
|
{
|
|
|
|
|
m_modelDirty = true;
|
|
|
|
|
update();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void TimelineRenderer::setRowHeightsDirty()
|
|
|
|
|
{
|
|
|
|
|
m_rowHeightsDirty = true;
|
|
|
|
|
update();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void TimelineRenderer::setNotesDirty()
|
|
|
|
|
{
|
|
|
|
|
m_notesDirty = true;
|
|
|
|
|
update();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void TimelineRenderer::setRowCountsDirty()
|
|
|
|
|
{
|
|
|
|
|
m_rowCountsDirty = true;
|
|
|
|
|
}
|
|
|
|
|
|