2017-10-02 16:18:40 +00:00
|
|
|
#include "FunctionsWidget.h"
|
|
|
|
#include "ui_FunctionsWidget.h"
|
2017-03-29 10:18:37 +00:00
|
|
|
|
2017-10-01 19:09:42 +00:00
|
|
|
#include "MainWindow.h"
|
|
|
|
#include "utils/Helpers.h"
|
|
|
|
#include "dialogs/CommentsDialog.h"
|
|
|
|
#include "dialogs/RenameDialog.h"
|
|
|
|
#include "dialogs/XrefsDialog.h"
|
2017-03-29 10:18:37 +00:00
|
|
|
|
2017-04-28 13:09:40 +00:00
|
|
|
#include <algorithm>
|
2017-03-29 10:18:37 +00:00
|
|
|
#include <QMenu>
|
|
|
|
#include <QDebug>
|
|
|
|
#include <QString>
|
2017-04-28 13:09:40 +00:00
|
|
|
#include <QResource>
|
2017-05-18 10:18:17 +00:00
|
|
|
#include <QShortcut>
|
2017-04-28 13:09:40 +00:00
|
|
|
|
2017-11-07 13:11:33 +00:00
|
|
|
FunctionModel::FunctionModel(QList<FunctionDescription> *functions, QSet<RVA> *import_addresses, bool nested, QFont default_font, QFont highlight_font, QObject *parent)
|
2017-04-28 13:38:01 +00:00
|
|
|
: QAbstractItemModel(parent),
|
|
|
|
functions(functions),
|
|
|
|
import_addresses(import_addresses),
|
|
|
|
highlight_font(highlight_font),
|
|
|
|
default_font(default_font),
|
|
|
|
nested(nested),
|
|
|
|
current_index(-1)
|
2017-04-28 13:09:40 +00:00
|
|
|
|
2017-04-28 13:38:01 +00:00
|
|
|
{
|
2017-11-07 13:11:33 +00:00
|
|
|
connect(Core(), SIGNAL(seekChanged(RVA)), this, SLOT(seekChanged(RVA)));
|
2017-12-20 22:50:26 +00:00
|
|
|
connect(Core(), SIGNAL(functionRenamed(const QString &, const QString &)), this, SLOT(functionRenamed(QString, QString)));
|
2017-04-28 13:09:40 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
QModelIndex FunctionModel::index(int row, int column, const QModelIndex &parent) const
|
|
|
|
{
|
2017-04-28 13:38:01 +00:00
|
|
|
if (!parent.isValid())
|
2017-04-28 13:09:40 +00:00
|
|
|
return createIndex(row, column, (quintptr)0); // root function nodes have id = 0
|
|
|
|
|
|
|
|
return createIndex(row, column, (quintptr)(parent.row() + 1)); // sub-nodes have id = function index + 1
|
|
|
|
}
|
|
|
|
|
|
|
|
QModelIndex FunctionModel::parent(const QModelIndex &index) const
|
|
|
|
{
|
2017-04-28 13:38:01 +00:00
|
|
|
if (!index.isValid() || index.column() != 0)
|
2017-04-28 13:09:40 +00:00
|
|
|
return QModelIndex();
|
|
|
|
|
2017-04-28 13:38:01 +00:00
|
|
|
if (index.internalId() == 0) // root function node
|
2017-04-28 13:09:40 +00:00
|
|
|
return QModelIndex();
|
|
|
|
else // sub-node
|
2017-04-28 13:38:01 +00:00
|
|
|
return this->index((int)(index.internalId() - 1), 0);
|
2017-04-28 13:09:40 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
int FunctionModel::rowCount(const QModelIndex &parent) const
|
|
|
|
{
|
2017-04-28 13:38:01 +00:00
|
|
|
if (!parent.isValid())
|
2017-04-28 13:09:40 +00:00
|
|
|
return functions->count();
|
|
|
|
|
2017-04-28 13:38:01 +00:00
|
|
|
if (nested)
|
2017-04-28 13:09:40 +00:00
|
|
|
{
|
2017-04-28 13:38:01 +00:00
|
|
|
if (parent.internalId() == 0)
|
2017-04-28 13:09:40 +00:00
|
|
|
return 3; // sub-nodes for nested functions
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2017-04-28 13:38:01 +00:00
|
|
|
int FunctionModel::columnCount(const QModelIndex &/*parent*/) const
|
2017-04-28 13:09:40 +00:00
|
|
|
{
|
2017-04-28 13:38:01 +00:00
|
|
|
if (nested)
|
2017-04-28 13:09:40 +00:00
|
|
|
return 1;
|
|
|
|
else
|
2017-11-04 15:28:02 +00:00
|
|
|
return ColumnCount;
|
2017-04-28 13:09:40 +00:00
|
|
|
}
|
|
|
|
|
2017-12-20 15:40:46 +00:00
|
|
|
bool FunctionModel::functionIsImport(ut64 addr) const
|
|
|
|
{
|
|
|
|
return import_addresses->contains(addr);
|
|
|
|
}
|
|
|
|
|
2017-04-28 13:09:40 +00:00
|
|
|
|
|
|
|
QVariant FunctionModel::data(const QModelIndex &index, int role) const
|
|
|
|
{
|
2017-04-28 13:38:01 +00:00
|
|
|
if (!index.isValid())
|
2017-04-28 13:09:40 +00:00
|
|
|
return QVariant();
|
|
|
|
|
|
|
|
int function_index;
|
|
|
|
bool subnode;
|
2017-04-28 13:38:01 +00:00
|
|
|
if (index.internalId() != 0) // sub-node
|
2017-04-28 13:09:40 +00:00
|
|
|
{
|
|
|
|
function_index = index.parent().row();
|
|
|
|
subnode = true;
|
|
|
|
}
|
|
|
|
else // root function node
|
|
|
|
{
|
|
|
|
function_index = index.row();
|
|
|
|
subnode = false;
|
|
|
|
}
|
|
|
|
|
|
|
|
const FunctionDescription &function = functions->at(function_index);
|
|
|
|
|
2017-04-28 13:38:01 +00:00
|
|
|
if (function_index >= functions->count())
|
2017-04-28 13:09:40 +00:00
|
|
|
return QVariant();
|
|
|
|
|
2017-04-28 13:38:01 +00:00
|
|
|
switch (role)
|
2017-04-28 13:09:40 +00:00
|
|
|
{
|
2017-04-28 13:38:01 +00:00
|
|
|
case Qt::DisplayRole:
|
|
|
|
if (nested)
|
|
|
|
{
|
|
|
|
if (subnode)
|
2017-04-28 13:09:40 +00:00
|
|
|
{
|
2017-04-28 13:38:01 +00:00
|
|
|
switch (index.row())
|
2017-04-28 13:09:40 +00:00
|
|
|
{
|
2017-04-28 13:38:01 +00:00
|
|
|
case 0:
|
|
|
|
return tr("Offset: %1").arg(RAddressString(function.offset));
|
|
|
|
case 1:
|
|
|
|
return tr("Size: %1").arg(RSizeString(function.size));
|
|
|
|
case 2:
|
2017-12-20 15:40:46 +00:00
|
|
|
return tr("Import: %1").arg(functionIsImport(function.offset) ? tr("true") : tr("false"));
|
2017-04-28 13:38:01 +00:00
|
|
|
default:
|
|
|
|
return QVariant();
|
2017-04-28 13:09:40 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
else
|
2017-04-28 13:38:01 +00:00
|
|
|
return function.name;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
switch (index.column())
|
2017-04-28 13:09:40 +00:00
|
|
|
{
|
2017-11-04 15:28:02 +00:00
|
|
|
case NameColumn:
|
2017-10-13 13:53:23 +00:00
|
|
|
return function.name;
|
2017-11-04 15:28:02 +00:00
|
|
|
case SizeColumn:
|
2017-04-28 13:38:01 +00:00
|
|
|
return RSizeString(function.size);
|
2017-11-04 15:28:02 +00:00
|
|
|
case OffsetColumn:
|
2017-10-13 13:53:23 +00:00
|
|
|
return RAddressString(function.offset);
|
2017-04-28 13:38:01 +00:00
|
|
|
default:
|
|
|
|
return QVariant();
|
2017-04-28 13:09:40 +00:00
|
|
|
}
|
2017-04-28 13:38:01 +00:00
|
|
|
}
|
2017-04-28 13:09:40 +00:00
|
|
|
|
2017-04-28 13:38:01 +00:00
|
|
|
case Qt::DecorationRole:
|
|
|
|
if (import_addresses->contains(function.offset) &&
|
2017-11-04 15:28:02 +00:00
|
|
|
(nested ? false : index.column() == ImportColumn))
|
2017-04-28 13:38:01 +00:00
|
|
|
return QIcon(":/img/icons/import_light.svg");
|
|
|
|
return QVariant();
|
2017-04-28 13:09:40 +00:00
|
|
|
|
2017-04-28 13:38:01 +00:00
|
|
|
case Qt::FontRole:
|
|
|
|
if (current_index == function_index)
|
|
|
|
return highlight_font;
|
|
|
|
return default_font;
|
2017-04-28 13:09:40 +00:00
|
|
|
|
2017-04-28 13:38:01 +00:00
|
|
|
case Qt::ToolTipRole:
|
|
|
|
{
|
2017-10-09 18:08:35 +00:00
|
|
|
QList<QString> info = CutterCore::getInstance()->cmd("afi @ " + function.name).split("\n");
|
2017-04-28 13:38:01 +00:00
|
|
|
if (info.length() > 2)
|
2017-04-28 13:09:40 +00:00
|
|
|
{
|
2017-04-28 13:38:01 +00:00
|
|
|
QString size = info[4].split(" ")[1];
|
|
|
|
QString complex = info[8].split(" ")[1];
|
|
|
|
QString bb = info[11].split(" ")[1];
|
|
|
|
return QString("Summary:\n\n Size: " + size +
|
|
|
|
"\n Cyclomatic complexity: " + complex +
|
|
|
|
"\n Basic blocks: " + bb +
|
2017-10-09 18:08:35 +00:00
|
|
|
"\n\nDisasm preview:\n\n" + CutterCore::getInstance()->cmd("pdi 10 @ " + function.name) +
|
|
|
|
"\nStrings:\n\n" + CutterCore::getInstance()->cmd("pdsf @ " + function.name));
|
2017-04-28 13:09:40 +00:00
|
|
|
}
|
2017-04-28 13:38:01 +00:00
|
|
|
return QVariant();
|
|
|
|
}
|
2017-04-28 13:09:40 +00:00
|
|
|
|
2017-12-20 15:40:46 +00:00
|
|
|
case Qt::ForegroundRole:
|
|
|
|
if (functionIsImport(function.offset))
|
|
|
|
return QVariant(ConfigColor("gui.imports"));
|
|
|
|
return QVariant(QColor(Qt::black));
|
|
|
|
|
2017-04-28 13:38:01 +00:00
|
|
|
case FunctionDescriptionRole:
|
|
|
|
return QVariant::fromValue(function);
|
2017-04-28 13:09:40 +00:00
|
|
|
|
2017-04-28 13:38:01 +00:00
|
|
|
case IsImportRole:
|
|
|
|
return import_addresses->contains(function.offset);
|
2017-04-28 13:09:40 +00:00
|
|
|
|
2017-04-28 13:38:01 +00:00
|
|
|
default:
|
|
|
|
return QVariant();
|
2017-04-28 13:09:40 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
QVariant FunctionModel::headerData(int section, Qt::Orientation orientation, int role) const
|
|
|
|
{
|
2017-04-28 13:38:01 +00:00
|
|
|
if (role == Qt::DisplayRole && orientation == Qt::Horizontal)
|
2017-04-28 13:09:40 +00:00
|
|
|
{
|
2017-04-28 13:38:01 +00:00
|
|
|
if (nested)
|
2017-04-28 13:09:40 +00:00
|
|
|
{
|
|
|
|
return tr("Name");
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
switch (section)
|
|
|
|
{
|
2017-11-04 15:28:02 +00:00
|
|
|
case NameColumn:
|
2017-10-13 13:53:23 +00:00
|
|
|
return tr("Name");
|
2017-11-04 15:28:02 +00:00
|
|
|
case SizeColumn:
|
2017-04-28 13:38:01 +00:00
|
|
|
return tr("Size");
|
2017-11-04 15:28:02 +00:00
|
|
|
case ImportColumn:
|
2017-04-28 13:38:01 +00:00
|
|
|
return tr("Imp.");
|
2017-11-04 15:28:02 +00:00
|
|
|
case OffsetColumn:
|
2017-10-13 13:53:23 +00:00
|
|
|
return tr("Offset");
|
2017-04-28 13:38:01 +00:00
|
|
|
default:
|
|
|
|
return QVariant();
|
2017-04-28 13:09:40 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return QVariant();
|
|
|
|
}
|
|
|
|
|
|
|
|
void FunctionModel::beginReloadFunctions()
|
|
|
|
{
|
|
|
|
beginResetModel();
|
|
|
|
}
|
|
|
|
|
|
|
|
void FunctionModel::endReloadFunctions()
|
|
|
|
{
|
|
|
|
updateCurrentIndex();
|
|
|
|
endResetModel();
|
|
|
|
}
|
|
|
|
|
2017-11-07 13:11:33 +00:00
|
|
|
void FunctionModel::seekChanged(RVA)
|
2017-04-28 13:09:40 +00:00
|
|
|
{
|
2017-11-07 13:11:33 +00:00
|
|
|
if (updateCurrentIndex())
|
|
|
|
{
|
|
|
|
emit dataChanged(index(0, 0), index(rowCount() - 1, columnCount() - 1));
|
|
|
|
}
|
2017-04-28 13:09:40 +00:00
|
|
|
}
|
|
|
|
|
2017-11-07 13:11:33 +00:00
|
|
|
bool FunctionModel::updateCurrentIndex()
|
2017-04-28 13:09:40 +00:00
|
|
|
{
|
|
|
|
int index = -1;
|
|
|
|
RVA offset = 0;
|
|
|
|
|
2017-11-07 13:11:33 +00:00
|
|
|
RVA seek = Core()->getOffset();
|
|
|
|
|
2017-04-28 13:38:01 +00:00
|
|
|
for (int i = 0; i < functions->count(); i++)
|
2017-04-28 13:09:40 +00:00
|
|
|
{
|
|
|
|
const FunctionDescription &function = functions->at(i);
|
|
|
|
|
2017-11-07 13:11:33 +00:00
|
|
|
if (function.contains(seek)
|
2017-04-28 13:38:01 +00:00
|
|
|
&& function.offset >= offset)
|
2017-04-28 13:09:40 +00:00
|
|
|
{
|
|
|
|
offset = function.offset;
|
|
|
|
index = i;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-11-07 13:11:33 +00:00
|
|
|
bool changed = current_index != index;
|
|
|
|
|
2017-04-28 13:09:40 +00:00
|
|
|
current_index = index;
|
2017-11-07 13:11:33 +00:00
|
|
|
|
|
|
|
return changed;
|
2017-04-28 13:09:40 +00:00
|
|
|
}
|
|
|
|
|
2017-04-28 13:38:01 +00:00
|
|
|
void FunctionModel::functionRenamed(const QString &prev_name, const QString &new_name)
|
2017-04-28 13:09:40 +00:00
|
|
|
{
|
2017-04-28 13:38:01 +00:00
|
|
|
for (int i = 0; i < functions->count(); i++)
|
2017-04-28 13:09:40 +00:00
|
|
|
{
|
|
|
|
FunctionDescription &function = (*functions)[i];
|
2017-04-28 13:38:01 +00:00
|
|
|
if (function.name == prev_name)
|
2017-04-28 13:09:40 +00:00
|
|
|
{
|
|
|
|
function.name = new_name;
|
2017-04-28 13:38:01 +00:00
|
|
|
emit dataChanged(index(i, 0), index(i, columnCount() - 1));
|
2017-04-28 13:09:40 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
FunctionSortFilterProxyModel::FunctionSortFilterProxyModel(FunctionModel *source_model, QObject *parent)
|
2017-04-28 13:38:01 +00:00
|
|
|
: QSortFilterProxyModel(parent)
|
2017-04-28 13:09:40 +00:00
|
|
|
{
|
|
|
|
setSourceModel(source_model);
|
|
|
|
setFilterCaseSensitivity(Qt::CaseInsensitive);
|
|
|
|
setSortCaseSensitivity(Qt::CaseInsensitive);
|
|
|
|
}
|
|
|
|
|
|
|
|
bool FunctionSortFilterProxyModel::filterAcceptsRow(int row, const QModelIndex &parent) const
|
|
|
|
{
|
|
|
|
QModelIndex index = sourceModel()->index(row, 0, parent);
|
|
|
|
FunctionDescription function = index.data(FunctionModel::FunctionDescriptionRole).value<FunctionDescription>();
|
|
|
|
return function.name.contains(filterRegExp());
|
|
|
|
}
|
|
|
|
|
|
|
|
bool FunctionSortFilterProxyModel::lessThan(const QModelIndex &left, const QModelIndex &right) const
|
|
|
|
{
|
2017-04-28 13:38:01 +00:00
|
|
|
if (!left.isValid() || !right.isValid())
|
2017-04-28 13:09:40 +00:00
|
|
|
return false;
|
|
|
|
|
2017-04-28 13:38:01 +00:00
|
|
|
if (left.parent().isValid() || right.parent().isValid())
|
2017-04-28 13:09:40 +00:00
|
|
|
return false;
|
|
|
|
|
|
|
|
FunctionDescription left_function = left.data(FunctionModel::FunctionDescriptionRole).value<FunctionDescription>();
|
|
|
|
FunctionDescription right_function = right.data(FunctionModel::FunctionDescriptionRole).value<FunctionDescription>();
|
|
|
|
|
|
|
|
|
2017-04-28 13:38:01 +00:00
|
|
|
if (static_cast<FunctionModel *>(sourceModel())->isNested())
|
2017-04-28 13:09:40 +00:00
|
|
|
{
|
|
|
|
return left_function.name < right_function.name;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
switch (left.column())
|
|
|
|
{
|
2017-11-04 15:28:02 +00:00
|
|
|
case FunctionModel::OffsetColumn:
|
2017-04-28 13:38:01 +00:00
|
|
|
return left_function.offset < right_function.offset;
|
2017-11-04 15:28:02 +00:00
|
|
|
case FunctionModel::SizeColumn:
|
2017-04-28 13:38:01 +00:00
|
|
|
if (left_function.size != right_function.size)
|
|
|
|
return left_function.size < right_function.size;
|
|
|
|
break;
|
2017-11-04 15:28:02 +00:00
|
|
|
case FunctionModel::ImportColumn:
|
2017-04-28 13:38:01 +00:00
|
|
|
{
|
|
|
|
bool left_is_import = left.data(FunctionModel::IsImportRole).toBool();
|
|
|
|
bool right_is_import = right.data(FunctionModel::IsImportRole).toBool();
|
|
|
|
if (!left_is_import && right_is_import)
|
|
|
|
return true;
|
|
|
|
break;
|
|
|
|
}
|
2017-11-04 15:28:02 +00:00
|
|
|
case FunctionModel::NameColumn:
|
2017-04-28 13:38:01 +00:00
|
|
|
return left_function.name < right_function.name;
|
|
|
|
default:
|
|
|
|
return false;
|
2017-04-28 13:09:40 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
return left_function.offset < right_function.offset;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-03-29 10:18:37 +00:00
|
|
|
FunctionsWidget::FunctionsWidget(MainWindow *main, QWidget *parent) :
|
2017-11-19 12:56:10 +00:00
|
|
|
QDockWidget(parent),
|
2017-04-13 15:14:02 +00:00
|
|
|
ui(new Ui::FunctionsWidget),
|
|
|
|
main(main)
|
2017-03-29 10:18:37 +00:00
|
|
|
{
|
|
|
|
ui->setupUi(this);
|
|
|
|
|
2017-04-28 13:09:40 +00:00
|
|
|
// Radare core found in:
|
|
|
|
this->main = main;
|
|
|
|
|
2017-05-18 10:18:17 +00:00
|
|
|
// leave the filter visible by default so users know it exists
|
|
|
|
//ui->filterLineEdit->setVisible(false);
|
|
|
|
|
|
|
|
// Ctrl-F to show/hide the filter entry
|
2017-12-19 18:56:18 +00:00
|
|
|
QShortcut *search_shortcut = new QShortcut(QKeySequence::Find, this);
|
2017-12-19 18:38:06 +00:00
|
|
|
connect(search_shortcut, &QShortcut::activated, ui->quickFilterView, &QuickFilterView::showFilter);
|
2017-05-18 10:18:17 +00:00
|
|
|
search_shortcut->setContext(Qt::WidgetWithChildrenShortcut);
|
|
|
|
|
|
|
|
// Esc to clear the filter entry
|
2017-12-21 11:08:49 +00:00
|
|
|
QShortcut *clear_shortcut = new QShortcut(QKeySequence(Qt::Key_Escape), this);
|
2017-12-19 18:38:06 +00:00
|
|
|
connect(clear_shortcut, &QShortcut::activated, ui->quickFilterView, &QuickFilterView::clearFilter);
|
2017-05-18 10:18:17 +00:00
|
|
|
clear_shortcut->setContext(Qt::WidgetWithChildrenShortcut);
|
|
|
|
|
2017-04-28 13:09:40 +00:00
|
|
|
QFontInfo font_info = ui->functionsTreeView->fontInfo();
|
|
|
|
QFont default_font = QFont(font_info.family(), font_info.pointSize());
|
|
|
|
QFont highlight_font = QFont(font_info.family(), font_info.pointSize(), QFont::Bold);
|
|
|
|
|
2017-11-07 13:11:33 +00:00
|
|
|
function_model = new FunctionModel(&functions, &import_addresses, false, default_font, highlight_font, this);
|
2017-04-28 13:09:40 +00:00
|
|
|
function_proxy_model = new FunctionSortFilterProxyModel(function_model, this);
|
|
|
|
ui->functionsTreeView->setModel(function_proxy_model);
|
2017-11-26 21:31:27 +00:00
|
|
|
ui->functionsTreeView->sortByColumn(FunctionModel::NameColumn, Qt::AscendingOrder);
|
2017-04-28 13:09:40 +00:00
|
|
|
|
2017-11-07 13:11:33 +00:00
|
|
|
nested_function_model = new FunctionModel(&functions, &import_addresses, true, default_font, highlight_font, this);
|
2017-04-28 13:09:40 +00:00
|
|
|
nested_function_proxy_model = new FunctionSortFilterProxyModel(nested_function_model, this);
|
|
|
|
ui->nestedFunctionsTreeView->setModel(nested_function_proxy_model);
|
2017-11-26 21:31:27 +00:00
|
|
|
ui->nestedFunctionsTreeView->sortByColumn(0, Qt::AscendingOrder);
|
2017-04-28 13:09:40 +00:00
|
|
|
|
2017-12-19 18:38:06 +00:00
|
|
|
connect(ui->quickFilterView, SIGNAL(filterTextChanged(const QString &)), function_proxy_model, SLOT(setFilterWildcard(const QString &)));
|
|
|
|
connect(ui->quickFilterView, SIGNAL(filterTextChanged(const QString &)), nested_function_proxy_model, SLOT(setFilterWildcard(const QString &)));
|
|
|
|
connect(ui->quickFilterView, &QuickFilterView::filterClosed, this, [this]()
|
|
|
|
{
|
|
|
|
getCurrentTreeView()->setFocus();
|
|
|
|
});
|
|
|
|
|
2017-11-19 12:56:10 +00:00
|
|
|
setScrollMode();
|
2017-04-28 13:09:40 +00:00
|
|
|
|
2017-03-29 10:18:37 +00:00
|
|
|
// Set Functions context menu
|
2017-04-28 13:09:40 +00:00
|
|
|
connect(ui->functionsTreeView, SIGNAL(customContextMenuRequested(const QPoint &)),
|
2017-03-29 10:18:37 +00:00
|
|
|
this, SLOT(showFunctionsContextMenu(const QPoint &)));
|
2017-04-28 13:09:40 +00:00
|
|
|
connect(ui->nestedFunctionsTreeView, SIGNAL(customContextMenuRequested(const QPoint &)),
|
2017-03-29 10:18:37 +00:00
|
|
|
this, SLOT(showFunctionsContextMenu(const QPoint &)));
|
|
|
|
|
2017-04-28 13:09:40 +00:00
|
|
|
|
2017-05-04 14:16:21 +00:00
|
|
|
connect(ui->functionsTreeView, SIGNAL(doubleClicked(const QModelIndex &)), this, SLOT(functionsTreeView_doubleClicked(const QModelIndex &)));
|
|
|
|
connect(ui->nestedFunctionsTreeView, SIGNAL(doubleClicked(const QModelIndex &)), this, SLOT(functionsTreeView_doubleClicked(const QModelIndex &)));
|
2017-04-13 15:14:02 +00:00
|
|
|
|
2017-03-29 10:18:37 +00:00
|
|
|
// Hide the tabs
|
|
|
|
QTabBar *tabs = ui->tabWidget->tabBar();
|
|
|
|
tabs->setVisible(false);
|
|
|
|
|
|
|
|
// Use a custom context menu on the dock title bar
|
|
|
|
//this->title_bar = this->titleBarWidget();
|
|
|
|
ui->actionHorizontal->setChecked(true);
|
|
|
|
this->setContextMenuPolicy(Qt::CustomContextMenu);
|
|
|
|
connect(this, SIGNAL(customContextMenuRequested(const QPoint &)),
|
|
|
|
this, SLOT(showTitleContextMenu(const QPoint &)));
|
|
|
|
|
2017-12-11 13:07:12 +00:00
|
|
|
connect(Core(), SIGNAL(functionsChanged()), this, SLOT(refreshTree()));
|
2017-11-19 12:56:10 +00:00
|
|
|
connect(Core(), SIGNAL(refreshAll()), this, SLOT(refreshTree()));
|
2017-04-13 15:14:02 +00:00
|
|
|
}
|
|
|
|
|
2017-11-19 12:56:10 +00:00
|
|
|
FunctionsWidget::~FunctionsWidget() {}
|
2017-04-13 15:14:02 +00:00
|
|
|
|
2017-04-28 13:09:40 +00:00
|
|
|
void FunctionsWidget::refreshTree()
|
2017-04-09 19:55:06 +00:00
|
|
|
{
|
2017-04-28 13:09:40 +00:00
|
|
|
function_model->beginReloadFunctions();
|
|
|
|
nested_function_model->beginReloadFunctions();
|
|
|
|
|
2017-10-09 18:08:35 +00:00
|
|
|
functions = CutterCore::getInstance()->getAllFunctions();
|
2017-03-29 10:18:37 +00:00
|
|
|
|
2017-04-28 13:09:40 +00:00
|
|
|
import_addresses.clear();
|
2017-10-09 18:08:35 +00:00
|
|
|
foreach (ImportDescription import, CutterCore::getInstance()->getAllImports())
|
2017-04-28 13:09:40 +00:00
|
|
|
import_addresses.insert(import.plt);
|
|
|
|
|
|
|
|
function_model->endReloadFunctions();
|
|
|
|
nested_function_model->endReloadFunctions();
|
|
|
|
|
|
|
|
// resize offset and size columns
|
|
|
|
ui->functionsTreeView->resizeColumnToContents(0);
|
|
|
|
ui->functionsTreeView->resizeColumnToContents(1);
|
|
|
|
ui->functionsTreeView->resizeColumnToContents(2);
|
2017-03-29 10:18:37 +00:00
|
|
|
}
|
|
|
|
|
2017-04-28 13:09:40 +00:00
|
|
|
QTreeView *FunctionsWidget::getCurrentTreeView()
|
2017-03-29 10:18:37 +00:00
|
|
|
{
|
2017-04-28 13:09:40 +00:00
|
|
|
if (ui->tabWidget->currentIndex() == 0)
|
|
|
|
return ui->functionsTreeView;
|
|
|
|
else
|
|
|
|
return ui->nestedFunctionsTreeView;
|
|
|
|
}
|
2017-04-09 17:09:35 +00:00
|
|
|
|
2017-05-04 14:16:21 +00:00
|
|
|
void FunctionsWidget::functionsTreeView_doubleClicked(const QModelIndex &index)
|
2017-04-28 13:09:40 +00:00
|
|
|
{
|
|
|
|
FunctionDescription function = index.data(FunctionModel::FunctionDescriptionRole).value<FunctionDescription>();
|
2017-10-09 18:08:35 +00:00
|
|
|
CutterCore::getInstance()->seek(function.offset);
|
2017-03-29 10:18:37 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void FunctionsWidget::showFunctionsContextMenu(const QPoint &pt)
|
|
|
|
{
|
2017-04-28 13:09:40 +00:00
|
|
|
QTreeView *treeView = getCurrentTreeView();
|
|
|
|
|
2017-03-29 10:18:37 +00:00
|
|
|
// Set functions popup menu
|
2017-04-28 13:09:40 +00:00
|
|
|
QMenu *menu = new QMenu(ui->functionsTreeView);
|
2017-03-29 10:18:37 +00:00
|
|
|
menu->clear();
|
|
|
|
menu->addAction(ui->actionDisasAdd_comment);
|
|
|
|
menu->addAction(ui->actionFunctionsRename);
|
2017-12-14 21:54:57 +00:00
|
|
|
menu->addAction(ui->actionFunctionsUndefine);
|
2017-03-29 10:18:37 +00:00
|
|
|
menu->addSeparator();
|
|
|
|
menu->addAction(ui->action_References);
|
|
|
|
|
2017-04-28 13:09:40 +00:00
|
|
|
menu->exec(treeView->mapToGlobal(pt));
|
2017-03-29 10:18:37 +00:00
|
|
|
|
2017-04-28 13:09:40 +00:00
|
|
|
delete menu;
|
2017-03-29 10:18:37 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void FunctionsWidget::on_actionDisasAdd_comment_triggered()
|
|
|
|
{
|
2017-04-28 13:09:40 +00:00
|
|
|
// Get selected item in functions tree view
|
|
|
|
QTreeView *treeView = getCurrentTreeView();
|
|
|
|
FunctionDescription function = treeView->selectionModel()->currentIndex().data(FunctionModel::FunctionDescriptionRole).value<FunctionDescription>();
|
|
|
|
|
2017-03-29 10:18:37 +00:00
|
|
|
// Create dialog
|
2017-04-09 19:55:06 +00:00
|
|
|
CommentsDialog *c = new CommentsDialog(this);
|
2017-04-28 13:09:40 +00:00
|
|
|
|
2017-04-09 19:55:06 +00:00
|
|
|
if (c->exec())
|
|
|
|
{
|
2017-03-29 10:18:37 +00:00
|
|
|
// Get new function name
|
|
|
|
QString comment = c->getComment();
|
|
|
|
// Rename function in r2 core
|
2017-10-09 18:08:35 +00:00
|
|
|
CutterCore::getInstance()->setComment(function.offset, comment);
|
2017-03-29 10:18:37 +00:00
|
|
|
// Seek to new renamed function
|
2017-10-09 18:08:35 +00:00
|
|
|
CutterCore::getInstance()->seek(function.offset);
|
2017-03-29 10:18:37 +00:00
|
|
|
// TODO: Refresh functions tree widget
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void FunctionsWidget::on_actionFunctionsRename_triggered()
|
|
|
|
{
|
2017-04-28 13:09:40 +00:00
|
|
|
// Get selected item in functions tree view
|
|
|
|
QTreeView *treeView = getCurrentTreeView();
|
|
|
|
FunctionDescription function = treeView->selectionModel()->currentIndex().data(FunctionModel::FunctionDescriptionRole).value<FunctionDescription>();
|
|
|
|
|
2017-03-29 10:18:37 +00:00
|
|
|
// Create dialog
|
2017-04-09 19:55:06 +00:00
|
|
|
RenameDialog *r = new RenameDialog(this);
|
2017-04-28 13:09:40 +00:00
|
|
|
|
2017-03-29 10:18:37 +00:00
|
|
|
// Set function name in dialog
|
2017-11-26 16:53:05 +00:00
|
|
|
r->setName(function.name);
|
2017-03-29 10:18:37 +00:00
|
|
|
// If user accepted
|
2017-04-09 19:55:06 +00:00
|
|
|
if (r->exec())
|
|
|
|
{
|
2017-03-29 10:18:37 +00:00
|
|
|
// Get new function name
|
2017-11-26 16:53:05 +00:00
|
|
|
QString new_name = r->getName();
|
2017-03-29 10:18:37 +00:00
|
|
|
// Rename function in r2 core
|
2017-10-09 18:08:35 +00:00
|
|
|
CutterCore::getInstance()->renameFunction(function.name, new_name);
|
2017-04-28 13:09:40 +00:00
|
|
|
|
2017-03-29 10:18:37 +00:00
|
|
|
// Scroll to show the new name in functions tree widget
|
2017-04-28 13:09:40 +00:00
|
|
|
//
|
|
|
|
// QAbstractItemView::EnsureVisible
|
|
|
|
// QAbstractItemView::PositionAtTop
|
|
|
|
// QAbstractItemView::PositionAtBottom
|
|
|
|
// QAbstractItemView::PositionAtCenter
|
|
|
|
//
|
|
|
|
//ui->functionsTreeWidget->scrollToItem(selected_rows.first(), QAbstractItemView::PositionAtTop);
|
2017-03-29 10:18:37 +00:00
|
|
|
// Seek to new renamed function
|
2017-10-09 18:08:35 +00:00
|
|
|
CutterCore::getInstance()->seek(function.offset);
|
2017-03-29 10:18:37 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-12-14 21:54:57 +00:00
|
|
|
void FunctionsWidget::on_actionFunctionsUndefine_triggered()
|
|
|
|
{
|
|
|
|
QTreeView *treeView = getCurrentTreeView();
|
|
|
|
FunctionDescription function = treeView->selectionModel()->currentIndex().data(FunctionModel::FunctionDescriptionRole).value<FunctionDescription>();
|
|
|
|
Core()->delFunction(function.offset);
|
|
|
|
}
|
|
|
|
|
2017-03-29 10:18:37 +00:00
|
|
|
void FunctionsWidget::on_action_References_triggered()
|
|
|
|
{
|
2017-04-28 13:09:40 +00:00
|
|
|
// Get selected item in functions tree view
|
|
|
|
QTreeView *treeView = getCurrentTreeView();
|
|
|
|
FunctionDescription function = treeView->selectionModel()->currentIndex().data(FunctionModel::FunctionDescriptionRole).value<FunctionDescription>();
|
2017-10-10 10:17:05 +00:00
|
|
|
XrefsDialog *x = new XrefsDialog(this);
|
2017-06-08 22:40:43 +00:00
|
|
|
x->fillRefsForAddress(function.offset, function.name, true);
|
2017-03-29 10:18:37 +00:00
|
|
|
x->exec();
|
|
|
|
}
|
|
|
|
|
|
|
|
void FunctionsWidget::showTitleContextMenu(const QPoint &pt)
|
|
|
|
{
|
|
|
|
// Set functions popup menu
|
|
|
|
QMenu *menu = new QMenu(this);
|
|
|
|
menu->clear();
|
|
|
|
menu->addAction(ui->actionHorizontal);
|
|
|
|
menu->addAction(ui->actionVertical);
|
|
|
|
|
2017-04-09 19:55:06 +00:00
|
|
|
if (ui->tabWidget->currentIndex() == 0)
|
|
|
|
{
|
2017-03-29 10:18:37 +00:00
|
|
|
ui->actionHorizontal->setChecked(true);
|
|
|
|
ui->actionVertical->setChecked(false);
|
2017-04-09 19:55:06 +00:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2017-03-29 10:18:37 +00:00
|
|
|
ui->actionVertical->setChecked(true);
|
|
|
|
ui->actionHorizontal->setChecked(false);
|
|
|
|
}
|
|
|
|
|
|
|
|
this->setContextMenuPolicy(Qt::CustomContextMenu);
|
|
|
|
|
|
|
|
menu->exec(this->mapToGlobal(pt));
|
|
|
|
delete menu;
|
|
|
|
}
|
|
|
|
|
|
|
|
void FunctionsWidget::on_actionHorizontal_triggered()
|
|
|
|
{
|
|
|
|
ui->tabWidget->setCurrentIndex(0);
|
|
|
|
}
|
|
|
|
|
|
|
|
void FunctionsWidget::on_actionVertical_triggered()
|
|
|
|
{
|
|
|
|
ui->tabWidget->setCurrentIndex(1);
|
|
|
|
}
|
|
|
|
|
2017-04-10 10:25:33 +00:00
|
|
|
void FunctionsWidget::resizeEvent(QResizeEvent *event)
|
2017-04-09 19:55:06 +00:00
|
|
|
{
|
2017-04-13 15:51:58 +00:00
|
|
|
if (main->responsive && isVisible())
|
2017-04-09 19:55:06 +00:00
|
|
|
{
|
2017-04-10 10:25:33 +00:00
|
|
|
if (event->size().width() >= event->size().height())
|
|
|
|
{
|
|
|
|
// Set horizontal view (list)
|
|
|
|
on_actionHorizontal_triggered();
|
|
|
|
}
|
|
|
|
else
|
2017-04-09 19:55:06 +00:00
|
|
|
{
|
2017-04-10 10:25:33 +00:00
|
|
|
// Set vertical view (Tree)
|
|
|
|
on_actionVertical_triggered();
|
2017-03-29 10:18:37 +00:00
|
|
|
}
|
|
|
|
}
|
2017-04-10 10:25:33 +00:00
|
|
|
QDockWidget::resizeEvent(event);
|
2017-03-29 10:18:37 +00:00
|
|
|
}
|
2017-04-13 15:14:02 +00:00
|
|
|
|
|
|
|
void FunctionsWidget::setScrollMode()
|
|
|
|
{
|
2017-04-28 13:09:40 +00:00
|
|
|
qhelpers::setVerticalScrollMode(ui->functionsTreeView);
|
2017-12-20 15:40:46 +00:00
|
|
|
}
|