#ifndef option_types_hh_INCLUDED #define option_types_hh_INCLUDED #include "exception.hh" #include "string.hh" #include "units.hh" #include "coord.hh" #include "array_view.hh" #include "id_map.hh" #include "flags.hh" #include "enum.hh" #include #include namespace Kakoune { template struct option_type_name; template using void_t = void; template struct option_type_name> { static decltype(T::option_type_name) name() { return T::option_type_name; } }; template struct option_type_name::value>::type> { static String name() { constexpr StringView type = WithBitOps::value ? "flags" : "enum"; auto name = enum_desc(Enum{}); return type + "(" + join(name | transform(std::mem_fn(&EnumDesc::name)), '|') + ")"; } }; inline String option_to_string(int opt) { return to_string(opt); } inline void option_from_string(StringView str, int& opt) { opt = str_to_int(str); } inline bool option_add(int& opt, StringView str) { auto val = str_to_int(str); opt += val; return val != 0; } template<> struct option_type_name { static StringView name() { return "int"; } }; inline String option_to_string(size_t opt) { return to_string(opt); } inline void option_from_string(StringView str, size_t& opt) { opt = str_to_int(str); } inline String option_to_string(bool opt) { return opt ? "true" : "false"; } inline void option_from_string(StringView str, bool& opt) { if (str == "true" or str == "yes") opt = true; else if (str == "false" or str == "no") opt = false; else throw runtime_error("boolean values are either true, yes, false or no"); } template<> struct option_type_name { static StringView name() { return "bool"; } }; constexpr char list_separator = ':'; template String option_to_string(const Vector& opt) { String res; for (size_t i = 0; i < opt.size(); ++i) { res += escape(option_to_string(opt[i]), list_separator, '\\'); if (i != opt.size() - 1) res += list_separator; } return res; } template void option_from_string(StringView str, Vector& opt) { opt.clear(); Vector elems = split(str, list_separator, '\\'); for (auto& elem: elems) { T opt_elem; option_from_string(elem, opt_elem); opt.push_back(opt_elem); } } template bool option_add(Vector& opt, StringView str) { Vector vec; option_from_string(str, vec); std::copy(std::make_move_iterator(vec.begin()), std::make_move_iterator(vec.end()), back_inserter(opt)); return not vec.empty(); } template struct option_type_name> { static String name() { return option_type_name::name() + StringView{"-list"}; } }; template String option_to_string(const IdMap& opt) { String res; for (auto it = begin(opt); it != end(opt); ++it) { if (it != begin(opt)) res += list_separator; String elem = escape(option_to_string(it->key), '=', '\\') + "=" + escape(option_to_string(it->value), '=', '\\'); res += escape(elem, list_separator, '\\'); } return res; } template void option_from_string(StringView str, IdMap& opt) { opt.clear(); for (auto& elem : split(str, list_separator, '\\')) { Vector pair_str = split(elem, '=', '\\'); if (pair_str.size() != 2) throw runtime_error("map option expects key=value"); String key; Value value; option_from_string(pair_str[0], key); option_from_string(pair_str[1], value); opt.append({ std::move(key), std::move(value) }); } } template struct option_type_name> { static String name() { return format("str-to-{}-map", option_type_name::name()); } }; constexpr char tuple_separator = '|'; template struct TupleOptionDetail { static String to_string(const std::tuple& opt) { return TupleOptionDetail::to_string(opt) + tuple_separator + escape(option_to_string(std::get(opt)), tuple_separator, '\\'); } static void from_string(ConstArrayView elems, std::tuple& opt) { option_from_string(elems[I], std::get(opt)); TupleOptionDetail::from_string(elems, opt); } }; template struct TupleOptionDetail<0, Types...> { static String to_string(const std::tuple& opt) { return option_to_string(std::get<0>(opt)); } static void from_string(ConstArrayView elems, std::tuple& opt) { option_from_string(elems[0], std::get<0>(opt)); } }; template String option_to_string(const std::tuple& opt) { return TupleOptionDetail::to_string(opt); } template void option_from_string(StringView str, std::tuple& opt) { auto elems = split(str, tuple_separator, '\\'); if (elems.size() != sizeof...(Types)) throw runtime_error(elems.size() < sizeof...(Types) ? "not enough elements in tuple" : "to many elements in tuple"); TupleOptionDetail::from_string(elems, opt); } template inline String option_to_string(const StronglyTypedNumber& opt) { return to_string(opt); } template inline void option_from_string(StringView str, StronglyTypedNumber& opt) { opt = StronglyTypedNumber{str_to_int(str)}; } template inline bool option_add(StronglyTypedNumber& opt, StringView str) { int val = str_to_int(str); opt += val; return val != 0; } template bool option_add(T&, StringView str) { throw runtime_error("no add operation supported for this option type"); } template inline void option_from_string(StringView str, LineAndColumn& opt) { auto vals = split(str, ','); if (vals.size() != 2) throw runtime_error("expected ,"); opt.line = str_to_int(vals[0]); opt.column = str_to_int(vals[1]); } template inline String option_to_string(const LineAndColumn& opt) { return format("{},{}", opt.line, opt.column); } enum class DebugFlags { None = 0, Hooks = 1 << 0, Shell = 1 << 1, Profile = 1 << 2, Keys = 1 << 3, }; template<> struct WithBitOps : std::true_type {}; constexpr Array, 4> enum_desc(DebugFlags) { return { { { DebugFlags::Hooks, "hooks" }, { DebugFlags::Shell, "shell" }, { DebugFlags::Profile, "profile" }, { DebugFlags::Keys, "keys" } } }; } template struct PrefixedList { P prefix; Vector list; }; template inline bool operator==(const PrefixedList& lhs, const PrefixedList& rhs) { return lhs.prefix == rhs.prefix and lhs.list == rhs.list; } template inline bool operator!=(const PrefixedList& lhs, const PrefixedList& rhs) { return not (lhs == rhs); } template inline String option_to_string(const PrefixedList& opt) { return format("{}:{}", opt.prefix, option_to_string(opt.list)); } template inline void option_from_string(StringView str, PrefixedList& opt) { auto it = find(str, ':'); option_from_string(StringView{str.begin(), it}, opt.prefix); if (it != str.end()) option_from_string({it+1, str.end()}, opt.list); } template inline bool option_add(PrefixedList& opt, StringView str) { return option_add(opt.list, str); } template using TimestampedList = PrefixedList; } #endif // option_types_hh_INCLUDED