mirror of
https://github.com/baldurk/renderdoc.git
synced 2026-05-04 17:10:47 +00:00
0ad5709df3
* This goes all the way back to the first iterations where these were the only structures and 'Fetch' referred to them returning data from the core code to the UI.
757 lines
19 KiB
C++
757 lines
19 KiB
C++
/******************************************************************************
|
|
* The MIT License (MIT)
|
|
*
|
|
* Copyright (c) 2016-2017 Baldur Karlsson
|
|
*
|
|
* Permission is hereby granted, free of charge, to any person obtaining a copy
|
|
* of this software and associated documentation files (the "Software"), to deal
|
|
* in the Software without restriction, including without limitation the rights
|
|
* to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
|
|
* copies of the Software, and to permit persons to whom the Software is
|
|
* furnished to do so, subject to the following conditions:
|
|
*
|
|
* The above copyright notice and this permission notice shall be included in
|
|
* all copies or substantial portions of the Software.
|
|
*
|
|
* THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
|
* IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
|
* FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
|
|
* AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
|
|
* LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
|
|
* OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
|
|
* THE SOFTWARE.
|
|
******************************************************************************/
|
|
|
|
#include "EventBrowser.h"
|
|
#include <QKeyEvent>
|
|
#include <QShortcut>
|
|
#include <QTimer>
|
|
#include "3rdparty/flowlayout/FlowLayout.h"
|
|
#include "Code/CaptureContext.h"
|
|
#include "Code/QRDUtils.h"
|
|
#include "ui_EventBrowser.h"
|
|
|
|
enum
|
|
{
|
|
COL_NAME = 0,
|
|
COL_EID = 1,
|
|
COL_DURATION = 2,
|
|
|
|
COL_CURRENT,
|
|
COL_FIND,
|
|
COL_BOOKMARK,
|
|
COL_LAST_EID,
|
|
};
|
|
|
|
EventBrowser::EventBrowser(CaptureContext &ctx, QWidget *parent)
|
|
: QFrame(parent), ui(new Ui::EventBrowser), m_Ctx(ctx)
|
|
{
|
|
ui->setupUi(this);
|
|
|
|
m_Ctx.AddLogViewer(this);
|
|
|
|
clearBookmarks();
|
|
|
|
ui->events->setClearSelectionOnFocusLoss(false);
|
|
|
|
ui->events->header()->resizeSection(COL_EID, 80);
|
|
|
|
ui->events->header()->setSectionResizeMode(COL_NAME, QHeaderView::Stretch);
|
|
ui->events->header()->setSectionResizeMode(COL_EID, QHeaderView::Interactive);
|
|
ui->events->header()->setSectionResizeMode(COL_DURATION, QHeaderView::Interactive);
|
|
|
|
// we set up the name column first, EID second, so that the name column gets the
|
|
// expand/collapse widgets. Then we need to put them back in order
|
|
ui->events->header()->moveSection(COL_NAME, COL_EID);
|
|
|
|
// Qt doesn't allow moving the column with the expand/collapse widgets, so this
|
|
// becomes quickly infuriating to rearrange, just disable until that can be fixed.
|
|
ui->events->header()->setSectionsMovable(false);
|
|
|
|
m_SizeDelegate = new SizeDelegate(QSize(0, 16));
|
|
ui->events->setItemDelegate(m_SizeDelegate);
|
|
|
|
m_FindHighlight = new QTimer(this);
|
|
m_FindHighlight->setInterval(400);
|
|
m_FindHighlight->setSingleShot(true);
|
|
connect(m_FindHighlight, &QTimer::timeout, this, &EventBrowser::findHighlight_timeout);
|
|
|
|
QObject::connect(ui->closeFind, &QToolButton::clicked, this, &EventBrowser::on_HideFindJump);
|
|
QObject::connect(ui->closeJump, &QToolButton::clicked, this, &EventBrowser::on_HideFindJump);
|
|
QObject::connect(ui->events, &RDTreeWidget::keyPress, this, &EventBrowser::events_keyPress);
|
|
ui->jumpStrip->hide();
|
|
ui->findStrip->hide();
|
|
ui->bookmarkStrip->hide();
|
|
|
|
m_BookmarkStripLayout = new FlowLayout(ui->bookmarkStrip, 0, 3, 3);
|
|
m_BookmarkSpacer = new QSpacerItem(40, 20, QSizePolicy::Expanding, QSizePolicy::Minimum);
|
|
|
|
ui->bookmarkStrip->setSizePolicy(QSizePolicy::Preferred, QSizePolicy::Minimum);
|
|
m_BookmarkStripLayout->addWidget(ui->bookmarkStripHeader);
|
|
m_BookmarkStripLayout->addItem(m_BookmarkSpacer);
|
|
|
|
m_CurrentIcon.addFile(QStringLiteral(":/flag_green.png"), QSize(), QIcon::Normal, QIcon::Off);
|
|
m_FindIcon.addFile(QStringLiteral(":/find.png"), QSize(), QIcon::Normal, QIcon::Off);
|
|
m_BookmarkIcon.addFile(QStringLiteral(":/asterisk_orange.png"), QSize(), QIcon::Normal, QIcon::Off);
|
|
|
|
Qt::Key keys[] = {
|
|
Qt::Key_1, Qt::Key_2, Qt::Key_3, Qt::Key_4, Qt::Key_5,
|
|
Qt::Key_6, Qt::Key_7, Qt::Key_8, Qt::Key_9, Qt::Key_0,
|
|
};
|
|
for(int i = 0; i < 10; i++)
|
|
{
|
|
QShortcut *sc = new QShortcut(QKeySequence(keys[i] | Qt::ControlModifier), this);
|
|
QObject::connect(sc, &QShortcut::activated, [this, i]() { jumpToBookmark(i); });
|
|
}
|
|
|
|
{
|
|
QShortcut *sc = new QShortcut(QKeySequence(Qt::Key_Left | Qt::ControlModifier), this);
|
|
QObject::connect(sc, &QShortcut::activated, this, &EventBrowser::on_stepPrev_clicked);
|
|
}
|
|
|
|
{
|
|
QShortcut *sc = new QShortcut(QKeySequence(Qt::Key_Right | Qt::ControlModifier), this);
|
|
QObject::connect(sc, &QShortcut::activated, this, &EventBrowser::on_stepNext_clicked);
|
|
}
|
|
}
|
|
|
|
EventBrowser::~EventBrowser()
|
|
{
|
|
m_Ctx.windowClosed(this);
|
|
m_Ctx.RemoveLogViewer(this);
|
|
delete ui;
|
|
delete m_SizeDelegate;
|
|
}
|
|
|
|
void EventBrowser::OnLogfileLoaded()
|
|
{
|
|
QTreeWidgetItem *frame = new QTreeWidgetItem(
|
|
(QTreeWidget *)NULL,
|
|
QStringList{QString("Frame #%1").arg(m_Ctx.FrameInfo().frameNumber), "", ""});
|
|
|
|
clearBookmarks();
|
|
|
|
QTreeWidgetItem *framestart = new QTreeWidgetItem(frame, QStringList{"Frame Start", "0", ""});
|
|
framestart->setData(COL_EID, Qt::UserRole, QVariant(0));
|
|
framestart->setData(COL_CURRENT, Qt::UserRole, QVariant(false));
|
|
framestart->setData(COL_FIND, Qt::UserRole, QVariant(false));
|
|
framestart->setData(COL_BOOKMARK, Qt::UserRole, QVariant(false));
|
|
framestart->setData(COL_LAST_EID, Qt::UserRole, QVariant(0));
|
|
|
|
uint lastEID = AddDrawcalls(frame, m_Ctx.CurDrawcalls());
|
|
frame->setData(COL_EID, Qt::UserRole, QVariant(0));
|
|
frame->setData(COL_LAST_EID, Qt::UserRole, QVariant(lastEID));
|
|
|
|
ui->events->insertTopLevelItem(0, frame);
|
|
|
|
ui->events->expandItem(frame);
|
|
|
|
m_Ctx.SetEventID({this}, lastEID, lastEID);
|
|
}
|
|
|
|
void EventBrowser::OnLogfileClosed()
|
|
{
|
|
clearBookmarks();
|
|
|
|
ui->events->clear();
|
|
}
|
|
|
|
void EventBrowser::OnEventChanged(uint32_t eventID)
|
|
{
|
|
SelectEvent(eventID);
|
|
highlightBookmarks();
|
|
}
|
|
|
|
uint EventBrowser::AddDrawcalls(QTreeWidgetItem *parent,
|
|
const rdctype::array<DrawcallDescription> &draws)
|
|
{
|
|
uint lastEID = 0;
|
|
|
|
for(int32_t i = 0; i < draws.count; i++)
|
|
{
|
|
QTreeWidgetItem *child = new QTreeWidgetItem(
|
|
parent, QStringList{QString(draws[i].name), QString("%1").arg(draws[i].eventID), "0.0"});
|
|
|
|
lastEID = AddDrawcalls(child, draws[i].children);
|
|
|
|
if(lastEID > draws[i].eventID)
|
|
child->setText(COL_EID, QString("%1-%2").arg(draws[i].eventID).arg(lastEID));
|
|
|
|
if(lastEID == 0)
|
|
{
|
|
lastEID = draws[i].eventID;
|
|
|
|
if((draws[i].flags & DrawFlags::SetMarker) && i + 1 < draws.count)
|
|
lastEID = draws[i + 1].eventID;
|
|
}
|
|
|
|
child->setData(COL_EID, Qt::UserRole, QVariant(draws[i].eventID));
|
|
child->setData(COL_CURRENT, Qt::UserRole, QVariant(false));
|
|
child->setData(COL_FIND, Qt::UserRole, QVariant(false));
|
|
child->setData(COL_BOOKMARK, Qt::UserRole, QVariant(false));
|
|
child->setData(COL_LAST_EID, Qt::UserRole, QVariant(lastEID));
|
|
}
|
|
|
|
return lastEID;
|
|
}
|
|
|
|
void EventBrowser::SetDrawcallTimes(QTreeWidgetItem *node,
|
|
const rdctype::array<CounterResult> &results)
|
|
{
|
|
if(node == NULL)
|
|
return;
|
|
|
|
// parent nodes take the value of the sum of their children
|
|
double duration = 0.0;
|
|
|
|
// look up leaf nodes in the dictionary
|
|
if(node->childCount() == 0)
|
|
{
|
|
uint eid = node->data(COL_EID, Qt::UserRole).toUInt();
|
|
|
|
duration = -1.0;
|
|
|
|
for(const CounterResult &r : results)
|
|
{
|
|
if(r.eventID == eid)
|
|
duration = r.value.d;
|
|
}
|
|
|
|
node->setText(COL_DURATION, duration < 0.0f ? "" : QString::number(duration * 1000000.0));
|
|
node->setData(COL_DURATION, Qt::UserRole, QVariant(duration));
|
|
|
|
return;
|
|
}
|
|
|
|
for(int i = 0; i < node->childCount(); i++)
|
|
{
|
|
SetDrawcallTimes(node->child(i), results);
|
|
|
|
double nd = node->child(i)->data(COL_DURATION, Qt::UserRole).toDouble();
|
|
|
|
if(nd > 0.0)
|
|
duration += nd;
|
|
}
|
|
|
|
node->setText(COL_DURATION, duration < 0.0f ? "" : QString::number(duration * 1000000.0));
|
|
node->setData(COL_DURATION, Qt::UserRole, QVariant(duration));
|
|
}
|
|
|
|
void EventBrowser::on_find_clicked()
|
|
{
|
|
ui->jumpStrip->hide();
|
|
ui->findStrip->show();
|
|
ui->findEvent->setFocus();
|
|
}
|
|
|
|
void EventBrowser::on_gotoEID_clicked()
|
|
{
|
|
ui->jumpStrip->show();
|
|
ui->findStrip->hide();
|
|
ui->jumpToEID->setFocus();
|
|
}
|
|
|
|
void EventBrowser::on_bookmark_clicked()
|
|
{
|
|
QTreeWidgetItem *n = ui->events->currentItem();
|
|
|
|
if(n)
|
|
toggleBookmark(n->data(COL_LAST_EID, Qt::UserRole).toUInt());
|
|
}
|
|
|
|
void EventBrowser::on_timeDraws_clicked()
|
|
{
|
|
m_Ctx.Renderer().AsyncInvoke([this](IReplayRenderer *r) {
|
|
|
|
GPUCounter counters[] = {GPUCounter::EventGPUDuration};
|
|
|
|
rdctype::array<CounterResult> results;
|
|
r->FetchCounters(counters, 1, &results);
|
|
|
|
GUIInvoke::call([this, results]() { SetDrawcallTimes(ui->events->topLevelItem(0), results); });
|
|
});
|
|
}
|
|
|
|
void EventBrowser::on_events_currentItemChanged(QTreeWidgetItem *current, QTreeWidgetItem *previous)
|
|
{
|
|
if(previous)
|
|
{
|
|
previous->setData(COL_CURRENT, Qt::UserRole, QVariant(false));
|
|
RefreshIcon(previous);
|
|
}
|
|
|
|
if(!current)
|
|
return;
|
|
|
|
current->setData(COL_CURRENT, Qt::UserRole, QVariant(true));
|
|
RefreshIcon(current);
|
|
|
|
uint EID = current->data(COL_EID, Qt::UserRole).toUInt();
|
|
uint lastEID = current->data(COL_LAST_EID, Qt::UserRole).toUInt();
|
|
|
|
m_Ctx.SetEventID({this}, EID, lastEID);
|
|
|
|
highlightBookmarks();
|
|
}
|
|
|
|
void EventBrowser::on_HideFindJump()
|
|
{
|
|
ui->jumpStrip->hide();
|
|
ui->findStrip->hide();
|
|
|
|
ui->jumpToEID->setText("");
|
|
|
|
ClearFindIcons();
|
|
ui->findEvent->setStyleSheet("");
|
|
}
|
|
|
|
void EventBrowser::on_jumpToEID_returnPressed()
|
|
{
|
|
bool ok = false;
|
|
uint eid = ui->jumpToEID->text().toUInt(&ok);
|
|
if(ok)
|
|
{
|
|
SelectEvent(eid);
|
|
}
|
|
}
|
|
|
|
void EventBrowser::findHighlight_timeout()
|
|
{
|
|
ClearFindIcons();
|
|
|
|
int results = SetFindIcons(ui->findEvent->text());
|
|
|
|
if(results > 0)
|
|
ui->findEvent->setStyleSheet("");
|
|
else
|
|
ui->findEvent->setStyleSheet("QLineEdit{background-color:#ff0000;}");
|
|
}
|
|
|
|
void EventBrowser::on_findEvent_textEdited(const QString &arg1)
|
|
{
|
|
if(arg1.isEmpty())
|
|
{
|
|
m_FindHighlight->stop();
|
|
|
|
ui->findEvent->setStyleSheet("");
|
|
ClearFindIcons();
|
|
}
|
|
else
|
|
{
|
|
m_FindHighlight->start(); // restart
|
|
}
|
|
}
|
|
|
|
void EventBrowser::on_findEvent_returnPressed()
|
|
{
|
|
// stop the timer, we'll manually fire it instantly
|
|
if(m_FindHighlight->isActive())
|
|
m_FindHighlight->stop();
|
|
|
|
if(!ui->findEvent->text().isEmpty())
|
|
Find(true);
|
|
|
|
findHighlight_timeout();
|
|
}
|
|
|
|
void EventBrowser::on_findEvent_keyPress(QKeyEvent *event)
|
|
{
|
|
if(event->key() == Qt::Key_F3)
|
|
{
|
|
// stop the timer, we'll manually fire it instantly
|
|
if(m_FindHighlight->isActive())
|
|
m_FindHighlight->stop();
|
|
|
|
if(!ui->findEvent->text().isEmpty())
|
|
Find(event->modifiers() & Qt::ShiftModifier ? false : true);
|
|
|
|
findHighlight_timeout();
|
|
|
|
event->accept();
|
|
}
|
|
}
|
|
|
|
void EventBrowser::on_findNext_clicked()
|
|
{
|
|
Find(true);
|
|
}
|
|
|
|
void EventBrowser::on_findPrev_clicked()
|
|
{
|
|
Find(false);
|
|
}
|
|
|
|
void EventBrowser::on_stepNext_clicked()
|
|
{
|
|
if(!m_Ctx.LogLoaded())
|
|
return;
|
|
|
|
const DrawcallDescription *draw = m_Ctx.CurDrawcall();
|
|
|
|
if(draw && draw->next > 0)
|
|
SelectEvent(draw->next);
|
|
}
|
|
|
|
void EventBrowser::on_stepPrev_clicked()
|
|
{
|
|
if(!m_Ctx.LogLoaded())
|
|
return;
|
|
|
|
const DrawcallDescription *draw = m_Ctx.CurDrawcall();
|
|
|
|
if(draw && draw->previous > 0)
|
|
SelectEvent(draw->previous);
|
|
}
|
|
|
|
void EventBrowser::events_keyPress(QKeyEvent *event)
|
|
{
|
|
if(!m_Ctx.LogLoaded())
|
|
return;
|
|
|
|
if(event->key() == Qt::Key_F3)
|
|
{
|
|
if(event->modifiers() == Qt::ShiftModifier)
|
|
Find(false);
|
|
else
|
|
Find(true);
|
|
}
|
|
|
|
if(event->modifiers() == Qt::ControlModifier)
|
|
{
|
|
if(event->key() == Qt::Key_F)
|
|
{
|
|
on_find_clicked();
|
|
event->accept();
|
|
}
|
|
else if(event->key() == Qt::Key_G)
|
|
{
|
|
on_gotoEID_clicked();
|
|
event->accept();
|
|
}
|
|
else if(event->key() == Qt::Key_B)
|
|
{
|
|
on_bookmark_clicked();
|
|
event->accept();
|
|
}
|
|
else if(event->key() == Qt::Key_T)
|
|
{
|
|
on_timeDraws_clicked();
|
|
event->accept();
|
|
}
|
|
}
|
|
}
|
|
|
|
void EventBrowser::clearBookmarks()
|
|
{
|
|
for(QToolButton *b : m_BookmarkButtons)
|
|
delete b;
|
|
|
|
m_Bookmarks.clear();
|
|
m_BookmarkButtons.clear();
|
|
|
|
ui->bookmarkStrip->setVisible(false);
|
|
}
|
|
|
|
void EventBrowser::toggleBookmark(uint32_t EID)
|
|
{
|
|
int index = m_Bookmarks.indexOf(EID);
|
|
|
|
QTreeWidgetItem *found = NULL;
|
|
FindEventNode(found, ui->events->topLevelItem(0), EID);
|
|
|
|
if(index >= 0)
|
|
{
|
|
delete m_BookmarkButtons.takeAt(index);
|
|
m_Bookmarks.removeAt(index);
|
|
|
|
if(found)
|
|
{
|
|
found->setData(COL_BOOKMARK, Qt::UserRole, QVariant(false));
|
|
RefreshIcon(found);
|
|
}
|
|
}
|
|
else
|
|
{
|
|
QToolButton *but = new QToolButton(this);
|
|
|
|
but->setText(QString::number(EID));
|
|
but->setCheckable(true);
|
|
but->setAutoRaise(true);
|
|
but->setProperty("eid", EID);
|
|
QObject::connect(but, &QToolButton::clicked, [this, but, EID]() {
|
|
but->setChecked(true);
|
|
SelectEvent(EID);
|
|
highlightBookmarks();
|
|
});
|
|
|
|
m_BookmarkButtons.push_back(but);
|
|
m_Bookmarks.push_back(EID);
|
|
|
|
highlightBookmarks();
|
|
|
|
if(found)
|
|
{
|
|
found->setData(COL_BOOKMARK, Qt::UserRole, QVariant(true));
|
|
RefreshIcon(found);
|
|
}
|
|
|
|
m_BookmarkStripLayout->removeItem(m_BookmarkSpacer);
|
|
m_BookmarkStripLayout->addWidget(but);
|
|
m_BookmarkStripLayout->addItem(m_BookmarkSpacer);
|
|
}
|
|
|
|
ui->bookmarkStrip->setVisible(!m_BookmarkButtons.isEmpty());
|
|
}
|
|
|
|
void EventBrowser::jumpToBookmark(int idx)
|
|
{
|
|
if(idx < 0 || idx >= m_Bookmarks.count() || !m_Ctx.LogLoaded())
|
|
return;
|
|
|
|
// don't exclude ourselves, so we're updated as normal
|
|
SelectEvent(m_Bookmarks[idx]);
|
|
}
|
|
|
|
void EventBrowser::highlightBookmarks()
|
|
{
|
|
for(QToolButton *b : m_BookmarkButtons)
|
|
{
|
|
if(b->property("eid").toUInt() == m_Ctx.CurEvent())
|
|
b->setChecked(true);
|
|
else
|
|
b->setChecked(false);
|
|
}
|
|
}
|
|
|
|
bool EventBrowser::hasBookmark(QTreeWidgetItem *node)
|
|
{
|
|
if(node)
|
|
return hasBookmark(node->data(COL_EID, Qt::UserRole).toUInt());
|
|
|
|
return false;
|
|
}
|
|
|
|
bool EventBrowser::hasBookmark(uint32_t EID)
|
|
{
|
|
return m_Bookmarks.contains(EID);
|
|
}
|
|
|
|
void EventBrowser::RefreshIcon(QTreeWidgetItem *item)
|
|
{
|
|
if(item->data(COL_CURRENT, Qt::UserRole).toBool())
|
|
item->setIcon(COL_NAME, m_CurrentIcon);
|
|
else if(item->data(COL_BOOKMARK, Qt::UserRole).toBool())
|
|
item->setIcon(COL_NAME, m_BookmarkIcon);
|
|
else if(item->data(COL_FIND, Qt::UserRole).toBool())
|
|
item->setIcon(COL_NAME, m_FindIcon);
|
|
else
|
|
item->setIcon(COL_NAME, QIcon());
|
|
}
|
|
|
|
bool EventBrowser::FindEventNode(QTreeWidgetItem *&found, QTreeWidgetItem *parent, uint32_t eventID)
|
|
{
|
|
// do a reverse search to find the last match (in case of 'set' markers that
|
|
// inherit the event of the next real draw).
|
|
for(int i = parent->childCount() - 1; i >= 0; i--)
|
|
{
|
|
QTreeWidgetItem *n = parent->child(i);
|
|
|
|
uint nEID = n->data(COL_LAST_EID, Qt::UserRole).toUInt();
|
|
uint fEID = found ? found->data(COL_LAST_EID, Qt::UserRole).toUInt() : 0;
|
|
|
|
if(nEID >= eventID && (found == NULL || nEID <= fEID))
|
|
found = n;
|
|
|
|
if(nEID == eventID && n->childCount() == 0)
|
|
return true;
|
|
|
|
if(n->childCount() > 0)
|
|
{
|
|
bool exact = FindEventNode(found, n, eventID);
|
|
if(exact)
|
|
return true;
|
|
}
|
|
}
|
|
|
|
return false;
|
|
}
|
|
|
|
void EventBrowser::ExpandNode(QTreeWidgetItem *node)
|
|
{
|
|
QTreeWidgetItem *n = node;
|
|
while(node != NULL)
|
|
{
|
|
node->setExpanded(true);
|
|
node = node->parent();
|
|
}
|
|
|
|
if(n)
|
|
ui->events->scrollToItem(n);
|
|
}
|
|
|
|
bool EventBrowser::SelectEvent(uint32_t eventID)
|
|
{
|
|
if(!m_Ctx.LogLoaded())
|
|
return false;
|
|
|
|
QTreeWidgetItem *found = NULL;
|
|
FindEventNode(found, ui->events->topLevelItem(0), eventID);
|
|
if(found != NULL)
|
|
{
|
|
ui->events->clearSelection();
|
|
ui->events->setItemSelected(found, true);
|
|
ui->events->setCurrentItem(found);
|
|
|
|
ExpandNode(found);
|
|
return true;
|
|
}
|
|
|
|
return false;
|
|
}
|
|
|
|
void EventBrowser::ClearFindIcons(QTreeWidgetItem *parent)
|
|
{
|
|
for(int i = 0; i < parent->childCount(); i++)
|
|
{
|
|
QTreeWidgetItem *n = parent->child(i);
|
|
|
|
n->setData(COL_FIND, Qt::UserRole, QVariant(false));
|
|
RefreshIcon(n);
|
|
|
|
if(n->childCount() > 0)
|
|
ClearFindIcons(n);
|
|
}
|
|
}
|
|
|
|
void EventBrowser::ClearFindIcons()
|
|
{
|
|
if(m_Ctx.LogLoaded())
|
|
ClearFindIcons(ui->events->topLevelItem(0));
|
|
}
|
|
|
|
int EventBrowser::SetFindIcons(QTreeWidgetItem *parent, QString filter)
|
|
{
|
|
int results = 0;
|
|
|
|
for(int i = 0; i < parent->childCount(); i++)
|
|
{
|
|
QTreeWidgetItem *n = parent->child(i);
|
|
|
|
if(n->text(COL_NAME).contains(filter, Qt::CaseInsensitive))
|
|
{
|
|
n->setData(COL_FIND, Qt::UserRole, QVariant(true));
|
|
RefreshIcon(n);
|
|
results++;
|
|
}
|
|
|
|
if(n->childCount() > 0)
|
|
{
|
|
results += SetFindIcons(n, filter);
|
|
}
|
|
}
|
|
|
|
return results;
|
|
}
|
|
|
|
int EventBrowser::SetFindIcons(QString filter)
|
|
{
|
|
if(filter.isEmpty())
|
|
return 0;
|
|
|
|
return SetFindIcons(ui->events->topLevelItem(0), filter);
|
|
}
|
|
|
|
QTreeWidgetItem *EventBrowser::FindNode(QTreeWidgetItem *parent, QString filter, uint32_t after)
|
|
{
|
|
for(int i = 0; i < parent->childCount(); i++)
|
|
{
|
|
QTreeWidgetItem *n = parent->child(i);
|
|
|
|
uint eid = n->data(COL_LAST_EID, Qt::UserRole).toUInt();
|
|
|
|
if(eid > after && n->text(COL_NAME).contains(filter, Qt::CaseInsensitive))
|
|
return n;
|
|
|
|
if(n->childCount() > 0)
|
|
{
|
|
QTreeWidgetItem *found = FindNode(n, filter, after);
|
|
|
|
if(found != NULL)
|
|
return found;
|
|
}
|
|
}
|
|
|
|
return NULL;
|
|
}
|
|
|
|
int EventBrowser::FindEvent(QTreeWidgetItem *parent, QString filter, uint32_t after, bool forward)
|
|
{
|
|
if(parent == NULL)
|
|
return -1;
|
|
|
|
for(int i = forward ? 0 : parent->childCount() - 1; i >= 0 && i < parent->childCount();
|
|
i += forward ? 1 : -1)
|
|
{
|
|
auto n = parent->child(i);
|
|
|
|
uint eid = n->data(COL_LAST_EID, Qt::UserRole).toUInt();
|
|
|
|
bool matchesAfter = (forward && eid > after) || (!forward && eid < after);
|
|
|
|
if(matchesAfter)
|
|
{
|
|
QString name = n->text(COL_NAME);
|
|
if(name.contains(filter, Qt::CaseInsensitive))
|
|
return (int)eid;
|
|
}
|
|
|
|
if(n->childCount() > 0)
|
|
{
|
|
int found = FindEvent(n, filter, after, forward);
|
|
|
|
if(found > 0)
|
|
return found;
|
|
}
|
|
}
|
|
|
|
return -1;
|
|
}
|
|
|
|
int EventBrowser::FindEvent(QString filter, uint32_t after, bool forward)
|
|
{
|
|
if(!m_Ctx.LogLoaded())
|
|
return 0;
|
|
|
|
return FindEvent(ui->events->topLevelItem(0), filter, after, forward);
|
|
}
|
|
|
|
void EventBrowser::Find(bool forward)
|
|
{
|
|
if(ui->findEvent->text().isEmpty())
|
|
return;
|
|
|
|
uint32_t curEID = m_Ctx.CurEvent();
|
|
if(!ui->events->selectedItems().isEmpty())
|
|
curEID = ui->events->selectedItems()[0]->data(COL_LAST_EID, Qt::UserRole).toUInt();
|
|
|
|
int eid = FindEvent(ui->findEvent->text(), curEID, forward);
|
|
if(eid >= 0)
|
|
{
|
|
SelectEvent((uint32_t)eid);
|
|
ui->findEvent->setStyleSheet("");
|
|
}
|
|
else // if(WrapSearch)
|
|
{
|
|
eid = FindEvent(ui->findEvent->text(), forward ? 0 : ~0U, forward);
|
|
if(eid >= 0)
|
|
{
|
|
SelectEvent((uint32_t)eid);
|
|
ui->findEvent->setStyleSheet("");
|
|
}
|
|
else
|
|
{
|
|
ui->findEvent->setStyleSheet("QLineEdit{background-color:#ff0000;}");
|
|
}
|
|
}
|
|
}
|