qosd: сдавленные наброски по абстрактному дереву
This commit is contained in:
@@ -0,0 +1,43 @@
|
||||
#pragma once
|
||||
#include <vector>
|
||||
#include <algorithm>
|
||||
|
||||
#include "ifaces/ILink.h"
|
||||
#include "Logger.h"
|
||||
|
||||
/// \brief Базовая реализация интерфейса ILink для хранения связей между элементами.
|
||||
/// \tparam TElem Тип элемента, между которыми устанавливается связь.
|
||||
template <class TElem>
|
||||
class BaseLink : public ILink<TElem> {
|
||||
public:
|
||||
using ElemPtr = std::shared_ptr<TElem>;
|
||||
|
||||
BaseLink(ElemPtr node) : owner_node_(node) {}
|
||||
ElemPtr getNode() const override { return owner_node_.lock(); }
|
||||
ElemPtr getParent() const override { return parent_.lock(); }
|
||||
void setParent(const ElemPtr& parent) override { parent_ = parent; }
|
||||
const std::vector<ElemPtr>& getChildren() const override { return children_; }
|
||||
|
||||
void addChild(const ElemPtr& child) override {
|
||||
this->children_.push_back(child);
|
||||
}
|
||||
|
||||
void removeChild(const ElemPtr& child) override {
|
||||
children_.erase(std::remove(children_.begin(), children_.end(), child), children_.end());
|
||||
}
|
||||
|
||||
void replaceChild(const ElemPtr& oldChild, const ElemPtr& newChild) override {
|
||||
auto it = std::find(children_.begin(), children_.end(), oldChild);
|
||||
if (it != children_.end()) {
|
||||
*it = newChild;
|
||||
}
|
||||
}
|
||||
|
||||
~BaseLink() override {
|
||||
Logger::get("Link").dbg("--- Destructor called for: BaseLink");
|
||||
}
|
||||
protected:
|
||||
std::vector<ElemPtr> children_;
|
||||
std::weak_ptr<TElem> owner_node_;
|
||||
std::weak_ptr<TElem> parent_;
|
||||
};
|
||||
@@ -0,0 +1,12 @@
|
||||
#pragma once
|
||||
#include "links/BaseLink.h"
|
||||
#include <stdexcept>
|
||||
|
||||
/// \brief Связь для листового узла, не допускающая дочерних элементов.
|
||||
class LeafLink : public BaseLink {
|
||||
public:
|
||||
using BaseLink::BaseLink;
|
||||
void addChild(const NodePtr&) override {
|
||||
throw std::logic_error("LeafLink cannot have children");
|
||||
}
|
||||
};
|
||||
@@ -0,0 +1,27 @@
|
||||
#pragma once
|
||||
#include <vector>
|
||||
|
||||
#include "ifaces/ILink.h"
|
||||
#include "Logger.h"
|
||||
|
||||
/// \brief Заглушка для не реализованной связи между элементами.
|
||||
/// Все методы возвращают пустые значения или не выполняют действий.
|
||||
/// \tparam TElem Тип элемента.
|
||||
template <class TElem>
|
||||
class NotImplementedLink : public ILink<TElem> {
|
||||
using ElemPtr = std::shared_ptr<TElem>>;
|
||||
public:
|
||||
NotImplementedLink(ElemPtr node) {}
|
||||
ElemPtr getNode() const override { return nullptr; }
|
||||
ElemPtr getParent() const override { return nullptr; }
|
||||
void setParent(const ElemPtr& parent) override { }
|
||||
const std::vector<ElemPtr>& getChildren() const override { return empty_; }
|
||||
void addChild(const ElemPtr& child) override { }
|
||||
void removeChild(const ElemPtr& child) override { }
|
||||
|
||||
~NotImplementedLink() override {
|
||||
Logger::get("Link").dbg("--- Destructor called for: NotImplementedLink");
|
||||
}
|
||||
private:
|
||||
std::vector<ElemPtr> empty_;
|
||||
};
|
||||
@@ -0,0 +1,17 @@
|
||||
#pragma once
|
||||
#include "links/BaseLink.h"
|
||||
|
||||
/// \brief Связь "один-ко-многим" между элементами одного типа.
|
||||
/// Каждый дочерний элемент должен быть того же типа, что и родитель.
|
||||
/// \tparam TElem Тип элемента.
|
||||
template <class TElem>
|
||||
class OneToManyLink : public BaseLink<TElem> {
|
||||
public:
|
||||
OneToManyLink(std::shared_ptr<TElem> e) : BaseLink<TElem>(e) {}
|
||||
void addChild(const std::shared_ptr<TElem>& child) override {
|
||||
/* Each child must be exactly the same type as parent */
|
||||
if (typeid(*child) != typeid(*this->owner_node_.lock()))
|
||||
throw std::logic_error("Foundling child");
|
||||
BaseLink<TElem>::addChild(child);
|
||||
}
|
||||
};
|
||||
@@ -0,0 +1,21 @@
|
||||
#pragma once
|
||||
#include "links/BaseLink.h"
|
||||
#include <stdexcept>
|
||||
|
||||
/// \brief Связь "один-к-одному" между элементами.
|
||||
/// Позволяет добавить только одного дочернего элемента.
|
||||
/// \tparam TElem Тип элемента.
|
||||
template <class TElem>
|
||||
class OneToOneLink : public BaseLink<TElem> {
|
||||
public:
|
||||
using ElemPtr = std::shared_ptr<TElem>;
|
||||
|
||||
OneToOneLink(ElemPtr e) : BaseLink<TElem>(e) {}
|
||||
|
||||
void addChild(const ElemPtr& child) override {
|
||||
if (!this->children_.empty())
|
||||
throw std::logic_error("Too many children");
|
||||
|
||||
BaseLink<TElem>::addChild(child);
|
||||
}
|
||||
};
|
||||
Reference in New Issue
Block a user