2023-07-31 14:21:14 +02:00
|
|
|
#include "pagesmodel.h"
|
|
|
|
|
|
|
|
#include <QDir>
|
2023-09-27 12:56:10 +02:00
|
|
|
#include <QFileSystemWatcher>
|
|
|
|
#include <QRegularExpression>
|
2023-07-31 14:21:14 +02:00
|
|
|
|
|
|
|
namespace {
|
|
|
|
const auto categoryUncategorized QStringLiteral("Uncategorized");
|
|
|
|
}
|
|
|
|
|
|
|
|
PagesModel::PagesModel(const QString &path, QObject *parent)
|
2023-09-27 12:56:10 +02:00
|
|
|
: QAbstractListModel{parent}, m_path{path},
|
|
|
|
fsWatcher(new QFileSystemWatcher(this))
|
2023-07-31 14:21:14 +02:00
|
|
|
{
|
2023-09-27 12:56:10 +02:00
|
|
|
m_items = load();
|
|
|
|
readMetadata(m_items);
|
|
|
|
|
|
|
|
fsWatcher->addPath(path);
|
|
|
|
|
|
|
|
connect(fsWatcher, &QFileSystemWatcher::directoryChanged,
|
|
|
|
this, &PagesModel::reload);
|
|
|
|
}
|
2023-07-31 14:21:14 +02:00
|
|
|
|
2023-09-27 13:00:27 +02:00
|
|
|
QList<PagesModelItem> PagesModel::load() const {
|
2023-07-31 14:21:14 +02:00
|
|
|
static QRegularExpression fileNameRegex(
|
|
|
|
QRegularExpression::anchoredPattern("(.*)Page\\.qml"));
|
2023-09-27 12:56:10 +02:00
|
|
|
|
|
|
|
QDir dir(m_path);
|
|
|
|
dir.setFilter(QDir::Files);
|
2023-07-31 14:21:14 +02:00
|
|
|
|
|
|
|
const QFileInfoList files = dir.entryInfoList();
|
2023-09-27 12:56:10 +02:00
|
|
|
QList<PagesModelItem> items;
|
2023-07-31 14:21:14 +02:00
|
|
|
|
2023-09-27 12:56:10 +02:00
|
|
|
std::for_each(files.begin(), files.end(), [this, &items] (auto &fileInfo) {
|
2023-07-31 14:21:14 +02:00
|
|
|
QString fileName = fileInfo.fileName();
|
|
|
|
QRegularExpressionMatch fileNameMatch = fileNameRegex.match(fileName);
|
|
|
|
|
|
|
|
if (!fileNameMatch.hasMatch())
|
|
|
|
return;
|
|
|
|
|
2023-09-27 12:56:10 +02:00
|
|
|
PagesModelItem item;
|
|
|
|
item.path = fileInfo.filePath();
|
|
|
|
item.title = fileNameMatch.captured(1);
|
|
|
|
item.lastModified = fileInfo.lastModified();
|
2023-07-31 14:21:14 +02:00
|
|
|
|
2023-09-27 12:56:10 +02:00
|
|
|
items << item;
|
|
|
|
});
|
|
|
|
|
|
|
|
return items;
|
|
|
|
}
|
|
|
|
|
|
|
|
void PagesModel::readMetadata(PagesModelItem& item) {
|
|
|
|
static QRegularExpression categoryRegex(
|
|
|
|
"^//(\\s)*category:(.+)$", QRegularExpression::MultilineOption);
|
|
|
|
|
|
|
|
QFile file(item.path);
|
|
|
|
file.open(QIODevice::ReadOnly);
|
|
|
|
QByteArray content = file.readAll();
|
|
|
|
|
|
|
|
QRegularExpressionMatch categoryMatch = categoryRegex.match(content);
|
|
|
|
QString category = categoryMatch.hasMatch()
|
|
|
|
? categoryMatch.captured(2).trimmed() : categoryUncategorized;
|
|
|
|
|
|
|
|
item.category = category;
|
|
|
|
}
|
2023-07-31 14:21:14 +02:00
|
|
|
|
2023-09-27 12:56:10 +02:00
|
|
|
void PagesModel::readMetadata(QList<PagesModelItem> &items) {
|
|
|
|
std::for_each(items.begin(), items.end(), [](auto&item) {
|
|
|
|
readMetadata(item);
|
2023-07-31 14:21:14 +02:00
|
|
|
});
|
|
|
|
}
|
|
|
|
|
2023-09-27 12:56:10 +02:00
|
|
|
void PagesModel::reload() {
|
2023-09-27 13:00:27 +02:00
|
|
|
const QList<PagesModelItem> currentItems = load();
|
|
|
|
std::unordered_map<QString, PagesModelItem> mapping;
|
2023-09-27 12:56:10 +02:00
|
|
|
|
|
|
|
for (const PagesModelItem &item : qAsConst(m_items))
|
|
|
|
mapping[item.title] = item;
|
|
|
|
|
|
|
|
std::vector<PagesModelItem> newItems;
|
|
|
|
std::vector<PagesModelItem> changedItems;
|
|
|
|
std::vector<PagesModelItem> removedItems;
|
|
|
|
|
2023-09-27 13:00:27 +02:00
|
|
|
for (const auto &item : currentItems) {
|
2023-09-27 12:56:10 +02:00
|
|
|
auto it = mapping.find(item.title);
|
|
|
|
|
|
|
|
if (it == mapping.end()) {
|
|
|
|
newItems.push_back(item);
|
|
|
|
} else {
|
|
|
|
if (item.lastModified != it->second.lastModified)
|
|
|
|
changedItems.push_back(item);
|
|
|
|
|
|
|
|
mapping.erase(it);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2023-09-27 13:00:27 +02:00
|
|
|
for (const auto& [key, value] : mapping)
|
2023-09-27 12:56:10 +02:00
|
|
|
removedItems.push_back(value);
|
|
|
|
|
|
|
|
for (auto& item : removedItems) {
|
|
|
|
|
|
|
|
auto it = std::find_if(m_items.begin(), m_items.end(), [&item](auto& it){
|
|
|
|
return it.title == item.title;
|
|
|
|
});
|
|
|
|
|
|
|
|
auto index = std::distance(m_items.begin(), it);
|
|
|
|
|
|
|
|
beginRemoveRows(QModelIndex{}, index, index);
|
|
|
|
m_items.removeAt(index);
|
|
|
|
endRemoveRows();
|
|
|
|
}
|
|
|
|
|
|
|
|
if (newItems.size()) {
|
|
|
|
beginInsertRows(QModelIndex{}, rowCount(), rowCount() + newItems.size() - 1);
|
|
|
|
|
|
|
|
for (auto& item : newItems) {
|
|
|
|
readMetadata(item);
|
|
|
|
m_items << item;
|
|
|
|
}
|
|
|
|
|
|
|
|
endInsertRows();
|
|
|
|
}
|
|
|
|
|
|
|
|
for (auto& item : changedItems) {
|
|
|
|
auto it = std::find_if(m_items.begin(), m_items.end(), [&item](auto& it){
|
|
|
|
return it.title == item.title;
|
|
|
|
});
|
|
|
|
|
|
|
|
auto index = std::distance(m_items.begin(), it);
|
|
|
|
const auto& previous = *it;
|
|
|
|
readMetadata(item);
|
|
|
|
|
|
|
|
if (previous.category != item.category) {
|
|
|
|
// For simplicity category change is handled by removing and
|
|
|
|
// adding item. In the future it can be changed to regular dataChanged
|
|
|
|
// event and handled properly in upstream models like SectionSDecoratorModel.
|
|
|
|
beginRemoveRows(QModelIndex{}, index, index);
|
|
|
|
m_items.removeAt(index);
|
|
|
|
endRemoveRows();
|
|
|
|
|
|
|
|
beginInsertRows(QModelIndex{}, rowCount(), rowCount());
|
|
|
|
m_items << item;
|
|
|
|
endInsertRows();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2023-07-31 14:21:14 +02:00
|
|
|
QHash<int, QByteArray> PagesModel::roleNames() const
|
|
|
|
{
|
|
|
|
static const QHash<int,QByteArray> roles {
|
|
|
|
{ TitleRole, QByteArrayLiteral("title") },
|
|
|
|
{ CategoryRole, QByteArrayLiteral("category") }
|
|
|
|
};
|
|
|
|
|
|
|
|
return roles;
|
|
|
|
}
|
|
|
|
|
|
|
|
int PagesModel::rowCount(const QModelIndex &parent) const
|
|
|
|
{
|
|
|
|
return m_items.length();
|
|
|
|
}
|
|
|
|
|
|
|
|
QVariant PagesModel::data(const QModelIndex &index, int role) const
|
|
|
|
{
|
|
|
|
if (!checkIndex(index, CheckIndexOption::IndexIsValid))
|
|
|
|
return {};
|
|
|
|
|
|
|
|
if (role == TitleRole)
|
|
|
|
return m_items.at(index.row()).title;
|
|
|
|
if (role == CategoryRole)
|
|
|
|
return m_items.at(index.row()).category;
|
|
|
|
|
|
|
|
return {};
|
|
|
|
}
|