Display the MemSpec in a TreeView

This commit is contained in:
2021-09-07 09:20:54 +02:00
parent b89a881ffa
commit 6d99853f3c
5 changed files with 211 additions and 9 deletions

View File

@@ -49,14 +49,12 @@ McConfigModel::McConfigModel(const TraceDB &traceFile, QObject *parent) : QAbstr
QString mcConfigJson = sqlQuery.value(0).toString();
parseJson(mcConfigJson);
qDebug() << roleNames();
}
void McConfigModel::parseJson(const QString &jsonString)
{
QJsonDocument jsonDocument = QJsonDocument::fromJson(jsonString.toUtf8());
QJsonObject mcConfigJson = jsonDocument["mcconfig"].toObject();
QJsonObject mcConfigJson = jsonDocument.object()["mcconfig"].toObject();
for (auto key : mcConfigJson.keys())
{
@@ -124,3 +122,152 @@ QVariant McConfigModel::headerData(int section, Qt::Orientation orientation, int
return QVariant();
}
MemSpecModel::MemSpecModel(const TraceDB &traceFile, QObject *parent) : QAbstractItemModel(parent)
{
QSqlDatabase db = traceFile.getDatabase();
QString query = "SELECT Memspec FROM GeneralInfo";
QSqlQuery sqlQuery = db.exec(query);
// The whole configuration is stored in a single cell.
sqlQuery.next();
QString memSpecJson = sqlQuery.value(0).toString();
parseJson(memSpecJson);
}
int MemSpecModel::Node::getRow() const
{
if (!parent)
return 0;
const auto &siblings = parent->children;
const auto siblingsIt = std::find_if(siblings.begin(), siblings.end(), [this](const std::unique_ptr<Node> &node){
return node.get() == this;
});
Q_ASSERT(siblingsIt != siblings.end());
return std::distance(siblings.begin(), siblingsIt);
}
void MemSpecModel::parseJson(const QString &jsonString)
{
QJsonDocument jsonDocument = QJsonDocument::fromJson(jsonString.toUtf8());
QJsonObject memSpecJson = jsonDocument.object()["memspec"].toObject();
std::function<void(const QJsonObject &, std::unique_ptr<Node> &)> addNodes;
addNodes = [&addNodes](const QJsonObject &obj, std::unique_ptr<Node> &parentNode)
{
for (auto key : obj.keys())
{
QJsonValue currentValue = obj.value(key);
QString value;
if (currentValue.isDouble())
{
value = QString::number(currentValue.toDouble());
}
else if (currentValue.isString())
{
value = currentValue.toString();
}
std::unique_ptr<Node> node = std::unique_ptr<Node>(new Node(std::make_pair(key, value), parentNode.get()));
addNodes(obj[key].toObject(), node);
parentNode->children.push_back(std::move(node));
}
};
addNodes(memSpecJson, rootNode);
}
int MemSpecModel::rowCount(const QModelIndex &parent) const
{
if (parent.column() > 0)
return 0;
const Node *parentNode;
if (!parent.isValid())
parentNode = rootNode.get();
else
parentNode = static_cast<const Node *>(parent.internalPointer());
return parentNode->childCount();
}
int MemSpecModel::columnCount(const QModelIndex &parent) const
{
Q_UNUSED(parent)
return 2;
}
QVariant MemSpecModel::data(const QModelIndex &index, int role) const
{
if (!index.isValid())
return QVariant();
if (role != Qt::DisplayRole && role != Qt::ToolTipRole)
return QVariant();
auto *node = static_cast<const Node *>(index.internalPointer());
if (index.column() == 0)
return QVariant(node->data.first);
else
return QVariant(node->data.second);
}
QVariant MemSpecModel::headerData(int section, Qt::Orientation orientation, int role) const
{
if (role != Qt::DisplayRole)
return QVariant();
if (orientation == Qt::Horizontal)
{
switch (section) {
case 0:
return "Field";
case 1:
return "Value";
default:
break;
}
}
return QVariant();
}
QModelIndex MemSpecModel::index(int row, int column, const QModelIndex &parent) const
{
if (!hasIndex(row, column, parent))
return QModelIndex();
const Node *parentNode;
if (!parent.isValid())
parentNode = rootNode.get();
else
parentNode = static_cast<const Node *>(parent.internalPointer());
const Node *node = parentNode->children[row].get();
return createIndex(row, column, const_cast<Node *>(node));
}
QModelIndex MemSpecModel::parent(const QModelIndex &index) const
{
if (!index.isValid())
return QModelIndex();
const Node *childNode = static_cast<const Node *>(index.internalPointer());
const Node *parentNode = childNode->parent;
if (!parentNode)
return QModelIndex();
return createIndex(parentNode->getRow(), 0, const_cast<Node *>(parentNode));
}

View File

@@ -44,13 +44,16 @@
class McConfigModel : public QAbstractTableModel
{
Q_OBJECT
public:
McConfigModel(const TraceDB &traceFile, QObject *parent = nullptr);
explicit McConfigModel(const TraceDB &traceFile, QObject *parent = nullptr);
~McConfigModel() {}
protected:
int rowCount(const QModelIndex &parent = QModelIndex()) const override;
int columnCount(const QModelIndex &parent = QModelIndex()) const override;
QVariant data(const QModelIndex &index, int role = Qt::DisplayRole) const override;
QVariant headerData(int section, Qt::Orientation orientation, int role = Qt::DisplayRole) const override;
@@ -65,9 +68,52 @@ private:
std::vector<std::pair<QString, QString>> entries;
};
class MemSpecsModel : public QAbstractItemModel
class MemSpecModel : public QAbstractItemModel
{
Q_OBJECT
public:
explicit MemSpecModel(const TraceDB &traceFile, QObject *parent = nullptr);
~MemSpecModel() {}
protected:
int rowCount(const QModelIndex &parent = QModelIndex()) const override;
int columnCount(const QModelIndex &parent) const override;
QVariant data(const QModelIndex &index, int role = Qt::DisplayRole) const override;
QVariant headerData(int section, Qt::Orientation orientation, int role = Qt::DisplayRole) const override;
QModelIndex index(int row, int column, const QModelIndex &parent) const override;
QModelIndex parent(const QModelIndex &index) const override;
private:
/**
* Parses the json file and adds json entries to the entries vector.
* In case of failure, nothing is added and therefore the model
* will stay empty.
*/
void parseJson(const QString &jsonString);
struct Node
{
using NodeData = std::pair<QString, QString>;
Node() {}
Node(NodeData data, const Node *parent) : data(data), parent(parent) {}
/**
* Gets the row relative to its parent.
*/
int getRow() const;
int childCount() const { return children.size(); }
NodeData data;
const Node *parent = nullptr;
std::vector<std::unique_ptr<Node>> children;
};
std::unique_ptr<Node> rootNode = std::unique_ptr<Node>(new Node);
};
#endif // CONFIGMODELS_H

View File

@@ -65,7 +65,9 @@
TraceFileTab::TraceFileTab(QWidget *parent, const QString &path) :
QWidget(parent), ui(new Ui::TraceFileTab), navigator(new TraceNavigator(path, this)),
mcConfigModel(new McConfigModel(navigator->TraceFile(), this)), savingChangesToDB(false)
mcConfigModel(new McConfigModel(navigator->TraceFile(), this)),
memSpecModel(new MemSpecModel(navigator->TraceFile(), this)),
savingChangesToDB(false)
{
ui->setupUi(this);
this->path = path;
@@ -84,6 +86,9 @@ TraceFileTab::TraceFileTab(QWidget *parent, const QString &path) :
ui->mcConfigView->setModel(mcConfigModel);
ui->mcConfigView->horizontalHeader()->setSectionResizeMode(QHeaderView::ResizeToContents);
ui->memSpecView->setModel(memSpecModel);
ui->memSpecView->header()->setSectionResizeMode(QHeaderView::ResizeToContents);
tracefileChanged();
}

View File

@@ -77,7 +77,7 @@ private:
QFileSystemWatcher *fileWatcher;
QAbstractItemModel *mcConfigModel;
//QAbstractItemModel *memSpecModel;
QAbstractItemModel *memSpecModel;
void setUpQueryEditor(QString path);
bool savingChangesToDB;

View File

@@ -350,7 +350,7 @@
</widget>
<widget class="QWidget" name="tabSimConfig">
<attribute name="title">
<string>Simulation Configuration</string>
<string>Configuration</string>
</attribute>
<layout class="QVBoxLayout" name="verticalLayout">
<item>
@@ -373,7 +373,11 @@
</widget>
</item>
<item>
<widget class="QTreeView" name="memSpecsView"/>
<widget class="QTreeView" name="memSpecView">
<property name="uniformRowHeights">
<bool>true</bool>
</property>
</widget>
</item>
</layout>
</widget>