implement value.hpp

Signed-off-by: Folling <mail@folling.io>
This commit is contained in:
Folling 2023-12-03 11:41:06 +01:00 committed by Folling
parent 643ece4342
commit 7536cf4c7c
No known key found for this signature in database
6 changed files with 93 additions and 96 deletions

View file

@ -1,6 +1,7 @@
#pragma once #pragma once
#include <boost/container/vector.hpp> #include <boost/container/vector.hpp>
#include <boost/variant2.hpp>
#include <values/value.hpp> #include <values/value.hpp>
@ -20,5 +21,5 @@ public:
~IkarusNumberValue() override = default; ~IkarusNumberValue() override = default;
public: public:
boost::container::vector<data_type> value{}; boost::variant2::variant<boost::variant2::monostate, boost::container::vector<data_type>> data{};
}; };

View file

@ -20,5 +20,5 @@ public:
~IkarusTextValue() override = default; ~IkarusTextValue() override = default;
public: public:
boost::container::vector<data_type> value{}; boost::variant2::variant<boost::variant2::monostate, boost::container::vector<data_type>> data{};
}; };

View file

@ -20,5 +20,5 @@ public:
~IkarusToggleValue() override = default; ~IkarusToggleValue() override = default;
public: public:
boost::container::vector<data_type> value{}; boost::variant2::variant<boost::variant2::monostate, boost::container::vector<data_type>> data{};
}; };

View file

