Merge branch 'feature/saveLoadItemsInFromFile' into develop

This commit is contained in:
2025-12-23 14:02:47 +01:00
10 changed files with 214 additions and 16 deletions

View File

@ -28,6 +28,7 @@ add_library(${TARGET_APP} STATIC
model/commands/insertrowscommand.h model/commands/insertrowscommand.cpp model/commands/insertrowscommand.h model/commands/insertrowscommand.cpp
model/commands/removerowscommand.h model/commands/removerowscommand.cpp model/commands/removerowscommand.h model/commands/removerowscommand.cpp
model/commands/edititemcommand.h model/commands/edititemcommand.cpp model/commands/edititemcommand.h model/commands/edititemcommand.cpp
data/filehandler.h data/filehandler.cpp
) )
include_directories(${CMAKE_CURRENT_BINARY_DIR}) include_directories(${CMAKE_CURRENT_BINARY_DIR})

53
data/filehandler.cpp Normal file
View File

@ -0,0 +1,53 @@
#include "filehandler.h"
#include <QDebug>
#include <QDir>
#include <QJsonDocument>
#include <QStandardPaths>
bool FileHandler::saveToFile(const QJsonDocument& doc, const QString& fileName) {
qDebug() << "saving file...";
QString path = QStandardPaths::standardLocations(QStandardPaths::AppDataLocation).at(0);
qDebug() << path;
QDir dir;
if (!dir.exists(path)) {
dir.mkpath(path);
}
// qDebug() << path + fileName;
const QString filePath = path + '/' + fileName;
QFile file(filePath);
if (!file.open(QIODevice::WriteOnly | QIODevice::Text)) {
qWarning() << "can't open file";
return false;
}
QTextStream out(&file);
out.setEncoding(QStringConverter::Utf8);
out << doc.toJson(QJsonDocument::Indented);
return true;
}
QByteArray FileHandler::loadJSONDataFromFile(const QString fileName) {
QByteArray jsonData;
QFile file;
QString path = QStandardPaths::standardLocations(QStandardPaths::AppDataLocation).at(0);
file.setFileName(path + "/" + fileName);
if (file.exists()) {
qDebug() << "File found, reading content...";
const bool successfulOpened = file.open(QIODevice::ReadOnly | QIODevice::Text);
if (successfulOpened) {
// TODO learn and decide on the differences between "readAll" and using
// streams
jsonData = file.readAll();
file.close();
} else {
qWarning() << "File could not be opened!";
}
} else {
qInfo() << "File not found. Returning empty result...";
}
return jsonData;
}
FileHandler::FileHandler() {}

17
data/filehandler.h Normal file
View File

@ -0,0 +1,17 @@
#ifndef FILEHANDLER_H
#define FILEHANDLER_H
class QJsonDocument;
class QString;
class QByteArray;
class FileHandler {
public:
static bool saveToFile(const QJsonDocument& doc, const QString& fileName);
static QByteArray loadJSONDataFromFile(const QString fileName);
private:
explicit FileHandler();
};
#endif // FILEHANDLER_H

View File

@ -51,7 +51,7 @@ QJsonArray JsonParser::extractItemArray(const QByteArray& jsonData, const QStrin
} else { } else {
QJsonObject rootObject = doc.object(); QJsonObject rootObject = doc.object();
itemArray = rootObject.value(QString("items")).toArray(); itemArray = rootObject.value(objectName).toArray();
} }
return itemArray; return itemArray;

View File

