/**************************************************************************** ** ** Copyright (C) 2014 Digia Plc and/or its subsidiary(-ies). ** Contact: http://www.qt-project.org/legal ** ** This file is part of Qt Creator. ** ** 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 ** conditions see http://qt.digia.com/licensing. For further information ** use the contact form at http://qt.digia.com/contact-us. ** ** GNU Lesser General Public License Usage ** Alternatively, this file may be used under the terms of the GNU Lesser ** General Public License version 2.1 as published by the Free Software ** Foundation and appearing in the file LICENSE.LGPL included in the ** packaging of this file. Please review the following information to ** ensure the GNU Lesser General Public License version 2.1 requirements ** will be met: http://www.gnu.org/licenses/old-licenses/lgpl-2.1.html. ** ** In addition, as a special exception, Digia gives you certain additional ** rights. These rights are described in the Digia Qt LGPL Exception ** version 1.1, included in the file LGPL_EXCEPTION.txt in this package. ** ****************************************************************************/ #include "qmlprofilertraceview.h" #include "qmlprofilertool.h" #include "qmlprofilerstatemanager.h" #include "qmlprofilermodelmanager.h" #include "qmlprofilertimelinemodelproxy.h" #include "timelinemodelaggregator.h" // Needed for the load&save actions in the context menu #include // Communication with the other views (limit events to range) #include "qmlprofilerviewmanager.h" #include #include #include #include #include #include #include #include #include #include #include #include using namespace QmlDebug; namespace QmlProfiler { namespace Internal { ///////////////////////////////////////////////////////// void ZoomControl::setRange(qint64 startTime, qint64 endTime) { if (m_startTime != startTime || m_endTime != endTime) { m_startTime = startTime; m_endTime = endTime; emit rangeChanged(); } } ///////////////////////////////////////////////////////// class QmlProfilerTraceView::QmlProfilerTraceViewPrivate { public: QmlProfilerTraceViewPrivate(QmlProfilerTraceView *qq) : q(qq) {} ~QmlProfilerTraceViewPrivate() { delete m_mainView; delete m_timebar; delete m_overview; } QmlProfilerTraceView *q; QmlProfilerStateManager *m_profilerState; Analyzer::IAnalyzerTool *m_profilerTool; QmlProfilerViewManager *m_viewContainer; QSize m_sizeHint; QQuickView *m_mainView; QQuickView *m_timebar; QQuickView *m_overview; QmlProfilerModelManager *m_modelManager; TimelineModelAggregator *m_modelProxy; ZoomControl *m_zoomControl; QToolButton *m_buttonRange; QToolButton *m_buttonLock; }; QmlProfilerTraceView::QmlProfilerTraceView(QWidget *parent, Analyzer::IAnalyzerTool *profilerTool, QmlProfilerViewManager *container, QmlProfilerModelManager *modelManager, QmlProfilerStateManager *profilerState) : QWidget(parent), d(new QmlProfilerTraceViewPrivate(this)) { setObjectName(QLatin1String("QML Profiler")); d->m_zoomControl = new ZoomControl(this); connect(d->m_zoomControl, SIGNAL(rangeChanged()), this, SLOT(updateRange())); QVBoxLayout *groupLayout = new QVBoxLayout; groupLayout->setContentsMargins(0, 0, 0, 0); groupLayout->setSpacing(0); d->m_mainView = new QmlProfilerQuickView(this); d->m_mainView->setResizeMode(QQuickView::SizeRootObjectToView); QWidget *mainViewContainer = QWidget::createWindowContainer(d->m_mainView); mainViewContainer->setSizePolicy(QSizePolicy::Expanding, QSizePolicy::Expanding); QHBoxLayout *toolsLayout = new QHBoxLayout; d->m_timebar = new QmlProfilerQuickView(this); d->m_timebar->setResizeMode(QQuickView::SizeRootObjectToView); QWidget *timeBarContainer = QWidget::createWindowContainer(d->m_timebar); timeBarContainer->setSizePolicy(QSizePolicy::Expanding, QSizePolicy::Fixed); timeBarContainer->setFixedHeight(24); d->m_overview = new QmlProfilerQuickView(this); d->m_overview->setResizeMode(QQuickView::SizeRootObjectToView); QWidget *overviewContainer = QWidget::createWindowContainer(d->m_overview); overviewContainer->setSizePolicy(QSizePolicy::Expanding, QSizePolicy::Fixed); overviewContainer->setFixedHeight(50); toolsLayout->addWidget(createToolbar()); toolsLayout->addWidget(timeBarContainer); emit enableToolbar(false); groupLayout->addLayout(toolsLayout); groupLayout->addWidget(mainViewContainer); groupLayout->addWidget(overviewContainer); setLayout(groupLayout); d->m_profilerTool = profilerTool; d->m_viewContainer = container; d->m_modelManager = modelManager; d->m_modelProxy = new TimelineModelAggregator(this); d->m_modelProxy->setModelManager(modelManager); connect(d->m_modelManager, SIGNAL(stateChanged()), this, SLOT(profilerDataModelStateChanged())); d->m_mainView->rootContext()->setContextProperty(QLatin1String("qmlProfilerModelProxy"), d->m_modelProxy); d->m_overview->rootContext()->setContextProperty(QLatin1String("qmlProfilerModelProxy"), d->m_modelProxy); d->m_profilerState = profilerState; // Minimum height: 5 rows of 20 pixels + scrollbar of 50 pixels + 20 pixels margin setMinimumHeight(170); } QmlProfilerTraceView::~QmlProfilerTraceView() { delete d; } ///////////////////////////////////////////////////////// // Initialize widgets void QmlProfilerTraceView::reset() { d->m_mainView->rootContext()->setContextProperty(QLatin1String("zoomControl"), d->m_zoomControl); d->m_timebar->rootContext()->setContextProperty(QLatin1String("zoomControl"), d->m_zoomControl); d->m_overview->rootContext()->setContextProperty(QLatin1String("zoomControl"), d->m_zoomControl); d->m_timebar->setSource(QUrl(QLatin1String("qrc:/qmlprofiler/TimeDisplay.qml"))); d->m_overview->setSource(QUrl(QLatin1String("qrc:/qmlprofiler/Overview.qml"))); d->m_mainView->setSource(QUrl(QLatin1String("qrc:/qmlprofiler/MainView.qml"))); QQuickItem *rootObject = d->m_mainView->rootObject(); connect(rootObject, SIGNAL(updateCursorPosition()), this, SLOT(updateCursorPosition())); connect(rootObject, SIGNAL(updateRangeButton()), this, SLOT(updateRangeButton())); connect(rootObject, SIGNAL(updateLockButton()), this, SLOT(updateLockButton())); connect(this, SIGNAL(jumpToPrev()), rootObject, SLOT(prevEvent())); connect(this, SIGNAL(jumpToNext()), rootObject, SLOT(nextEvent())); connect(rootObject, SIGNAL(selectedEventChanged(int)), this, SIGNAL(selectedEventChanged(int))); connect(rootObject, SIGNAL(changeToolTip(QString)), this, SLOT(updateToolTip(QString))); connect(this, SIGNAL(enableToolbar(bool)), this, SLOT(setZoomSliderEnabled(bool))); connect(this, SIGNAL(showZoomSlider(bool)), this, SLOT(setZoomSliderVisible(bool))); } void QmlProfilerTraceView::setZoomSliderEnabled(bool enabled) { QQuickItem *zoomSlider = d->m_mainView->rootObject()->findChild(QLatin1String("zoomSliderToolBar")); if (zoomSlider->isEnabled() != enabled) zoomSlider->setEnabled(enabled); } void QmlProfilerTraceView::setZoomSliderVisible(bool visible) { QQuickItem *zoomSlider = d->m_mainView->rootObject()->findChild(QLatin1String("zoomSliderToolBar")); if (zoomSlider->isVisible() != visible) zoomSlider->setVisible(visible); } QWidget *QmlProfilerTraceView::createToolbar() { Utils::StyledBar *bar = new Utils::StyledBar(this); bar->setStyleSheet(QLatin1String("background: #9B9B9B")); bar->setSingleRow(true); bar->setFixedWidth(150); bar->setFixedHeight(24); QHBoxLayout *toolBarLayout = new QHBoxLayout(bar); toolBarLayout->setMargin(0); toolBarLayout->setSpacing(0); QToolButton *buttonPrev= new QToolButton; buttonPrev->setIcon(QIcon(QLatin1String(":/qmlprofiler/ico_prev.png"))); buttonPrev->setToolTip(tr("Jump to previous event.")); connect(buttonPrev, SIGNAL(clicked()), this, SIGNAL(jumpToPrev())); connect(this, SIGNAL(enableToolbar(bool)), buttonPrev, SLOT(setEnabled(bool))); QToolButton *buttonNext= new QToolButton; buttonNext->setIcon(QIcon(QLatin1String(":/qmlprofiler/ico_next.png"))); buttonNext->setToolTip(tr("Jump to next event.")); connect(buttonNext, SIGNAL(clicked()), this, SIGNAL(jumpToNext())); connect(this, SIGNAL(enableToolbar(bool)), buttonNext, SLOT(setEnabled(bool))); QToolButton *buttonZoomControls = new QToolButton; buttonZoomControls->setIcon(QIcon(QLatin1String(":/qmlprofiler/ico_zoom.png"))); buttonZoomControls->setToolTip(tr("Show zoom slider.")); buttonZoomControls->setCheckable(true); buttonZoomControls->setChecked(false); connect(buttonZoomControls, SIGNAL(toggled(bool)), this, SIGNAL(showZoomSlider(bool))); connect(this, SIGNAL(enableToolbar(bool)), buttonZoomControls, SLOT(setEnabled(bool))); d->m_buttonRange = new QToolButton; d->m_buttonRange->setIcon(QIcon(QLatin1String(":/qmlprofiler/ico_rangeselection.png"))); d->m_buttonRange->setToolTip(tr("Select range.")); d->m_buttonRange->setCheckable(true); d->m_buttonRange->setChecked(false); connect(d->m_buttonRange, SIGNAL(clicked(bool)), this, SLOT(toggleRangeMode(bool))); connect(this, SIGNAL(enableToolbar(bool)), d->m_buttonRange, SLOT(setEnabled(bool))); connect(this, SIGNAL(rangeModeChanged(bool)), d->m_buttonRange, SLOT(setChecked(bool))); d->m_buttonLock = new QToolButton; d->m_buttonLock->setIcon(QIcon(QLatin1String(":/qmlprofiler/ico_selectionmode.png"))); d->m_buttonLock->setToolTip(tr("View event information on mouseover.")); d->m_buttonLock->setCheckable(true); d->m_buttonLock->setChecked(false); connect(d->m_buttonLock, SIGNAL(clicked(bool)), this, SLOT(toggleLockMode(bool))); connect(this, SIGNAL(enableToolbar(bool)), d->m_buttonLock, SLOT(setEnabled(bool))); connect(this, SIGNAL(lockModeChanged(bool)), d->m_buttonLock, SLOT(setChecked(bool))); toolBarLayout->addWidget(buttonPrev); toolBarLayout->addWidget(buttonNext); toolBarLayout->addWidget(new Utils::StyledSeparator()); toolBarLayout->addWidget(buttonZoomControls); toolBarLayout->addWidget(new Utils::StyledSeparator()); toolBarLayout->addWidget(d->m_buttonRange); toolBarLayout->addWidget(d->m_buttonLock); return bar; } ///////////////////////////////////////////////////////// bool QmlProfilerTraceView::hasValidSelection() const { QQuickItem *rootObject = d->m_mainView->rootObject(); if (rootObject) return rootObject->property("selectionRangeReady").toBool(); return false; } qint64 QmlProfilerTraceView::selectionStart() const { QQuickItem *rootObject = d->m_mainView->rootObject(); if (rootObject) return rootObject->property("selectionRangeStart").toLongLong(); return 0; } qint64 QmlProfilerTraceView::selectionEnd() const { QQuickItem *rootObject = d->m_mainView->rootObject(); if (rootObject) return rootObject->property("selectionRangeEnd").toLongLong(); return 0; } void QmlProfilerTraceView::clearDisplay() { d->m_zoomControl->setRange(0,0); d->m_mainView->rootObject()->setProperty("scrollY", QVariant(0)); QMetaObject::invokeMethod(d->m_mainView->rootObject(), "clearAll"); QMetaObject::invokeMethod(d->m_overview->rootObject(), "clearDisplay"); } void QmlProfilerTraceView::selectNextEventByHash(const QString &hash) { QQuickItem *rootObject = d->m_mainView->rootObject(); if (rootObject) QMetaObject::invokeMethod(rootObject, "selectNextByHash", Q_ARG(QVariant,QVariant(hash))); } void QmlProfilerTraceView::selectNextEventByLocation(const QString &filename, const int line, const int column) { int eventId = d->m_modelProxy->getEventIdForLocation(filename, line, column); if (eventId != -1) { QQuickItem *rootObject = d->m_mainView->rootObject(); if (rootObject) QMetaObject::invokeMethod(rootObject, "selectNextById", Q_ARG(QVariant,QVariant(eventId))); } } ///////////////////////////////////////////////////////// // Goto source location void QmlProfilerTraceView::updateCursorPosition() { QQuickItem *rootObject = d->m_mainView->rootObject(); emit gotoSourceLocation(rootObject->property("fileName").toString(), rootObject->property("lineNumber").toInt(), rootObject->property("columnNumber").toInt()); } ///////////////////////////////////////////////////////// // Toolbar buttons void QmlProfilerTraceView::toggleRangeMode(bool active) { QQuickItem *rootObject = d->m_mainView->rootObject(); bool rangeMode = rootObject->property("selectionRangeMode").toBool(); if (active != rangeMode) { if (active) d->m_buttonRange->setIcon(QIcon(QLatin1String(":/qmlprofiler/ico_rangeselected.png"))); else d->m_buttonRange->setIcon(QIcon(QLatin1String(":/qmlprofiler/ico_rangeselection.png"))); rootObject->setProperty("selectionRangeMode", QVariant(active)); } } void QmlProfilerTraceView::updateRangeButton() { bool rangeMode = d->m_mainView->rootObject()->property("selectionRangeMode").toBool(); if (rangeMode) d->m_buttonRange->setIcon(QIcon(QLatin1String(":/qmlprofiler/ico_rangeselected.png"))); else d->m_buttonRange->setIcon(QIcon(QLatin1String(":/qmlprofiler/ico_rangeselection.png"))); emit rangeModeChanged(rangeMode); } void QmlProfilerTraceView::toggleLockMode(bool active) { QQuickItem *rootObject = d->m_mainView->rootObject(); bool lockMode = !rootObject->property("selectionLocked").toBool(); if (active != lockMode) { rootObject->setProperty("selectionLocked", QVariant(!active)); rootObject->setProperty("selectedItem", QVariant(-1)); } } void QmlProfilerTraceView::updateLockButton() { bool lockMode = !d->m_mainView->rootObject()->property("selectionLocked").toBool(); emit lockModeChanged(lockMode); } //////////////////////////////////////////////////////// // Zoom control void QmlProfilerTraceView::updateRange() { if (!d->m_modelManager) return; qreal duration = d->m_zoomControl->endTime() - d->m_zoomControl->startTime(); if (duration <= 0) return; if (d->m_modelManager->traceTime()->duration() <= 0) return; QMetaObject::invokeMethod(d->m_mainView->rootObject()->findChild(QLatin1String("zoomSliderToolBar")), "updateZoomLevel"); } //////////////////////////////////////////////////////// void QmlProfilerTraceView::updateToolTip(const QString &text) { setToolTip(text); } void QmlProfilerTraceView::resizeEvent(QResizeEvent *event) { QWidget::resizeEvent(event); emit resized(); } //////////////////////////////////////////////////////////////// // Context menu void QmlProfilerTraceView::contextMenuEvent(QContextMenuEvent *ev) { showContextMenu(ev->globalPos()); } void QmlProfilerTraceView::showContextMenu(QPoint position) { QMenu menu; QAction *viewAllAction = 0; QmlProfilerTool *profilerTool = qobject_cast(d->m_profilerTool); if (profilerTool) menu.addActions(profilerTool->profilerContextMenuActions()); menu.addSeparator(); QAction *getLocalStatsAction = menu.addAction(tr("Limit Events Pane to Current Range")); if (!d->m_viewContainer->hasValidSelection()) getLocalStatsAction->setEnabled(false); QAction *getGlobalStatsAction = menu.addAction(tr("Show Full Range in Events Pane")); if (d->m_viewContainer->hasGlobalStats()) getGlobalStatsAction->setEnabled(false); if (!d->m_modelProxy->isEmpty()) { menu.addSeparator(); viewAllAction = menu.addAction(tr("Reset Zoom")); } QAction *selectedAction = menu.exec(position); if (selectedAction) { if (selectedAction == viewAllAction) { d->m_zoomControl->setRange( d->m_modelManager->traceTime()->startTime(), d->m_modelManager->traceTime()->endTime()); } if (selectedAction == getLocalStatsAction) { d->m_viewContainer->getStatisticsInRange( d->m_viewContainer->selectionStart(), d->m_viewContainer->selectionEnd()); } if (selectedAction == getGlobalStatsAction) d->m_viewContainer->getStatisticsInRange(-1, -1); } } //////////////////////////////////////////////////////////////// // Profiler State void QmlProfilerTraceView::profilerDataModelStateChanged() { switch (d->m_modelManager->state()) { case QmlProfilerDataState::Empty: break; case QmlProfilerDataState::ClearingData: d->m_mainView->hide(); emit enableToolbar(false); break; case QmlProfilerDataState::AcquiringData: break; case QmlProfilerDataState::ProcessingData: break; case QmlProfilerDataState::Done: emit enableToolbar(true); d->m_mainView->show(); break; default: break; } } bool QmlProfilerQuickView::event(QEvent *ev) { // We assume context menus can only be triggered by mouse press, mouse release, or // pre-synthesized events from the window system. bool relayed = false; switch (ev->type()) { case QEvent::ContextMenu: // In the case of mouse clicks the active popup gets automatically closed before they show // up here. That's not necessarily the case with keyboard triggered context menu events, so // we just ignore them if there is a popup already. Also, the event's pos() and globalPos() // don't make much sense in this case, so we just put the menu in the upper left corner. if (QApplication::activePopupWidget() == 0) { ev->accept(); parent->showContextMenu(parent->mapToGlobal(QPoint(0,0))); relayed = true; } break; case QEvent::MouseButtonPress: case QEvent::MouseButtonRelease: { QMouseEvent *orig = static_cast(ev); QCoreApplication::instance()->postEvent(parent->window()->windowHandle(), new QMouseEvent(orig->type(), parent->window()->mapFromGlobal(orig->globalPos()), orig->button(), orig->buttons(), orig->modifiers())); relayed = true; break; } default: break; } // QQuickView will eat mouse events even if they're not accepted by any QML construct. So we // ignore the return value of event() above. return QQuickView::event(ev) || relayed; } } // namespace Internal } // namespace QmlProfiler