@ -2,9 +2,10 @@
#include <string> #include <string>
#include <fmt/format.h> #include <boost/functional/overloaded_function.hpp>
#include <boost/type_traits.hpp>
#include <cppbase/templates.hpp> #include <fmt/format.h>
#include <ikarus/objects/properties/property_type.h> #include <ikarus/objects/properties/property_type.h>
@ -14,12 +15,11 @@
#include <values/value.hpp> #include <values/value.hpp>
IkarusValue::IkarusValue(Data data): IkarusValue::IkarusValue(Data data):
_data(data) {} data(data) {}
cppbase::Result<IkarusValue, IkarusValue::FromJsonError> IkarusValue::from_json(boost::json::value const& json) { cppbase::Result<IkarusValue, IkarusValue::FromJsonError> IkarusValue::from_json(boost::json::value const& json) {
bool const * intermediate = nullptr;
boost::int64_t const * type = nullptr; boost::int64_t const * type = nullptr;
boost::json::array const * data = nullptr; boost::json::value const * data = nullptr;
if (auto const * obj = json.if_object(); obj == nullptr) { if (auto const * obj = json.if_object(); obj == nullptr) {
return cppbase::err(FromJsonError{}); return cppbase::err(FromJsonError{});
@ -30,103 +30,91 @@ cppbase::Result<IkarusValue, IkarusValue::FromJsonError> IkarusValue::from_json(
return cppbase::err(FromJsonError{}); return cppbase::err(FromJsonError{});
} }
if (auto const * intermediate_value = obj->if_contains("intermediate"); intermediate_value == nullptr) { if (data = obj->if_contains("data"); data == nullptr) {
return cppbase::err(FromJsonError{});
} else if (intermediate = intermediate_value->if_bool(); intermediate == nullptr) {
return cppbase::err(FromJsonError{}); return cppbase::err(FromJsonError{});
} }
if (auto const * data = obj->if_contains("data"); data == nullptr) {
return cppbase::err(FromJsonError{});
} else if (auto * array = data->if_array(); array == nullptr) {
return cppbase::err(FromJsonError{});
}
auto assign_value =
[]<typename T>(
auto const& value, boost::json::array const& array, auto convert = [](auto const& val) { return val; }
) {
for (auto const& data : array) {
if (auto res = boost::json::try_value_to<T>(data); res.has_error()) {
return cppbase::err(FromJsonError{});
} else {
value->get_data().push_back(convert(res.value()));
return cppbase::ok();
}
}
};
switch (*type) { switch (*type) {
case IkarusPropertyType_Toggle: { case IkarusPropertyType_Toggle: {
auto * ret = new IkarusToggleValue{}; auto * ret = new IkarusToggleValue{};
ret->set_intermediate(*intermediate); auto res = boost::json::try_value_to<boost::container::vector<IkarusToggleValue::data_type>>(*data);
assign_value.operator()<bool, bool>(ret, *data); if (res.has_error()) {
return cppbase::err(FromJsonError{});
}
ret->data = std::move(res.value());
} }
case IkarusPropertyType_Number: { case IkarusPropertyType_Number: {
auto * ret = new IkarusToggleValue{}; auto * ret = new IkarusNumberValue{};
ret->set_intermediate(*intermediate); auto res = boost::json::try_value_to<boost::container::vector<IkarusNumberValue::data_type>>(*data);
assign_value.operator()<long double>(ret, *data); if (res.has_error()) {
return cppbase::err(FromJsonError{});
}
ret->data = std::move(res.value());
} }
case IkarusPropertyType_Text: { case IkarusPropertyType_Text: {
auto * ret = new IkarusToggleValue{}; auto * ret = new IkarusTextValue{};
ret->set_intermediate(*intermediate); auto res = boost::json::try_value_to<boost::container::vector<IkarusTextValue::data_type>>(*data);
assign_value.operator()<std::string>(ret, *data); if (res.has_error()) {
return cppbase::err(FromJsonError{});
}
ret->data = std::move(res.value());
}
default: {
return cppbase::err(FromJsonError{});
} }
default: return cppbase::err(FromJsonError{});
} }
} }
} }
boost::json::value IkarusValue::to_json() const { boost::json::value IkarusValue::to_json() const {
auto type = boost::variant2::visit(
boost::make_overloaded_function(
[]([[maybe_unused]] IkarusToggleValue const * value) { return IkarusPropertyType_Toggle; },
[]([[maybe_unused]] IkarusNumberValue const * value) { return IkarusPropertyType_Number; },
[]([[maybe_unused]] IkarusTextValue const * value) { return IkarusPropertyType_Text; }
),
data
);
auto data_json = boost::variant2::visit([](auto const * value) { return boost::json::value_from(value->data); }, data);
return { return {
{"indeterminate", _indeterminate}, {"type", type},
{"type", {"data", data_json}
std::visit(
cppbase::overloaded{
[]([[maybe_unused]] IkarusToggleValue const * value) { return IkarusPropertyType_Toggle; },
[]([[maybe_unused]] IkarusNumberValue const * value) { return IkarusPropertyType_Number; },
[]([[maybe_unused]] IkarusTextValue const * value) { return IkarusPropertyType_Text; },
}, _data
)},
{"data", std::visit([](auto const * value) { return boost::json::value_from(value->get_data()); }, _data)}
}; };
} }
bool IkarusValue::is_indeterminate() const {
return _indeterminate;
}
void IkarusValue::set_indeterminate(bool value) {
_indeterminate = value;
}
IkarusValue::Data& IkarusValue::get_data() {
return _data;
}
IkarusValue::Data const& IkarusValue::get_data() const {
return _data;
}
bool ikarus_value_is_indeterminate(IkarusValue const * value) { bool ikarus_value_is_indeterminate(IkarusValue const * value) {
return value->is_indeterminate(); return boost::variant2::visit(
[](auto const * value) { return boost::variant2::holds_alternative<boost::variant2::monostate>(value->data); },
value->data
);
} }
void ikarus_value_set_indeterminate(IkarusValue * value, bool indeterminate) { void ikarus_value_set_indeterminate(IkarusValue * value, bool indeterminate) {
value->set_intermediate(indeterminate); if (indeterminate) {
return boost::variant2::visit([](auto const * value) { value->data = boost::variant2::monostate{}; }, value->data);
}
return boost::variant2::visit(
[](auto const * value) { value->data = boost::remove_pointer_t<decltype(value)>::data_type{}; }, value->data
);
} }
char const * ikarus_value_to_string(IkarusValue const * value) { char const * ikarus_value_to_string(IkarusValue const * value) {
auto const str = std::visit( auto const str = boost::variant2::visit(
cppbase::overloaded{ boost::make_overloaded_function(
[](IkarusToggleValue const * toggle_value) -> std::string { return toggle_value->get_data() ? "true" : "false"; }, [](IkarusToggleValue const * toggle_value) -> std::string { return toggle_value->data ? "true" : "false"; },
[](IkarusNumberValue const * number_value) -> std::string { return fmt::format("{}", number_value->get_value()); }, [](IkarusNumberValue const * number_value) -> std::string { return fmt::format("{}", number_value->data); },
[](IkarusTextValue const * text_value) -> std::string { return fmt::format("{}", text_value->get_value()); }, [](IkarusTextValue const * text_value) -> std::string { return fmt::format("{}", text_value->data); }
}, ),
value->get_data() value->data
); );
return strdup(str.data()); return strdup(str.data());
@ -139,13 +127,13 @@ void ikarus_value_visit(
void (*text_visitor)(IkarusTextValue *, void *), void (*text_visitor)(IkarusTextValue *, void *),
void * data void * data
) { ) {
std::visit( boost::variant2::visit(
cppbase::overloaded{ boost::make_overloaded_function(
[toggle_visitor, data](IkarusToggleValue * toggle_value) { toggle_visitor(toggle_value, data); }, [toggle_visitor, data](IkarusToggleValue * toggle_value) { toggle_visitor(toggle_value, data); },
[number_visitor, data](IkarusNumberValue * number_value) { number_visitor(number_value, data); }, [number_visitor, data](IkarusNumberValue * number_value) { number_visitor(number_value, data); },
[text_visitor, data](IkarusTextValue * text_value) { text_visitor(text_value, data); }, [text_visitor, data](IkarusTextValue * text_value) { text_visitor(text_value, data); }
}, ),
value->get_data() value->data
); );
} }
@ -156,12 +144,12 @@ void ikarus_value_visit_const(
void (*text_visitor)(IkarusTextValue const *, void *), void (*text_visitor)(IkarusTextValue const *, void *),
void * data void * data
) { ) {
std::visit( boost::variant2::visit(
cppbase::overloaded{ boost::make_overloaded_function(
[toggle_visitor, data](IkarusToggleValue const * toggle_value) { toggle_visitor(toggle_value, data); }, [toggle_visitor, data](IkarusToggleValue const * toggle_value) { toggle_visitor(toggle_value, data); },
[number_visitor, data](IkarusNumberValue const * number_value) { number_visitor(number_value, data); }, [number_visitor, data](IkarusNumberValue const * number_value) { number_visitor(number_value, data); },
[text_visitor, data](IkarusTextValue const * text_value) { text_visitor(text_value, data); }, [text_visitor, data](IkarusTextValue const * text_value) { text_visitor(text_value, data); }
}, ),
value->get_data() value->data
); );
} }

