PrismLauncher/launcher/SeparatorPrefixTree.h

231 lines
7.0 KiB
C
Raw Normal View History

#pragma once
#include <QMap>
#include <QString>
#include <QStringList>
template <char Tseparator>
class SeparatorPrefixTree {
public:
SeparatorPrefixTree(QStringList paths) { insert(paths); }
SeparatorPrefixTree(bool contained = false) { m_contained = contained; }
2018-07-15 14:51:05 +02:00
void insert(QStringList paths)
{
for (auto& path : paths) {
2018-07-15 14:51:05 +02:00
insert(path);
}
}
2018-07-15 14:51:05 +02:00
/// insert an exact path into the tree
SeparatorPrefixTree& insert(QString path)
2018-07-15 14:51:05 +02:00
{
auto sepIndex = path.indexOf(Tseparator);
if (sepIndex == -1) {
2018-07-15 14:51:05 +02:00
children[path] = SeparatorPrefixTree(true);
return children[path];
} else {
2018-07-15 14:51:05 +02:00
auto prefix = path.left(sepIndex);
if (!children.contains(prefix)) {
2018-07-15 14:51:05 +02:00
children[prefix] = SeparatorPrefixTree(false);
}
return children[prefix].insert(path.mid(sepIndex + 1));
}
}
2018-07-15 14:51:05 +02:00
/// is the path fully contained in the tree?
bool contains(QString path) const
{
auto node = find(path);
return node != nullptr;
}
2018-07-15 14:51:05 +02:00
/// does the tree cover a path? That means the prefix of the path is contained in the tree
bool covers(QString path) const
{
// if we found some valid node, it's good enough. the tree covers the path
if (m_contained) {
2018-07-15 14:51:05 +02:00
return true;
}
auto sepIndex = path.indexOf(Tseparator);
if (sepIndex == -1) {
2018-07-15 14:51:05 +02:00
auto found = children.find(path);
if (found == children.end()) {
2018-07-15 14:51:05 +02:00
return false;
}
return (*found).covers(QString());
} else {
2018-07-15 14:51:05 +02:00
auto prefix = path.left(sepIndex);
auto found = children.find(prefix);
if (found == children.end()) {
2018-07-15 14:51:05 +02:00
return false;
}
return (*found).covers(path.mid(sepIndex + 1));
}
}
2018-07-15 14:51:05 +02:00
/// return the contained path that covers the path specified
QString cover(QString path) const
{
// if we found some valid node, it's good enough. the tree covers the path
if (m_contained) {
2018-07-15 14:51:05 +02:00
return QString("");
}
auto sepIndex = path.indexOf(Tseparator);
if (sepIndex == -1) {
2018-07-15 14:51:05 +02:00
auto found = children.find(path);
if (found == children.end()) {
2018-07-15 14:51:05 +02:00
return QString();
}
auto nested = (*found).cover(QString());
if (nested.isNull()) {
2018-07-15 14:51:05 +02:00
return nested;
}
if (nested.isEmpty())
2018-07-15 14:51:05 +02:00
return path;
return path + Tseparator + nested;
} else {
2018-07-15 14:51:05 +02:00
auto prefix = path.left(sepIndex);
auto found = children.find(prefix);
if (found == children.end()) {
2018-07-15 14:51:05 +02:00
return QString();
}
auto nested = (*found).cover(path.mid(sepIndex + 1));
if (nested.isNull()) {
2018-07-15 14:51:05 +02:00
return nested;
}
if (nested.isEmpty())
2018-07-15 14:51:05 +02:00
return prefix;
return prefix + Tseparator + nested;
}
}
2018-07-15 14:51:05 +02:00
/// Does the path-specified node exist in the tree? It does not have to be contained.
bool exists(QString path) const
{
auto sepIndex = path.indexOf(Tseparator);
if (sepIndex == -1) {
2018-07-15 14:51:05 +02:00
auto found = children.find(path);
if (found == children.end()) {
2018-07-15 14:51:05 +02:00
return false;
}
return true;
} else {
2018-07-15 14:51:05 +02:00
auto prefix = path.left(sepIndex);
auto found = children.find(prefix);
if (found == children.end()) {
2018-07-15 14:51:05 +02:00
return false;
}
return (*found).exists(path.mid(sepIndex + 1));
}
}
2018-07-15 14:51:05 +02:00
/// find a node in the tree by name
const SeparatorPrefixTree* find(QString path) const
2018-07-15 14:51:05 +02:00
{
auto sepIndex = path.indexOf(Tseparator);
if (sepIndex == -1) {
2018-07-15 14:51:05 +02:00
auto found = children.find(path);
if (found == children.end()) {
2018-07-15 14:51:05 +02:00
return nullptr;
}
return &(*found);
} else {
2018-07-15 14:51:05 +02:00
auto prefix = path.left(sepIndex);
auto found = children.find(prefix);
if (found == children.end()) {
2018-07-15 14:51:05 +02:00
return nullptr;
}
return (*found).find(path.mid(sepIndex + 1));
}
}
2018-07-15 14:51:05 +02:00
/// is this a leaf node?
bool leaf() const { return children.isEmpty(); }
2018-07-15 14:51:05 +02:00
/// is this node actually contained in the tree, or is it purely structural?
bool contained() const { return m_contained; }
2018-07-15 14:51:05 +02:00
/// Remove a path from the tree
bool remove(QString path) { return removeInternal(path) != Failed; }
2018-07-15 14:51:05 +02:00
/// Clear all children of this node tree node
void clear() { children.clear(); }
2018-07-15 14:51:05 +02:00
QStringList toStringList() const
{
QStringList collected;
// collecting these is more expensive.
auto iter = children.begin();
while (iter != children.end()) {
2018-07-15 14:51:05 +02:00
QStringList list = iter.value().toStringList();
for (int i = 0; i < list.size(); i++) {
2018-07-15 14:51:05 +02:00
list[i] = iter.key() + Tseparator + list[i];
}
collected.append(list);
if ((*iter).m_contained) {
2018-07-15 14:51:05 +02:00
collected.append(iter.key());
}
iter++;
}
return collected;
}
private:
enum Removal { Failed, Succeeded, HasChildren };
2018-07-15 14:51:05 +02:00
Removal removeInternal(QString path = QString())
{
if (path.isEmpty()) {
if (!m_contained) {
2018-07-15 14:51:05 +02:00
// remove all children - we are removing a prefix
clear();
return Succeeded;
}
m_contained = false;
if (children.size()) {
2018-07-15 14:51:05 +02:00
return HasChildren;
}
return Succeeded;
}
Removal remStatus = Failed;
QString childToRemove;
auto sepIndex = path.indexOf(Tseparator);
if (sepIndex == -1) {
2018-07-15 14:51:05 +02:00
childToRemove = path;
auto found = children.find(childToRemove);
if (found == children.end()) {
2018-07-15 14:51:05 +02:00
return Failed;
}
remStatus = (*found).removeInternal();
} else {
2018-07-15 14:51:05 +02:00
childToRemove = path.left(sepIndex);
auto found = children.find(childToRemove);
if (found == children.end()) {
2018-07-15 14:51:05 +02:00
return Failed;
}
remStatus = (*found).removeInternal(path.mid(sepIndex + 1));
}
switch (remStatus) {
2018-07-15 14:51:05 +02:00
case Failed:
case HasChildren: {
2018-07-15 14:51:05 +02:00
return remStatus;
}
case Succeeded: {
2018-07-15 14:51:05 +02:00
children.remove(childToRemove);
if (m_contained) {
2018-07-15 14:51:05 +02:00
return HasChildren;
}
if (children.size()) {
2018-07-15 14:51:05 +02:00
return HasChildren;
}
return Succeeded;
}
}
return Failed;
}
private:
QMap<QString, SeparatorPrefixTree<Tseparator>> children;
2018-07-15 14:51:05 +02:00
bool m_contained = false;
};