kakoune/src/register_manager.hh

112 lines
2.8 KiB
C++
Raw Normal View History

2011-09-23 16:31:57 +02:00
#ifndef register_manager_hh_INCLUDED
#define register_manager_hh_INCLUDED
2015-04-23 22:51:48 +02:00
#include "array_view.hh"
#include "exception.hh"
2011-09-23 16:31:57 +02:00
#include "utils.hh"
#include "unordered_map.hh"
2015-04-23 22:51:48 +02:00
#include "string.hh"
2015-01-12 14:58:41 +01:00
#include "vector.hh"
2011-09-23 16:31:57 +02:00
namespace Kakoune
{
2015-04-23 22:51:48 +02:00
class Context;
class Register
{
public:
virtual ~Register() = default;
2015-04-23 22:51:48 +02:00
virtual void set(Context& context, ConstArrayView<String> values) = 0;
virtual ConstArrayView<String> get(const Context& context) = 0;
2015-04-23 22:51:48 +02:00
};
// static value register, which can be modified
// using operator=, so should be user modifiable
class StaticRegister : public Register
{
public:
void set(Context&, ConstArrayView<String> values) override
{
m_content = Vector<String, MemoryDomain::Registers>(values.begin(), values.end());
}
ConstArrayView<String> get(const Context&) override
{
if (m_content.empty())
return ConstArrayView<String>(String::ms_empty);
else
return ConstArrayView<String>(m_content);
}
protected:
Vector<String, MemoryDomain::Registers> m_content;
};
// Dynamic value register, use it's RegisterRetriever
// to get it's value when needed.
template<typename Getter, typename Setter>
class DynamicRegister : public StaticRegister
{
public:
DynamicRegister(Getter getter, Setter setter)
2017-02-22 10:44:19 +01:00
: m_getter(std::move(getter)), m_setter(std::move(setter)) {}
void set(Context& context, ConstArrayView<String> values) override
{
m_setter(context, values);
}
ConstArrayView<String> get(const Context& context) override
{
m_content = m_getter(context);
return StaticRegister::get(context);
}
private:
Getter m_getter;
Setter m_setter;
};
template<typename Func>
std::unique_ptr<Register> make_dyn_reg(Func func)
{
auto setter = [](Context&, ConstArrayView<String>)
{
throw runtime_error("this register is not assignable");
};
return make_unique<DynamicRegister<Func, decltype(setter)>>(std::move(func), setter);
}
template<typename Getter, typename Setter>
std::unique_ptr<Register> make_dyn_reg(Getter getter, Setter setter)
{
return make_unique<DynamicRegister<Getter, Setter>>(std::move(getter), std::move(setter));
}
class NullRegister : public Register
{
public:
void set(Context&, ConstArrayView<String>) override {}
ConstArrayView<String> get(const Context&) override
{
return ConstArrayView<String>(String::ms_empty);
}
};
2011-09-23 16:31:57 +02:00
class RegisterManager : public Singleton<RegisterManager>
{
public:
Register& operator[](StringView reg) const;
Register& operator[](Codepoint c) const;
void add_register(char c, std::unique_ptr<Register> reg);
2011-09-23 16:31:57 +02:00
protected:
2015-01-14 20:16:32 +01:00
UnorderedMap<char, std::unique_ptr<Register>, MemoryDomain::Registers> m_registers;
2011-09-23 16:31:57 +02:00
};
}
#endif // register_manager_hh_INCLUDED