#ifndef option_types_hh_INCLUDED #define option_types_hh_INCLUDED #include "array_view.hh" #include "coord.hh" #include "exception.hh" #include "flags.hh" #include "hash_map.hh" #include "option.hh" #include "ranges.hh" #include "string.hh" #include "string_utils.hh" #include "units.hh" #include #include namespace Kakoune { template constexpr decltype(T::option_type_name) option_type_name(Meta::Type) { return T::option_type_name; } template std::enable_if_t::value, String> option_type_name(Meta::Type) { return format("{}({})", with_bit_ops(Meta::Type{}) ? "flags" : "enum", join(enum_desc(Meta::Type{}) | transform(&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; } constexpr StringView option_type_name(Meta::Type) { 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"); } constexpr StringView option_type_name(Meta::Type) { return "bool"; } inline String option_to_string(Codepoint opt) { return to_string(opt); } inline void option_from_string(StringView str, Codepoint& opt) { if (str.char_length() != 1) throw runtime_error{format("'{}' is not a single codepoint", str)}; opt = str[0_char]; } constexpr StringView option_type_name(Meta::Type) { return "codepoint"; } constexpr char list_separator = ':'; template String option_to_string(const Vector& opt) { return join(opt | transform([](const T& t) { return option_to_string(t); }), list_separator); } template void option_list_postprocess(Vector& opt) {} template void option_from_string(StringView str, Vector& opt) { opt.clear(); for (auto&& elem : str | split(list_separator, '\\') | transform(unescape)) { T opt_elem; option_from_string(elem, opt_elem); opt.push_back(opt_elem); } option_list_postprocess(opt); } template bool option_add(Vector& opt, StringView str) { Vector vec; option_from_string(str, vec); opt.insert(opt.end(), std::make_move_iterator(vec.begin()), std::make_move_iterator(vec.end())); option_list_postprocess(opt); return not vec.empty(); } template String option_type_name(Meta::Type>) { return option_type_name(Meta::Type{}) + "-list"_sv; } template String option_to_string(const HashMap& opt) { String res; for (auto it = opt.begin(); it != opt.end(); ++it) { if (it != opt.begin()) 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 bool option_add(HashMap& opt, StringView str) { bool changed = false; for (auto&& elem : str | split(list_separator, '\\') | transform(unescape)) { struct error : runtime_error { error(size_t) : runtime_error{"map option expects key=value"} {} }; auto key_value = elem | split('=', '\\') | transform(unescape<'=', '\\'>) | static_gather(); HashItem item; option_from_string(key_value[0], item.key); option_from_string(key_value[1], item.value); opt.insert(std::move(item)); changed = true; } return changed; } template void option_from_string(StringView str, HashMap& opt) { opt.clear(); option_add(opt, str); } template String option_type_name(Meta::Type>) { return format("{}-to-{}-map", option_type_name(Meta::Type{}), option_type_name(Meta::Type{})); } 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) { struct error : runtime_error { error(size_t i) : runtime_error{i < sizeof...(Types) ? "not enough elements in tuple" : "too many elements in tuple"} {} }; auto elems = str | split(tuple_separator, '\\') | transform(unescape) | static_gather(); 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; } struct WorstMatch { template WorstMatch(T&&) {} }; inline bool option_add(WorstMatch, StringView) { throw runtime_error("no add operation supported for this option type"); } class Context; inline void option_update(WorstMatch, const Context&) { throw runtime_error("no update operation supported for this option type"); } template inline void option_from_string(StringView str, LineAndColumn& opt) { struct error : runtime_error { error(size_t) : runtime_error{"expected ,"} {} }; auto vals = str | split(',') | static_gather(); 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); } template{}))> EnableIfWithBitOps option_to_string(Flags flags) { constexpr auto desc = enum_desc(Meta::Type{}); String res; for (int i = 0; i < desc.size(); ++i) { if (not (flags & desc[i].value)) continue; if (not res.empty()) res += "|"; res += desc[i].name; } return res; } template{}))> EnableIfWithoutBitOps option_to_string(Enum e) { constexpr auto desc = enum_desc(Meta::Type{}); auto it = find_if(desc, [e](const EnumDesc& d) { return d.value == e; }); if (it != desc.end()) return it->name.str(); kak_assert(false); return {}; } template{}))> EnableIfWithBitOps option_from_string(StringView str, Flags& flags) { constexpr auto desc = enum_desc(Meta::Type{}); flags = Flags{}; for (auto s : str | split('|')) { auto it = find_if(desc, [s](const EnumDesc& d) { return d.name == s; }); if (it == desc.end()) throw runtime_error(format("invalid flag value '{}'", s)); flags |= it->value; } } template{}))> EnableIfWithoutBitOps option_from_string(StringView str, Enum& e) { constexpr auto desc = enum_desc(Meta::Type{}); auto it = find_if(desc, [str](const EnumDesc& d) { return d.name == str; }); if (it == desc.end()) throw runtime_error(format("invalid enum value '{}'", str)); e = it->value; } template{}))> EnableIfWithBitOps option_add(Flags& opt, StringView str) { Flags res = Flags{}; option_from_string(str, res); opt |= res; return res != (Flags)0; } template inline String option_to_string(const PrefixedList& opt) { if (opt.list.empty()) return format("{}", escape(option_to_string(opt.prefix), list_separator, '\\')); else return format("{}{}{}", escape(option_to_string(opt.prefix), list_separator, '\\'), list_separator, option_to_string(opt.list)); } template inline void option_from_string(StringView str, PrefixedList& opt) { auto it = find(str, list_separator); 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); } } #endif // option_types_hh_INCLUDED