View file

@ -1,14 +1,13 @@
#pragma once #pragma once
#include <variant>
#include <boost/json.hpp> #include <boost/json.hpp>
#include <boost/variant2.hpp>
#include <cppbase/result.hpp> #include <cppbase/result.hpp>
struct IkarusValue { struct IkarusValue {
public: public:
using Data = std::variant<struct IkarusToggleValue *, struct IkarusNumberValue *, struct IkarusTextValue *>; using Data = boost::variant2::variant<struct IkarusToggleValue *, struct IkarusNumberValue *, struct IkarusTextValue *>;
public: public:
explicit IkarusValue(Data data); explicit IkarusValue(Data data);
@ -28,6 +27,5 @@ public:
[[nodiscard]] boost::json::value to_json() const; [[nodiscard]] boost::json::value to_json() const;
public: public:
bool indeterminate{false};
Data data; Data data;
}; };

View file

@ -2,31 +2,41 @@
template<typename V> template<typename V>
typename V::data_type const * ikarus_value_base_get(V * value, size_t idx) { typename V::data_type const * ikarus_value_base_get(V * value, size_t idx) {
if (value->is_indeterminate()) { if (auto * data = value->data.template get_if<std::vector<typename V::data_type>>(); data != nullptr) {
return nullptr; return &(*data)[idx];
} }
return &value->value[idx]; return nullptr;
} }
template<typename V> template<typename V>
size_t ikarus_value_base_get_size(V const * value) { size_t ikarus_value_base_get_size(V const * value) {
return value->value.size(); if (auto * data = value->data.template get_if<std::vector<typename V::data_type>>(); data != nullptr) {
return data->size();
}
return 0;
} }
template<typename V> template<typename V>
void ikarus_value_base_set(V * value, size_t idx, typename V::data_type new_data) { void ikarus_value_base_set(V * value, size_t idx, typename V::data_type new_data) {
value->value[idx] = new_data; if (auto * data = value->data.template get_if<std::vector<typename V::data_type>>(); data != nullptr) {
(*data)[idx] = new_data;
}
} }
template<typename V> template<typename V>
void ikarus_value_base_remove(V * value, size_t idx) { void ikarus_value_base_remove(V * value, size_t idx) {
value->value.erase(value->value.begin() + idx); if (auto * data = value->data.template get_if<std::vector<typename V::data_type>>(); data != nullptr) {
data->erase(data->begin() + idx);
}
} }
template<typename V> template<typename V>
void ikarus_value_base_insert(V * value, size_t idx, typename V::data_type new_data) { void ikarus_value_base_insert(V * value, size_t idx, typename V::data_type new_data) {
value->value.insert(value->value.begin() + idx, new_data); if (auto * data = value->data.template get_if<std::vector<typename V::data_type>>(); data != nullptr) {
data->insert(data->begin() + idx, new_data);
}
} }
template<typename V> template<typename V>