cutter/src/widgets/FlagsWidget.cpp

285 lines
8.3 KiB
C++
Raw Normal View History

#include "FlagsWidget.h"
#include "ui_FlagsWidget.h"
#include "core/MainWindow.h"
2018-10-17 07:55:53 +00:00
#include "common/Helpers.h"
2019-03-14 09:28:42 +00:00
#include <QComboBox>
#include <QMenu>
#include <QShortcut>
#include <QTreeWidget>
#include <QStandardItemModel>
2020-06-14 08:52:33 +00:00
#include <QInputDialog>
2019-03-14 09:28:42 +00:00
2017-05-06 10:46:28 +00:00
FlagsModel::FlagsModel(QList<FlagDescription> *flags, QObject *parent)
2021-01-24 14:50:13 +00:00
: AddressableItemModel<QAbstractListModel>(parent), flags(flags)
2017-05-04 14:16:21 +00:00
{
}
2017-05-06 10:46:28 +00:00
int FlagsModel::rowCount(const QModelIndex &) const
2017-05-04 14:16:21 +00:00
{
return flags->count();
}
2017-05-06 10:46:28 +00:00
int FlagsModel::columnCount(const QModelIndex &) const
2017-05-04 14:16:21 +00:00
{
return Columns::COUNT;
}
QVariant FlagsModel::data(const QModelIndex &index, int role) const
{
2017-06-03 12:27:23 +00:00
if (index.row() >= flags->count())
2017-05-04 14:16:21 +00:00
return QVariant();
const FlagDescription &flag = flags->at(index.row());
2018-03-21 20:32:32 +00:00
switch (role) {
2017-06-03 12:27:23 +00:00
case Qt::DisplayRole:
2018-03-21 20:32:32 +00:00
switch (index.column()) {
2017-06-03 12:27:23 +00:00
case SIZE:
2021-09-14 13:32:04 +00:00
return RzSizeString(flag.size);
2017-06-03 12:27:23 +00:00
case OFFSET:
2021-09-14 13:32:04 +00:00
return RzAddressString(flag.offset);
2017-06-03 12:27:23 +00:00
case NAME:
return flag.name;
case REALNAME:
return flag.realname;
case COMMENT:
return Core()->getCommentAt(flag.offset);
2017-05-04 14:16:21 +00:00
default:
return QVariant();
2017-06-03 12:27:23 +00:00
}
case FlagDescriptionRole:
return QVariant::fromValue(flag);
default:
return QVariant();
2017-05-04 14:16:21 +00:00
}
}
QVariant FlagsModel::headerData(int section, Qt::Orientation, int role) const
{
2018-03-21 20:32:32 +00:00
switch (role) {
2017-06-03 12:27:23 +00:00
case Qt::DisplayRole:
2018-03-21 20:32:32 +00:00
switch (section) {
2017-06-03 12:27:23 +00:00
case SIZE:
return tr("Size");
case OFFSET:
return tr("Offset");
case NAME:
return tr("Name");
case REALNAME:
return tr("Real Name");
case COMMENT:
return tr("Comment");
2017-05-04 14:16:21 +00:00
default:
return QVariant();
2017-06-03 12:27:23 +00:00
}
default:
return QVariant();
2017-05-04 14:16:21 +00:00
}
}
RVA FlagsModel::address(const QModelIndex &index) const
{
const FlagDescription &flag = flags->at(index.row());
return flag.offset;
}
2017-05-04 14:16:21 +00:00
QString FlagsModel::name(const QModelIndex &index) const
{
const FlagDescription &flag = flags->at(index.row());
return flag.name;
}
2017-05-04 14:16:21 +00:00
const FlagDescription *FlagsModel::description(QModelIndex index) const
{
if (index.row() < flags->size()) {
return &flags->at(index.row());
}
return nullptr;
}
2017-05-04 14:16:21 +00:00
FlagsSortFilterProxyModel::FlagsSortFilterProxyModel(FlagsModel *source_model, QObject *parent)
: AddressableFilterProxyModel(source_model, parent)
2017-05-04 14:16:21 +00:00
{
}
bool FlagsSortFilterProxyModel::filterAcceptsRow(int row, const QModelIndex &parent) const
{
QModelIndex index = sourceModel()->index(row, 0, parent);
FlagDescription flag = index.data(FlagsModel::FlagDescriptionRole).value<FlagDescription>();
return qhelpers::filterStringContains(flag.name, this);
2017-05-04 14:16:21 +00:00
}
bool FlagsSortFilterProxyModel::lessThan(const QModelIndex &left, const QModelIndex &right) const
{
auto source = static_cast<FlagsModel *>(sourceModel());
auto left_flag = source->description(left);
auto right_flag = source->description(right);
2017-05-04 14:16:21 +00:00
2018-03-21 20:32:32 +00:00
switch (left.column()) {
2017-06-03 12:27:23 +00:00
case FlagsModel::SIZE:
if (left_flag->size != right_flag->size)
return left_flag->size < right_flag->size;
2017-06-03 12:27:23 +00:00
// fallthrough
case FlagsModel::OFFSET:
if (left_flag->offset != right_flag->offset)
return left_flag->offset < right_flag->offset;
2017-06-03 12:27:23 +00:00
// fallthrough
case FlagsModel::NAME:
return left_flag->name < right_flag->name;
2021-01-24 14:50:13 +00:00
case FlagsModel::REALNAME:
return left_flag->realname < right_flag->realname;
case FlagsModel::COMMENT:
return Core()->getCommentAt(left_flag->offset) < Core()->getCommentAt(right_flag->offset);
2017-06-03 12:27:23 +00:00
default:
break;
2017-05-04 14:16:21 +00:00
}
// fallback
return left_flag->offset < right_flag->offset;
2017-05-04 14:16:21 +00:00
}
2021-01-24 14:50:13 +00:00
FlagsWidget::FlagsWidget(MainWindow *main)
: CutterDockWidget(main), ui(new Ui::FlagsWidget), main(main), tree(new CutterTreeWidget(this))
{
ui->setupUi(this);
// Add Status Bar footer
tree->addStatusBar(ui->verticalLayout);
2017-05-06 10:46:28 +00:00
flags_model = new FlagsModel(&flags, this);
2017-05-04 14:16:21 +00:00
flags_proxy_model = new FlagsSortFilterProxyModel(flags_model, this);
2021-01-24 14:50:13 +00:00
connect(ui->filterLineEdit, &QLineEdit::textChanged, flags_proxy_model,
&QSortFilterProxyModel::setFilterWildcard);
ui->flagsTreeView->setMainWindow(mainWindow);
2017-05-04 14:16:21 +00:00
ui->flagsTreeView->setModel(flags_proxy_model);
ui->flagsTreeView->sortByColumn(FlagsModel::OFFSET, Qt::AscendingOrder);
// Ctrl-F to move the focus to the Filter search box
QShortcut *searchShortcut = new QShortcut(QKeySequence::Find, this);
2021-01-24 14:50:13 +00:00
connect(searchShortcut, &QShortcut::activated, ui->filterLineEdit,
[this]() { ui->filterLineEdit->setFocus(); });
searchShortcut->setContext(Qt::WidgetWithChildrenShortcut);
// Esc to clear the filter entry
QShortcut *clearShortcut = new QShortcut(QKeySequence(Qt::Key_Escape), this);
connect(clearShortcut, &QShortcut::activated, [this] {
if (ui->filterLineEdit->text().isEmpty()) {
ui->flagsTreeView->setFocus();
} else {
ui->filterLineEdit->setText("");
}
});
clearShortcut->setContext(Qt::WidgetWithChildrenShortcut);
2021-01-24 14:50:13 +00:00
connect(ui->filterLineEdit, &QLineEdit::textChanged, this,
[this] { tree->showItemsNumber(flags_proxy_model->rowCount()); });
setScrollMode();
connect(Core(), &CutterCore::flagsChanged, this, &FlagsWidget::flagsChanged);
connect(Core(), &CutterCore::codeRebased, this, &FlagsWidget::flagsChanged);
connect(Core(), &CutterCore::refreshAll, this, &FlagsWidget::refreshFlagspaces);
2021-01-24 14:50:13 +00:00
connect(Core(), &CutterCore::commentsChanged, this,
[this]() { qhelpers::emitColumnChanged(flags_model, FlagsModel::COMMENT); });
auto menu = ui->flagsTreeView->getItemContextMenu();
menu->addSeparator();
menu->addAction(ui->actionRename);
menu->addAction(ui->actionDelete);
addAction(ui->actionRename);
addAction(ui->actionDelete);
}
FlagsWidget::~FlagsWidget() {}
void FlagsWidget::on_flagspaceCombo_currentTextChanged(const QString &arg1)
{
2017-10-01 18:08:12 +00:00
Q_UNUSED(arg1);
refreshFlags();
}
2018-03-11 16:40:52 +00:00
void FlagsWidget::on_actionRename_triggered()
{
2021-01-24 14:50:13 +00:00
FlagDescription flag = ui->flagsTreeView->selectionModel()
->currentIndex()
.data(FlagsModel::FlagDescriptionRole)
.value<FlagDescription>();
2018-03-11 16:40:52 +00:00
2020-06-14 08:52:33 +00:00
bool ok;
QString newName = QInputDialog::getText(this, tr("Rename flag %1").arg(flag.name),
2021-01-24 14:50:13 +00:00
tr("Flag name:"), QLineEdit::Normal, flag.name, &ok);
2020-06-14 08:52:33 +00:00
if (ok && !newName.isEmpty()) {
Core()->renameFlag(flag.name, newName);
2018-03-11 16:40:52 +00:00
}
}
void FlagsWidget::on_actionDelete_triggered()
{
2021-01-24 14:50:13 +00:00
FlagDescription flag = ui->flagsTreeView->selectionModel()
->currentIndex()
.data(FlagsModel::FlagDescriptionRole)
.value<FlagDescription>();
2018-03-11 16:40:52 +00:00
Core()->delFlag(flag.name);
}
void FlagsWidget::flagsChanged()
{
refreshFlagspaces();
}
void FlagsWidget::refreshFlagspaces()
{
int cur_idx = ui->flagspaceCombo->currentIndex();
if (cur_idx < 0)
cur_idx = 0;
2021-01-24 14:50:13 +00:00
disableFlagRefresh =
true; // prevent duplicate flag refresh caused by flagspaceCombo modifications
ui->flagspaceCombo->clear();
ui->flagspaceCombo->addItem(tr("(all)"));
for (const FlagspaceDescription &i : Core()->getAllFlagspaces()) {
ui->flagspaceCombo->addItem(i.name, QVariant::fromValue(i));
}
if (cur_idx > 0)
ui->flagspaceCombo->setCurrentIndex(cur_idx);
disableFlagRefresh = false;
refreshFlags();
}
void FlagsWidget::refreshFlags()
{
if (disableFlagRefresh) {
return;
}
QString flagspace;
QVariant flagspace_data = ui->flagspaceCombo->currentData();
2017-06-03 12:27:23 +00:00
if (flagspace_data.isValid())
flagspace = flagspace_data.value<FlagspaceDescription>().name;
flags_model->beginResetModel();
2018-04-12 06:33:30 +00:00
flags = Core()->getAllFlags(flagspace);
flags_model->endResetModel();
tree->showItemsNumber(flags_proxy_model->rowCount());
// TODO: this is not a very good place for the following:
2017-05-04 14:16:21 +00:00
QStringList flagNames;
for (const FlagDescription &i : flags)
2017-05-04 14:16:21 +00:00
flagNames.append(i.name);
main->refreshOmniBar(flagNames);
}
void FlagsWidget::setScrollMode()
{
2017-05-04 14:16:21 +00:00
qhelpers::setVerticalScrollMode(ui->flagsTreeView);
}