@ -3,6 +3,7 @@
#include <QCoreApplication> #include <QCoreApplication>
#include <QDateTime> #include <QDateTime>
#include <QDebug> #include <QDebug>
#include <QJsonDocument>
#include <QProcess> #include <QProcess>
#include <QSettings> #include <QSettings>
#include <QString> #include <QString>
@ -10,6 +11,7 @@
#include "../../ApplicationConfig.h" #include "../../ApplicationConfig.h"
#include "CoreConfig.h" #include "CoreConfig.h"
#include "constants.h" #include "constants.h"
#include "data/filehandler.h"
#include "model/tablemodel.h" #include "model/tablemodel.h"
#include <QtGui/QUndoStack> #include <QtGui/QUndoStack>
@ -19,6 +21,15 @@ using namespace std;
GenericCore::GenericCore() { GenericCore::GenericCore() {
qDebug() << "Creating core..."; qDebug() << "Creating core...";
QCoreApplication::setOrganizationName("Working-Copy Collective");
QCoreApplication::setOrganizationDomain("working-copy.org");
#ifdef QT_DEBUG
QCoreApplication::setApplicationName(QString(APPLICATION_NAME) + "-dev");
#else
QCoreApplication::setApplicationName(QString(APPLICATION_NAME));
#endif
// TODO let the model own its undo stack (& use TableModel::getUndoStack() if necessary) // TODO let the model own its undo stack (& use TableModel::getUndoStack() if necessary)
m_modelUndoStack = new QUndoStack(this); m_modelUndoStack = new QUndoStack(this);
@ -79,9 +90,51 @@ QUndoStack* GenericCore::getModelUndoStack() const { return m_modelUndoStack; }
std::shared_ptr<TableModel> GenericCore::getModel() const { return m_mainModel; } std::shared_ptr<TableModel> GenericCore::getModel() const { return m_mainModel; }
/**
* Save items to default file (in standard location).
* @brief GenericCore::saveItems Saves item fo file.
*/
void GenericCore::saveItems() {
qDebug() << "saving items...";
const QJsonDocument doc = m_mainModel->getAllItemsAsJsonDoc();
const bool successfulSave = FileHandler::saveToFile(doc, "items.json");
if (successfulSave) {
// QStringList completedTaskStrings = m_model->completedTasks();
// appendCompletedTasksToFile(completedTaskStrings, "completed.txt");
m_modelUndoStack->setClean();
emit displayStatusMessage(QString("Items saved."));
} else {
emit displayStatusMessage(QString("Error: Items couldn't be saved."));
}
}
void GenericCore::setupModels() { void GenericCore::setupModels() {
m_mainModel = make_shared<TableModel>(m_modelUndoStack, this); m_mainModel = make_shared<TableModel>(m_modelUndoStack, this);
// TODO add QAbstractItemModelTester // TODO add QAbstractItemModelTester
initModelData();
}
/**
* Initializing model with data. Tries to read items from default file. Generating example items as
* fallback.
* @brief GenericCore::initModelData
*/
void GenericCore::initModelData() {
qInfo() << "Trying to read model data from file...";
const QByteArray jsonDoc = FileHandler::loadJSONDataFromFile("items.json");
// qDebug() << "jsonDoc:" << jsonDoc;
// TODO decide on lack of file(s) (config, data) if example items should be generated
// (see welcome wizard)
if (jsonDoc.isEmpty()) {
qDebug() << "No item content in file. Generating example items...";
const QByteArray exampleItems = m_mainModel->generateExampleItems();
m_mainModel->insertItems(0, exampleItems);
} else {
qDebug() << "Item in file found.";
m_mainModel->insertItems(0, jsonDoc);
}
m_modelUndoStack->clear();
} }
QString GenericCore::getMaintenanceToolFilePath() const { QString GenericCore::getMaintenanceToolFilePath() const {

View File

@ -25,6 +25,8 @@ class GenericCore : public QObject {
QUndoStack* getModelUndoStack() const; QUndoStack* getModelUndoStack() const;
std::shared_ptr<TableModel> getModel() const; std::shared_ptr<TableModel> getModel() const;
void saveItems();
signals: signals:
void displayStatusMessage(QString message); void displayStatusMessage(QString message);
@ -33,6 +35,7 @@ class GenericCore : public QObject {
std::shared_ptr<TableModel> m_mainModel; std::shared_ptr<TableModel> m_mainModel;
void setupModels(); void setupModels();
void initModelData();
QString getMaintenanceToolFilePath() const; QString getMaintenanceToolFilePath() const;
}; };

View File

@ -1,5 +1,10 @@
#include "modelitem.h" #include "modelitem.h"
#include "tablemodel.h"
#include <QJsonObject>
#include <QJsonValue>
ModelItem::ModelItem(const QHash<int, QVariant> values) ModelItem::ModelItem(const QHash<int, QVariant> values)
: m_values(values) {} : m_values(values) {}
@ -38,3 +43,27 @@ bool ModelItem::setItemData(const QMap<int, QVariant>& changedValues) {
return valueChanged; return valueChanged;
} }
QJsonObject ModelItem::toJsonObject() const {
QJsonObject itemObject;
// itemObject.insert("uuid", m_uuid.toString());
// itemObject.insert("entryDateUTC", m_entryDateUTC.toString(Qt::ISODate));
itemObject.insert(TableModel::ROLE_NAMES.value(TableModel::NameRole),
data(TableModel::NameRole).toString());
itemObject.insert(TableModel::ROLE_NAMES.value(TableModel::DescriptionRole),
data(TableModel::DescriptionRole).toString());
itemObject.insert(TableModel::ROLE_NAMES.value(TableModel::InfoRole),
data(TableModel::InfoRole).toString());
itemObject.insert(TableModel::ROLE_NAMES.value(TableModel::AmountRole),
data(TableModel::AmountRole).toInt());
itemObject.insert(TableModel::ROLE_NAMES.value(TableModel::FactorRole),
data(TableModel::FactorRole).toReal());
// if (m_modifiedDateUTC.isValid()) {
// itemObject.insert("modifiedDateUTC", m_modifiedDateUTC.toString(Qt::ISODate));
// }
// if (m_endDateUTC.isValid()) {
// itemObject.insert("endDateUTC", m_endDateUTC.toString(Qt::ISODate));
// }
return itemObject;
}

View File

@ -12,6 +12,9 @@ class ModelItem {
// TODO change return value to list of changed roles // TODO change return value to list of changed roles
bool setItemData(const QMap<int, QVariant>& changedValues); bool setItemData(const QMap<int, QVariant>& changedValues);
// QString toString() const;
QJsonObject toJsonObject() const;
private: private:
QHash<int, QVariant> m_values; QHash<int, QVariant> m_values;
}; };

View File

@ -6,6 +6,10 @@
#include "commands/removerowscommand.h" #include "commands/removerowscommand.h"
#include "modelitem.h" #include "modelitem.h"
#include <QJsonArray>
#include <QJsonDocument>
#include <QJsonObject>
QHash<int, QByteArray> TableModel::ROLE_NAMES = {{NameRole, "Name"}, QHash<int, QByteArray> TableModel::ROLE_NAMES = {{NameRole, "Name"},
{DescriptionRole, "Description"}, {DescriptionRole, "Description"},
{InfoRole, "Info"}, {InfoRole, "Info"},
@ -13,21 +17,35 @@ QHash<int, QByteArray> TableModel::ROLE_NAMES = {{NameRole, "Name"},
{FactorRole, "Factor"}}; {FactorRole, "Factor"}};
QList<QString> TableModel::intColumns = {"Amount", "Factor"}; QList<QString> TableModel::intColumns = {"Amount", "Factor"};
QByteArray TableModel::generateExampleItems() {
QJsonDocument doc = QJsonDocument();
QJsonObject rootObject;
QJsonArray array;
for (int row = 0; row < 5; ++row) {
QJsonObject itemObject;
// itemObject.insert("uuid", m_uuid.toString());
// itemObject.insert("entryDateUTC", m_entryDateUTC.toString(Qt::ISODate));
itemObject.insert(TableModel::ROLE_NAMES.value(TableModel::NameRole),
QString("Item %1").arg(row));
itemObject.insert(TableModel::ROLE_NAMES.value(TableModel::DescriptionRole),
QString("This is item %1").arg(row));
itemObject.insert(TableModel::ROLE_NAMES.value(TableModel::InfoRole),
QString("Info of item %1").arg(row));
itemObject.insert(TableModel::ROLE_NAMES.value(TableModel::AmountRole), row);
itemObject.insert(TableModel::ROLE_NAMES.value(TableModel::FactorRole), row * 1.1);
array.append(itemObject);
}
rootObject.insert("items", array);
doc.setObject(rootObject);
return doc.toJson();
}
TableModel::TableModel(QUndoStack* undoStack, QObject* parent) TableModel::TableModel(QUndoStack* undoStack, QObject* parent)
: QAbstractTableModel{parent} : QAbstractTableModel{parent}
, m_undoStack(undoStack) { , m_undoStack(undoStack) {}
for (int row = 0; row < 5; ++row) {
QHash<int, QVariant> values;
values[NameRole] = QString("Item %1").arg(row);
values[DescriptionRole] = QString("This is item %1").arg(row);
values[InfoRole] = QString("Info of item %1").arg(row);
values[AmountRole] = row;
values[FactorRole] = row * 1.1;
shared_ptr<ModelItem> item = make_unique<ModelItem>(values);
m_items.append(std::move(item));
}
}
Qt::ItemFlags TableModel::flags(const QModelIndex& index) const { Qt::ItemFlags TableModel::flags(const QModelIndex& index) const {
return Qt::ItemIsEditable | QAbstractTableModel::flags(index); return Qt::ItemIsEditable | QAbstractTableModel::flags(index);
@ -111,6 +129,21 @@ bool TableModel::setItemData(const QModelIndex& index, const QMap<int, QVariant>
return false; return false;
} }
QJsonDocument TableModel::getAllItemsAsJsonDoc() const {
QJsonDocument doc = QJsonDocument();
QJsonObject rootObject;
QJsonArray array;
foreach (shared_ptr<ModelItem> item, m_items) {
QJsonObject itemObject = item->toJsonObject();
array.append(itemObject);
}
rootObject.insert("items", array);
doc.setObject(rootObject);
return doc;
}
bool TableModel::removeRows(int firstRow, int nRows, const QModelIndex& parentIndex) { bool TableModel::removeRows(int firstRow, int nRows, const QModelIndex& parentIndex) {
if (parentIndex != QModelIndex()) { if (parentIndex != QModelIndex()) {
qWarning() << "Removing of child rows is not supported yet!"; qWarning() << "Removing of child rows is not supported yet!";
@ -143,7 +176,7 @@ void TableModel::insertItems(int startPosition,
startPosition = m_items.size(); startPosition = m_items.size();
} }
QList<QHash<int, QVariant>> valueList = JsonParser::toItemValuesList(jsonDoc); QList<QHash<int, QVariant>> valueList = JsonParser::toItemValuesList(jsonDoc, "items");
InsertRowsCommand* insertCommand = new InsertRowsCommand(this, startPosition, valueList); InsertRowsCommand* insertCommand = new InsertRowsCommand(this, startPosition, valueList);
m_undoStack->push(insertCommand); m_undoStack->push(insertCommand);

View File

@ -19,6 +19,7 @@ class TableModel : public QAbstractTableModel {
enum UserRoles { NameRole = Qt::UserRole + 1, DescriptionRole, InfoRole, AmountRole, FactorRole }; enum UserRoles { NameRole = Qt::UserRole + 1, DescriptionRole, InfoRole, AmountRole, FactorRole };
static QHash<int, QByteArray> ROLE_NAMES; static QHash<int, QByteArray> ROLE_NAMES;
static QList<QString> intColumns; static QList<QString> intColumns;
static QByteArray generateExampleItems();
explicit TableModel(QUndoStack* undoStack, QObject* parent = nullptr); explicit TableModel(QUndoStack* undoStack, QObject* parent = nullptr);
@ -34,15 +35,20 @@ class TableModel : public QAbstractTableModel {
bool setData(const QModelIndex& index, const QVariant& value, int role) override; bool setData(const QModelIndex& index, const QVariant& value, int role) override;
bool setItemData(const QModelIndex& index, const QMap<int, QVariant>& roles) override; bool setItemData(const QModelIndex& index, const QMap<int, QVariant>& roles) override;
QJsonDocument getAllItemsAsJsonDoc() const;
public slots: public slots:
// bool insertRows(int position, int rows, const QModelIndex& parentIndex = QModelIndex()) // bool insertRows(int position, int rows, const QModelIndex& parentIndex = QModelIndex())
// override; // override;
bool removeRows(int firstRow, int nRows, const QModelIndex& parentIndex = QModelIndex()) override; bool removeRows(int firstRow, int nRows, const QModelIndex& parentIndex = QModelIndex()) override;
void appendItems(const QByteArray& jsonDoc); void appendItems(const QByteArray& jsonDoc);
void insertItems(int startPosition, const QByteArray& jsonDoc, const QModelIndex& parentIndex); void insertItems(int startPosition,
const QByteArray& jsonDoc,
const QModelIndex& parentIndex = QModelIndex());
private: private:
/// *** members *** /// *** members ***
// TODO shared_ptr -> unique_ptr
QList<shared_ptr<ModelItem>> m_items; QList<shared_ptr<ModelItem>> m_items;
QUndoStack* m_undoStack; QUndoStack* m_undoStack;