2023-11-11 21:18:12 +03:00
|
|
|
//
|
|
|
|
// VarVisitor.h
|
|
|
|
//
|
|
|
|
// Library: Foundation
|
|
|
|
// Package: Dynamic
|
|
|
|
// Module: VarVisitor
|
|
|
|
//
|
|
|
|
// Definition of the VarVisitor class.
|
|
|
|
//
|
|
|
|
// Copyright (c) 2023, Applied Informatics Software Engineering GmbH.
|
|
|
|
// and Contributors.
|
|
|
|
//
|
|
|
|
// SPDX-License-Identifier: BSL-1.0
|
|
|
|
//
|
|
|
|
|
|
|
|
|
2024-02-06 09:59:38 +01:00
|
|
|
#ifndef Foundation_VarVisitor_INCLUDED
|
|
|
|
#define Foundation_VarVisitor_INCLUDED
|
|
|
|
|
|
|
|
|
2023-11-11 21:18:12 +03:00
|
|
|
#include "Poco/Dynamic/Var.h"
|
|
|
|
#include <unordered_map>
|
|
|
|
#include <functional>
|
|
|
|
|
2024-02-06 09:59:38 +01:00
|
|
|
|
2023-11-11 21:18:12 +03:00
|
|
|
namespace Poco {
|
|
|
|
namespace Details {
|
|
|
|
|
2024-02-06 09:59:38 +01:00
|
|
|
|
|
|
|
#ifndef POCO_DOC
|
|
|
|
|
|
|
|
|
2023-11-11 21:18:12 +03:00
|
|
|
struct TypeInfoHash
|
|
|
|
{
|
|
|
|
inline std::size_t operator()(std::type_info const& t) const { return t.hash_code(); }
|
|
|
|
};
|
|
|
|
|
2024-02-06 09:59:38 +01:00
|
|
|
|
2023-11-11 21:18:12 +03:00
|
|
|
struct EqualRef
|
|
|
|
{
|
|
|
|
template <typename T>
|
|
|
|
bool operator()(std::reference_wrapper<T> a, std::reference_wrapper<T> b) const
|
|
|
|
{
|
|
|
|
return a.get() == b.get();
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
|
2024-02-06 09:59:38 +01:00
|
|
|
using TypeInfoRef = std::reference_wrapper<std::type_info const>;
|
2023-11-11 21:18:12 +03:00
|
|
|
using HandlerCaller = std::function<void(const Poco::Dynamic::Var&)>;
|
|
|
|
|
2024-02-06 09:59:38 +01:00
|
|
|
|
2023-11-11 21:18:12 +03:00
|
|
|
template <typename T>
|
2024-02-06 09:59:38 +01:00
|
|
|
using HandlerPointer = void (*)(const T&);
|
|
|
|
|
2023-11-11 21:18:12 +03:00
|
|
|
|
|
|
|
template <typename T>
|
|
|
|
using Handler = std::function<void(const T&)>;
|
|
|
|
|
2024-02-06 09:59:38 +01:00
|
|
|
|
|
|
|
#endif // POCO_DOC
|
|
|
|
|
|
|
|
|
|
|
|
} // Details
|
|
|
|
|
2023-11-11 21:18:12 +03:00
|
|
|
|
|
|
|
namespace Dynamic {
|
|
|
|
|
2024-02-06 09:59:38 +01:00
|
|
|
|
2023-11-11 21:18:12 +03:00
|
|
|
class Foundation_API Visitor
|
|
|
|
/// VarVisitor class.
|
|
|
|
{
|
|
|
|
std::unordered_map<Details::TypeInfoRef,
|
|
|
|
Details::HandlerCaller,
|
|
|
|
Details::TypeInfoHash,
|
|
|
|
Details::EqualRef> _handlers;
|
|
|
|
|
|
|
|
public:
|
|
|
|
template <typename T>
|
2024-02-06 09:59:38 +01:00
|
|
|
bool addHandler(const Details::Handler<T>& f)
|
|
|
|
/// Add handler for specific type T which holds in Var.
|
|
|
|
/// This method is more safe, because it saves copy of handler : lambda or std::function.
|
|
|
|
/// Returns true if handler was added.
|
2023-11-11 21:18:12 +03:00
|
|
|
{
|
|
|
|
auto result = _handlers.emplace(std::ref(typeid(T)),
|
|
|
|
Details::HandlerCaller([handler = f](const Poco::Dynamic::Var& x)
|
|
|
|
{
|
|
|
|
handler(x.extract<T>());
|
|
|
|
}));
|
|
|
|
return result.second;
|
|
|
|
}
|
|
|
|
|
|
|
|
template <typename T>
|
|
|
|
bool addHandler(Details::HandlerPointer<T> f)
|
2024-02-06 09:59:38 +01:00
|
|
|
/// Add handler for specific type T which holds in Var.
|
|
|
|
/// This method is less safe, because it saves only copy of function pointer.
|
|
|
|
/// Returns true if handler was added.
|
2023-11-11 21:18:12 +03:00
|
|
|
{
|
|
|
|
auto result = _handlers.emplace(std::ref(typeid(T)),
|
|
|
|
Details::HandlerCaller([handlerPointer = f](const Poco::Dynamic::Var& x)
|
|
|
|
{
|
|
|
|
handlerPointer(x.extract<T>());
|
|
|
|
}));
|
|
|
|
return result.second;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool visit(const Poco::Dynamic::Var& x) const;
|
2024-02-06 09:59:38 +01:00
|
|
|
/// Find handler for held type and if it exists call handler.
|
|
|
|
/// Returns true if hanlder was found othrewise returns false.
|
2023-11-11 21:18:12 +03:00
|
|
|
};
|
|
|
|
|
2024-02-06 09:59:38 +01:00
|
|
|
|
2023-11-11 21:18:12 +03:00
|
|
|
} } // namespace Poco::Dynamic
|
|
|
|
|
2024-02-06 09:59:38 +01:00
|
|
|
|
2023-11-11 21:18:12 +03:00
|
|
|
#endif // Foundation_VarVisitor_INCLUDED
|