mirror of
https://github.com/rizinorg/cutter.git
synced 2024-12-19 03:16:10 +00:00
Refactor SymbolsWidget to use Model/View architecture (#473)
This commit is contained in:
parent
4287da4888
commit
fd829986af
@ -1,6 +1,5 @@
|
|||||||
#include <QMenu>
|
#include <QMenu>
|
||||||
#include <QResizeEvent>
|
#include <QResizeEvent>
|
||||||
#include <QDebug>
|
|
||||||
|
|
||||||
#include "CommentsWidget.h"
|
#include "CommentsWidget.h"
|
||||||
#include "ui_CommentsWidget.h"
|
#include "ui_CommentsWidget.h"
|
||||||
|
@ -4,8 +4,111 @@
|
|||||||
#include "MainWindow.h"
|
#include "MainWindow.h"
|
||||||
#include "utils/Helpers.h"
|
#include "utils/Helpers.h"
|
||||||
|
|
||||||
#include <QTreeWidget>
|
SymbolsModel::SymbolsModel(QList<SymbolDescription> *symbols, QObject *parent)
|
||||||
|
: QAbstractListModel(parent),
|
||||||
|
symbols(symbols)
|
||||||
|
{
|
||||||
|
}
|
||||||
|
|
||||||
|
int SymbolsModel::rowCount(const QModelIndex &) const
|
||||||
|
{
|
||||||
|
return symbols->count();
|
||||||
|
}
|
||||||
|
|
||||||
|
int SymbolsModel::columnCount(const QModelIndex &) const
|
||||||
|
{
|
||||||
|
return SymbolsModel::ColumnCount;
|
||||||
|
}
|
||||||
|
|
||||||
|
QVariant SymbolsModel::data(const QModelIndex &index, int role) const
|
||||||
|
{
|
||||||
|
if (index.row() >= symbols->count())
|
||||||
|
return QVariant();
|
||||||
|
|
||||||
|
const SymbolDescription &symbol = symbols->at(index.row());
|
||||||
|
|
||||||
|
switch (role) {
|
||||||
|
case Qt::DisplayRole:
|
||||||
|
switch (index.column()) {
|
||||||
|
case SymbolsModel::AddressColumn:
|
||||||
|
return RAddressString(symbol.vaddr);
|
||||||
|
case SymbolsModel::TypeColumn:
|
||||||
|
return QString("%1 %2").arg(symbol.bind, symbol.type).trimmed();
|
||||||
|
case SymbolsModel::NameColumn:
|
||||||
|
return symbol.name;
|
||||||
|
default:
|
||||||
|
return QVariant();
|
||||||
|
}
|
||||||
|
case SymbolsModel::SymbolDescriptionRole:
|
||||||
|
return QVariant::fromValue(symbol);
|
||||||
|
default:
|
||||||
|
return QVariant();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
QVariant SymbolsModel::headerData(int section, Qt::Orientation, int role) const
|
||||||
|
{
|
||||||
|
switch (role) {
|
||||||
|
case Qt::DisplayRole:
|
||||||
|
switch (section) {
|
||||||
|
case SymbolsModel::AddressColumn:
|
||||||
|
return tr("Address");
|
||||||
|
case SymbolsModel::TypeColumn:
|
||||||
|
return tr("Type");
|
||||||
|
case SymbolsModel::NameColumn:
|
||||||
|
return tr("Name");
|
||||||
|
default:
|
||||||
|
return QVariant();
|
||||||
|
}
|
||||||
|
default:
|
||||||
|
return QVariant();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
void SymbolsModel::beginReloadSymbols()
|
||||||
|
{
|
||||||
|
beginResetModel();
|
||||||
|
}
|
||||||
|
|
||||||
|
void SymbolsModel::endReloadSymbols()
|
||||||
|
{
|
||||||
|
endResetModel();
|
||||||
|
}
|
||||||
|
|
||||||
|
SymbolsProxyModel::SymbolsProxyModel(SymbolsModel *sourceModel, QObject *parent)
|
||||||
|
: QSortFilterProxyModel(parent)
|
||||||
|
{
|
||||||
|
setSourceModel(sourceModel);
|
||||||
|
setFilterCaseSensitivity(Qt::CaseInsensitive);
|
||||||
|
setSortCaseSensitivity(Qt::CaseInsensitive);
|
||||||
|
}
|
||||||
|
|
||||||
|
bool SymbolsProxyModel::filterAcceptsRow(int row, const QModelIndex &parent) const
|
||||||
|
{
|
||||||
|
QModelIndex index = sourceModel()->index(row, 0, parent);
|
||||||
|
auto symbol = index.data(SymbolsModel::SymbolDescriptionRole).value<SymbolDescription>();
|
||||||
|
|
||||||
|
return symbol.name.contains(filterRegExp());
|
||||||
|
}
|
||||||
|
|
||||||
|
bool SymbolsProxyModel::lessThan(const QModelIndex &left, const QModelIndex &right) const
|
||||||
|
{
|
||||||
|
auto leftSymbol = left.data(SymbolsModel::SymbolDescriptionRole).value<SymbolDescription>();
|
||||||
|
auto rightSymbol = right.data(SymbolsModel::SymbolDescriptionRole).value<SymbolDescription>();
|
||||||
|
|
||||||
|
switch (left.column()) {
|
||||||
|
case SymbolsModel::AddressColumn:
|
||||||
|
return leftSymbol.vaddr < rightSymbol.vaddr;
|
||||||
|
case SymbolsModel::TypeColumn:
|
||||||
|
return leftSymbol.type < rightSymbol.type;
|
||||||
|
case SymbolsModel::NameColumn:
|
||||||
|
return leftSymbol.name < rightSymbol.name;
|
||||||
|
default:
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
SymbolsWidget::SymbolsWidget(MainWindow *main, QAction *action) :
|
SymbolsWidget::SymbolsWidget(MainWindow *main, QAction *action) :
|
||||||
CutterDockWidget(main, action),
|
CutterDockWidget(main, action),
|
||||||
@ -13,41 +116,52 @@ SymbolsWidget::SymbolsWidget(MainWindow *main, QAction *action) :
|
|||||||
{
|
{
|
||||||
ui->setupUi(this);
|
ui->setupUi(this);
|
||||||
|
|
||||||
ui->symbolsTreeWidget->sortByColumn(2, Qt::AscendingOrder);
|
symbolsModel = new SymbolsModel(&symbols, this);
|
||||||
|
symbolsProxyModel = new SymbolsProxyModel(symbolsModel, this);
|
||||||
|
ui->symbolsTreeView->setModel(symbolsProxyModel);
|
||||||
|
ui->symbolsTreeView->sortByColumn(SymbolsModel::AddressColumn, Qt::AscendingOrder);
|
||||||
|
|
||||||
|
// Ctrl-F to show/hide the filter entry
|
||||||
|
QShortcut *searchShortcut = new QShortcut(QKeySequence::Find, this);
|
||||||
|
connect(searchShortcut, &QShortcut::activated, ui->quickFilterView, &QuickFilterView::showFilter);
|
||||||
|
searchShortcut->setContext(Qt::WidgetWithChildrenShortcut);
|
||||||
|
|
||||||
|
// Esc to clear the filter entry
|
||||||
|
QShortcut *clearShortcut = new QShortcut(QKeySequence(Qt::Key_Escape), this);
|
||||||
|
connect(clearShortcut, &QShortcut::activated, ui->quickFilterView, &QuickFilterView::clearFilter);
|
||||||
|
clearShortcut->setContext(Qt::WidgetWithChildrenShortcut);
|
||||||
|
|
||||||
|
connect(ui->quickFilterView, SIGNAL(filterTextChanged(const QString &)),
|
||||||
|
symbolsProxyModel, SLOT(setFilterWildcard(const QString &)));
|
||||||
|
connect(ui->quickFilterView, SIGNAL(filterClosed()), ui->symbolsTreeView, SLOT(setFocus()));
|
||||||
|
|
||||||
setScrollMode();
|
setScrollMode();
|
||||||
|
|
||||||
connect(Core(), SIGNAL(refreshAll()), this, SLOT(fillSymbols()));
|
connect(Core(), SIGNAL(refreshAll()), this, SLOT(refreshSymbols()));
|
||||||
}
|
}
|
||||||
|
|
||||||
SymbolsWidget::~SymbolsWidget() {}
|
SymbolsWidget::~SymbolsWidget() {}
|
||||||
|
|
||||||
|
|
||||||
void SymbolsWidget::on_symbolsTreeWidget_itemDoubleClicked(QTreeWidgetItem *item, int column)
|
void SymbolsWidget::on_symbolsTreeView_doubleClicked(const QModelIndex &index)
|
||||||
{
|
{
|
||||||
if (column < 0)
|
if (!index.isValid())
|
||||||
return;
|
return;
|
||||||
|
|
||||||
// Get offset and name of item double clicked
|
auto symbol = index.data(SymbolsModel::SymbolDescriptionRole).value<SymbolDescription>();
|
||||||
SymbolDescription symbol = item->data(0, Qt::UserRole).value<SymbolDescription>();
|
|
||||||
Core()->seek(symbol.vaddr);
|
Core()->seek(symbol.vaddr);
|
||||||
}
|
}
|
||||||
|
|
||||||
void SymbolsWidget::fillSymbols()
|
void SymbolsWidget::refreshSymbols()
|
||||||
{
|
{
|
||||||
ui->symbolsTreeWidget->clear();
|
symbolsModel->beginReloadSymbols();
|
||||||
for (auto symbol : Core()->getAllSymbols()) {
|
symbols = Core()->getAllSymbols();
|
||||||
QTreeWidgetItem *item = new QTreeWidgetItem();
|
symbolsModel->endReloadSymbols();
|
||||||
item->setText(0, RAddressString(symbol.vaddr));
|
|
||||||
item->setText(1, QString("%1 %2").arg(symbol.bind, symbol.type).trimmed());
|
qhelpers::adjustColumns(ui->symbolsTreeView, SymbolsModel::ColumnCount, 0);
|
||||||
item->setText(2, symbol.name);
|
|
||||||
item->setData(0, Qt::UserRole, QVariant::fromValue(symbol));
|
|
||||||
ui->symbolsTreeWidget->addTopLevelItem(item);
|
|
||||||
}
|
|
||||||
qhelpers::adjustColumns(ui->symbolsTreeWidget, 0);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
void SymbolsWidget::setScrollMode()
|
void SymbolsWidget::setScrollMode()
|
||||||
{
|
{
|
||||||
qhelpers::setVerticalScrollMode(ui->symbolsTreeWidget);
|
qhelpers::setVerticalScrollMode(ui->symbolsTreeView);
|
||||||
}
|
}
|
||||||
|
@ -2,7 +2,10 @@
|
|||||||
#define SYMBOLSWIDGET_H
|
#define SYMBOLSWIDGET_H
|
||||||
|
|
||||||
#include <memory>
|
#include <memory>
|
||||||
|
#include <QAbstractListModel>
|
||||||
|
#include <QSortFilterProxyModel>
|
||||||
|
|
||||||
|
#include "Cutter.h"
|
||||||
#include "CutterDockWidget.h"
|
#include "CutterDockWidget.h"
|
||||||
|
|
||||||
class MainWindow;
|
class MainWindow;
|
||||||
@ -12,6 +15,42 @@ namespace Ui {
|
|||||||
class SymbolsWidget;
|
class SymbolsWidget;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
class SymbolsModel: public QAbstractListModel
|
||||||
|
{
|
||||||
|
Q_OBJECT
|
||||||
|
|
||||||
|
private:
|
||||||
|
QList<SymbolDescription> *symbols;
|
||||||
|
|
||||||
|
public:
|
||||||
|
enum Column { AddressColumn = 0, TypeColumn, NameColumn, ColumnCount };
|
||||||
|
enum Role { SymbolDescriptionRole = Qt::UserRole };
|
||||||
|
|
||||||
|
SymbolsModel(QList<SymbolDescription> *exports, QObject *parent = Q_NULLPTR);
|
||||||
|
|
||||||
|
int rowCount(const QModelIndex &parent = QModelIndex()) const;
|
||||||
|
int columnCount(const QModelIndex &parent = QModelIndex()) const;
|
||||||
|
|
||||||
|
QVariant data(const QModelIndex &index, int role) const;
|
||||||
|
QVariant headerData(int section, Qt::Orientation orientation, int role = Qt::DisplayRole) const;
|
||||||
|
|
||||||
|
void beginReloadSymbols();
|
||||||
|
void endReloadSymbols();
|
||||||
|
};
|
||||||
|
|
||||||
|
class SymbolsProxyModel : public QSortFilterProxyModel
|
||||||
|
{
|
||||||
|
Q_OBJECT
|
||||||
|
|
||||||
|
public:
|
||||||
|
SymbolsProxyModel(SymbolsModel *sourceModel, QObject *parent = Q_NULLPTR);
|
||||||
|
|
||||||
|
protected:
|
||||||
|
bool filterAcceptsRow(int row, const QModelIndex &parent) const override;
|
||||||
|
bool lessThan(const QModelIndex &left, const QModelIndex &right) const override;
|
||||||
|
};
|
||||||
|
|
||||||
|
|
||||||
class SymbolsWidget : public CutterDockWidget
|
class SymbolsWidget : public CutterDockWidget
|
||||||
{
|
{
|
||||||
Q_OBJECT
|
Q_OBJECT
|
||||||
@ -21,12 +60,15 @@ public:
|
|||||||
~SymbolsWidget();
|
~SymbolsWidget();
|
||||||
|
|
||||||
private slots:
|
private slots:
|
||||||
void on_symbolsTreeWidget_itemDoubleClicked(QTreeWidgetItem *item, int column);
|
void on_symbolsTreeView_doubleClicked(const QModelIndex &index);
|
||||||
|
|
||||||
void fillSymbols();
|
void refreshSymbols();
|
||||||
|
|
||||||
private:
|
private:
|
||||||
std::unique_ptr<Ui::SymbolsWidget> ui;
|
std::unique_ptr<Ui::SymbolsWidget> ui;
|
||||||
|
QList<SymbolDescription> symbols;
|
||||||
|
SymbolsModel *symbolsModel;
|
||||||
|
SymbolsProxyModel *symbolsProxyModel;
|
||||||
|
|
||||||
void setScrollMode();
|
void setScrollMode();
|
||||||
};
|
};
|
||||||
|
@ -28,47 +28,29 @@
|
|||||||
<number>0</number>
|
<number>0</number>
|
||||||
</property>
|
</property>
|
||||||
<item>
|
<item>
|
||||||
<widget class="QTreeWidget" name="symbolsTreeWidget">
|
<widget class="QTreeView" name="symbolsTreeView">
|
||||||
<property name="styleSheet">
|
|
||||||
<string notr="true">QTreeWidget::item
|
|
||||||
{
|
|
||||||
padding-top: 1px;
|
|
||||||
padding-bottom: 1px;
|
|
||||||
}
|
|
||||||
</string>
|
|
||||||
</property>
|
|
||||||
<property name="frameShape">
|
|
||||||
<enum>QFrame::NoFrame</enum>
|
|
||||||
</property>
|
|
||||||
<property name="lineWidth">
|
|
||||||
<number>0</number>
|
|
||||||
</property>
|
|
||||||
<property name="indentation">
|
|
||||||
<number>8</number>
|
|
||||||
</property>
|
|
||||||
<property name="sortingEnabled">
|
<property name="sortingEnabled">
|
||||||
<bool>true</bool>
|
<bool>true</bool>
|
||||||
</property>
|
</property>
|
||||||
<column>
|
<attribute name="headerShowSortIndicator" stdset="0">
|
||||||
<property name="text">
|
<bool>true</bool>
|
||||||
<string>Address</string>
|
</attribute>
|
||||||
</property>
|
|
||||||
</column>
|
|
||||||
<column>
|
|
||||||
<property name="text">
|
|
||||||
<string>Type</string>
|
|
||||||
</property>
|
|
||||||
</column>
|
|
||||||
<column>
|
|
||||||
<property name="text">
|
|
||||||
<string>Name</string>
|
|
||||||
</property>
|
|
||||||
</column>
|
|
||||||
</widget>
|
</widget>
|
||||||
</item>
|
</item>
|
||||||
|
<item>
|
||||||
|
<widget class="QuickFilterView" name="quickFilterView" native="true"/>
|
||||||
|
</item>
|
||||||
</layout>
|
</layout>
|
||||||
</widget>
|
</widget>
|
||||||
</widget>
|
</widget>
|
||||||
|
<customwidgets>
|
||||||
|
<customwidget>
|
||||||
|
<class>QuickFilterView</class>
|
||||||
|
<extends>QWidget</extends>
|
||||||
|
<header>widgets/QuickFilterView.h</header>
|
||||||
|
<container>1</container>
|
||||||
|
</customwidget>
|
||||||
|
</customwidgets>
|
||||||
<resources/>
|
<resources/>
|
||||||
<connections/>
|
<connections/>
|
||||||
</ui>
|
</ui>
|
||||||
|
Loading…
Reference in New Issue
Block a user