qosd: сдавленные наброски по абстрактному дереву

This commit is contained in:
Сергей Маринкевич
2025-07-21 15:52:36 +07:00
commit 6a54024223
20 changed files with 651 additions and 0 deletions
+50
View File
@@ -0,0 +1,50 @@
#pragma once
#include <iostream>
#include <memory>
#include "ifaces/ILinkMixin.h"
#include "ifaces/INode.h"
#include "Logger.h"
/// \brief Базовый миксин для реализации ILinkMixin для INode.
/// Предоставляет базовую реализацию методов для работы с дочерними узлами и родителем.
class BaseLinkMixin : public virtual ILinkMixin<INode>,
public std::enable_shared_from_this<ILinkMixin<INode>> {
using ElemPtr = std::shared_ptr<INode>;
public:
~BaseLinkMixin() override {
Logger::get("Mixin").dbg("--- Destructor called for: BaseLinkMixin");
}
void linkChild(const ElemPtr& child) override {
getLink()->addChild(child);
auto childLink = child->getLink();
childLink->setParent(getNode());
}
void unlinkParent() override {
auto link = getLink();
ElemPtr parent = link->getParent();
if (!parent)
throw std::logic_error("Have no parent!");
auto parentLink = parent->getLink();
parentLink->removeChild(getNode());
getLink()->setParent(nullptr);
}
const std::vector<ElemPtr>& children() override {
auto link = getLink();
return link->getChildren();
}
protected:
ElemPtr getNode() {
return std::dynamic_pointer_cast<INode>(shared_from_this());
}
};
+48
View File
@@ -0,0 +1,48 @@
#pragma once
#include <iostream>
#include "mixins/LazyLinkMixin.h"
#include <memory>
#include "links/OneToManyLink.h"
#include "links/OneToOneLink.h"
#include "Logger.h"
/// \brief Миксин для иерархических связей между элементами.
/// Автоматически выбирает тип связи (один-ко-многим или один-к-одному) в зависимости от типа дочернего элемента.
/// \tparam TElem Тип дочернего элемента.
template <class TElem>
class HierarchicalLinkMixin : public LazyLinkMixin<OneToOneLink<TElem>> {
using LinkPtr = std::shared_ptr<ILink<TElem>>;
using ElemPtr = std::shared_ptr<TElem>;
public:
~HierarchicalLinkMixin() override {
Logger::get("Mixin").dbg("--- Destructor called for: HierarchicalLinkMixin");
}
void linkChild(const ElemPtr& child) override {
hierarchicalInit(child);
LazyLinkMixin<OneToOneLink<TElem>>::linkChild(child);
}
protected:
void hierarchicalInit(const ElemPtr& child) {
Logger::get("Mixin").dbg("--- hierarchicalInit called");
if (this->link_ && !this->link_->getChildren().empty())
return;
LinkPtr newLink;
if (typeid(*child) == typeid(*this)) {
Logger::get("Mixin").dbg("--- Mutate to OneToMany");
newLink = std::make_shared<OneToManyLink<TElem>>(this->getNode());
} else {
Logger::get("Mixin").dbg("--- Mutate to OneToOne");
newLink = std::make_shared<OneToOneLink<TElem>>(this->getNode());
}
if (newLink && this->link_)
newLink->setParent(this->link_->getParent());
this->link_ = newLink;
}
};
+39
View File
@@ -0,0 +1,39 @@
#pragma once
#include <iostream>
#include "mixins/BaseLinkMixin.h"
#include <memory>
#include "Logger.h"
/// \brief Миксин для ленивой инициализации связи (link) с дочерними элементами.
/// \tparam TLink Тип используемой связи (link).
template <class TLink>
class LazyLinkMixin : public BaseLinkMixin {
public:
void unlinkParent() override {
/* No link -- no parent, who'll unlinked? */
if (!this->link_)
throw std::logic_error("Link isn't inited!");
BaseLinkMixin::unlinkParent();
}
LinkPtr getLink() override {
lazyInit();
return this->link_;
}
LazyLinkMixin() {}
~LazyLinkMixin() override {
Logger::get("Mixin").dbg("--- Destructor called for: LazyLinkMixin");
}
protected:
void lazyInit() {
if (!link_) {
link_ = std::make_shared<TLink>(
BaseLinkMixin::getNode());
}
}
LinkPtr link_;
};