cutter/src/widgets/VTablesWidget.cpp
2021-09-14 15:32:04 +02:00

204 lines
6.3 KiB
C++

#include <QShortcut>
#include <QModelIndex>
#include "core/MainWindow.h"
#include "common/Helpers.h"
#include "VTablesWidget.h"
#include "ui_VTablesWidget.h"
VTableModel::VTableModel(QList<VTableDescription> *vtables, QObject *parent)
: QAbstractItemModel(parent), vtables(vtables)
{
}
QModelIndex VTableModel::index(int row, int column, const QModelIndex &parent) const
{
return createIndex(row, column, (quintptr)parent.isValid() ? parent.row() : -1);
}
QModelIndex VTableModel::parent(const QModelIndex &index) const
{
return index.isValid() && index.internalId() != (quintptr)-1
? this->index(index.internalId(), index.column())
: QModelIndex();
}
int VTableModel::rowCount(const QModelIndex &parent) const
{
return parent.isValid()
? (parent.parent().isValid() ? 0 : vtables->at(parent.row()).methods.count())
: vtables->count();
}
int VTableModel::columnCount(const QModelIndex &) const
{
return Columns::COUNT;
}
QVariant VTableModel::data(const QModelIndex &index, int role) const
{
QModelIndex parent = index.parent();
if (parent.isValid()) {
const BinClassMethodDescription &res = vtables->at(parent.row()).methods.at(index.row());
switch (role) {
case Qt::DisplayRole:
switch (index.column()) {
case NAME:
return res.name;
case ADDRESS:
return RzAddressString(res.addr);
}
break;
case VTableDescriptionRole:
return QVariant::fromValue(res);
default:
break;
}
} else
switch (role) {
case Qt::DisplayRole:
switch (index.column()) {
case NAME:
return tr("VTable") + " " + QString::number(index.row() + 1);
case ADDRESS:
return RzAddressString(vtables->at(index.row()).addr);
}
break;
case VTableDescriptionRole: {
const VTableDescription &res = vtables->at(index.row());
return QVariant::fromValue(res);
}
default:
break;
}
return QVariant();
}
QVariant VTableModel::headerData(int section, Qt::Orientation, int role) const
{
switch (role) {
case Qt::DisplayRole:
switch (section) {
case NAME:
return tr("Name");
case ADDRESS:
return tr("Address");
default:
break;
}
break;
default:
break;
}
return QVariant();
}
VTableSortFilterProxyModel::VTableSortFilterProxyModel(VTableModel *model, QObject *parent)
: QSortFilterProxyModel(parent)
{
setSourceModel(model);
setFilterCaseSensitivity(Qt::CaseInsensitive);
setSortCaseSensitivity(Qt::CaseInsensitive);
setFilterKeyColumn(VTableModel::NAME);
#if QT_VERSION >= QT_VERSION_CHECK(5, 10, 0)
setRecursiveFilteringEnabled(true);
#endif
}
bool VTableSortFilterProxyModel::filterAcceptsRow(int source_row,
const QModelIndex &source_parent) const
{
if (QSortFilterProxyModel::filterAcceptsRow(source_row, source_parent))
return true;
if (source_parent.isValid())
return QSortFilterProxyModel::filterAcceptsRow(source_parent.row(), QModelIndex());
#if QT_VERSION < QT_VERSION_CHECK(5, 10, 0)
else {
QAbstractItemModel *const model = sourceModel();
const QModelIndex source = model->index(source_row, 0, QModelIndex());
const int rows = model->rowCount(source);
for (int i = 0; i < rows; ++i)
if (QSortFilterProxyModel::filterAcceptsRow(i, source))
return true;
}
#endif
return false;
}
VTablesWidget::VTablesWidget(MainWindow *main)
: CutterDockWidget(main), ui(new Ui::VTablesWidget), tree(new CutterTreeWidget(this))
{
ui->setupUi(this);
// Add Status Bar footer
tree->addStatusBar(ui->verticalLayout);
model = new VTableModel(&vtables, this);
proxy = new VTableSortFilterProxyModel(model, this);
ui->vTableTreeView->setModel(proxy);
ui->vTableTreeView->sortByColumn(VTableModel::ADDRESS, Qt::AscendingOrder);
// Esc to clear the filter entry
QShortcut *clear_shortcut = new QShortcut(QKeySequence(Qt::Key_Escape), this);
connect(clear_shortcut, &QShortcut::activated, ui->quickFilterView,
&QuickFilterView::clearFilter);
clear_shortcut->setContext(Qt::WidgetWithChildrenShortcut);
// Ctrl-F to show/hide the filter entry
QShortcut *search_shortcut = new QShortcut(QKeySequence::Find, this);
connect(search_shortcut, &QShortcut::activated, ui->quickFilterView,
&QuickFilterView::showFilter);
search_shortcut->setContext(Qt::WidgetWithChildrenShortcut);
connect(ui->quickFilterView, &QuickFilterView::filterTextChanged, proxy,
&QSortFilterProxyModel::setFilterWildcard);
connect(ui->quickFilterView, &QuickFilterView::filterClosed, ui->vTableTreeView,
[this]() { ui->vTableTreeView->setFocus(); });
connect(ui->quickFilterView, &QuickFilterView::filterTextChanged, this,
[this] { tree->showItemsNumber(proxy->rowCount()); });
connect(Core(), &CutterCore::codeRebased, this, &VTablesWidget::refreshVTables);
connect(Core(), &CutterCore::refreshAll, this, &VTablesWidget::refreshVTables);
refreshDeferrer = createRefreshDeferrer([this]() { refreshVTables(); });
}
VTablesWidget::~VTablesWidget() {}
void VTablesWidget::refreshVTables()
{
if (!refreshDeferrer->attemptRefresh(nullptr)) {
return;
}
model->beginResetModel();
vtables = Core()->getAllVTables();
model->endResetModel();
qhelpers::adjustColumns(ui->vTableTreeView, 3, 0);
ui->vTableTreeView->setColumnWidth(0, 200);
tree->showItemsNumber(proxy->rowCount());
}
void VTablesWidget::on_vTableTreeView_doubleClicked(const QModelIndex &index)
{
if (!index.isValid()) {
return;
}
QModelIndex parent = index.parent();
if (parent.isValid()) {
Core()->seekAndShow(index.data(VTableModel::VTableDescriptionRole)
.value<BinClassMethodDescription>()
.addr);
} else {
Core()->seekAndShow(
index.data(VTableModel::VTableDescriptionRole).value<VTableDescription>().addr);
}
}