e8c8833211
Keep it as is, commit undo in current buffer, and we will commit it in the new buffer when edition level gets back to 0.
247 lines
5.4 KiB
C++
247 lines
5.4 KiB
C++
#include "context.hh"
|
|
|
|
#include "alias_registry.hh"
|
|
#include "client.hh"
|
|
#include "user_interface.hh"
|
|
#include "register_manager.hh"
|
|
#include "window.hh"
|
|
|
|
namespace Kakoune
|
|
{
|
|
|
|
Context::Context() = default;
|
|
Context::~Context() = default;
|
|
|
|
Context::Context(InputHandler& input_handler, SelectionList selections,
|
|
String name)
|
|
: m_input_handler{&input_handler},
|
|
m_selections{std::move(selections)},
|
|
m_name(std::move(name))
|
|
{}
|
|
|
|
Buffer& Context::buffer() const
|
|
{
|
|
if (not has_buffer())
|
|
throw runtime_error("no buffer in context");
|
|
return const_cast<Buffer&>((*m_selections).buffer());
|
|
}
|
|
|
|
Window& Context::window() const
|
|
{
|
|
if (not has_window())
|
|
throw runtime_error("no window in context");
|
|
return *m_window;
|
|
}
|
|
|
|
InputHandler& Context::input_handler() const
|
|
{
|
|
if (not has_input_handler())
|
|
throw runtime_error("no input handler in context");
|
|
return *m_input_handler;
|
|
}
|
|
|
|
Client& Context::client() const
|
|
{
|
|
if (not has_client())
|
|
throw runtime_error("no client in context");
|
|
return *m_client;
|
|
}
|
|
|
|
UserInterface& Context::ui() const
|
|
{
|
|
if (not has_ui())
|
|
throw runtime_error("no user interface in context");
|
|
return client().ui();
|
|
}
|
|
|
|
Scope& Context::scope() const
|
|
{
|
|
if (has_window())
|
|
return window();
|
|
if (has_buffer())
|
|
return buffer();
|
|
return GlobalScope::instance();
|
|
}
|
|
|
|
OptionManager& Context::options() const
|
|
{
|
|
return scope().options();
|
|
}
|
|
|
|
HookManager& Context::hooks() const
|
|
{
|
|
return scope().hooks();
|
|
}
|
|
|
|
KeymapManager& Context::keymaps() const
|
|
{
|
|
return scope().keymaps();
|
|
}
|
|
|
|
AliasRegistry& Context::aliases() const
|
|
{
|
|
return scope().aliases();
|
|
}
|
|
|
|
void Context::set_client(Client& client)
|
|
{
|
|
kak_assert(not has_client());
|
|
m_client.reset(&client);
|
|
}
|
|
|
|
void Context::set_window(Window& window)
|
|
{
|
|
kak_assert(&window.buffer() == &buffer());
|
|
m_window.reset(&window);
|
|
}
|
|
|
|
void Context::print_status(DisplayLine status) const
|
|
{
|
|
if (has_client())
|
|
client().print_status(std::move(status));
|
|
}
|
|
|
|
void Context::push_jump()
|
|
{
|
|
const SelectionList& jump = selections();
|
|
if (m_current_jump != m_jump_list.end())
|
|
{
|
|
auto begin = m_current_jump;
|
|
if (&buffer() != &begin->buffer() or *begin != jump)
|
|
++begin;
|
|
m_jump_list.erase(begin, m_jump_list.end());
|
|
}
|
|
m_jump_list.erase(std::remove(begin(m_jump_list), end(m_jump_list), jump),
|
|
end(m_jump_list));
|
|
m_jump_list.push_back(jump);
|
|
m_current_jump = m_jump_list.end();
|
|
}
|
|
|
|
const SelectionList& Context::jump_forward()
|
|
{
|
|
if (m_current_jump != m_jump_list.end() and
|
|
m_current_jump + 1 != m_jump_list.end())
|
|
{
|
|
SelectionList& res = *++m_current_jump;
|
|
res.update();
|
|
return res;
|
|
}
|
|
throw runtime_error("no next jump");
|
|
}
|
|
|
|
const SelectionList& Context::jump_backward()
|
|
{
|
|
if (m_current_jump != m_jump_list.end() and
|
|
*m_current_jump != selections())
|
|
{
|
|
push_jump();
|
|
SelectionList& res = *--m_current_jump;
|
|
res.update();
|
|
return res;
|
|
}
|
|
if (m_current_jump != m_jump_list.begin())
|
|
{
|
|
if (m_current_jump == m_jump_list.end())
|
|
{
|
|
push_jump();
|
|
--m_current_jump;
|
|
}
|
|
SelectionList& res = *--m_current_jump;
|
|
res.update();
|
|
return res;
|
|
}
|
|
throw runtime_error("no previous jump");
|
|
}
|
|
|
|
void Context::forget_jumps_to_buffer(Buffer& buffer)
|
|
{
|
|
for (auto it = m_jump_list.begin(); it != m_jump_list.end();)
|
|
{
|
|
if (&it->buffer() == &buffer)
|
|
{
|
|
if (it < m_current_jump)
|
|
--m_current_jump;
|
|
else if (it == m_current_jump)
|
|
m_current_jump = m_jump_list.end()-1;
|
|
|
|
it = m_jump_list.erase(it);
|
|
}
|
|
else
|
|
++it;
|
|
}
|
|
}
|
|
|
|
void Context::change_buffer(Buffer& buffer)
|
|
{
|
|
if (&buffer == &this->buffer())
|
|
return;
|
|
|
|
if (m_edition_level > 0)
|
|
this->buffer().commit_undo_group();
|
|
|
|
m_window.reset();
|
|
if (has_client())
|
|
client().change_buffer(buffer);
|
|
else
|
|
m_selections = SelectionList{buffer, Selection{}};
|
|
if (has_input_handler())
|
|
input_handler().reset_normal_mode();
|
|
}
|
|
|
|
SelectionList& Context::selections()
|
|
{
|
|
if (not m_selections)
|
|
throw runtime_error("no selections in context");
|
|
(*m_selections).update();
|
|
return *m_selections;
|
|
}
|
|
|
|
const SelectionList& Context::selections() const
|
|
{
|
|
return const_cast<Context&>(*this).selections();
|
|
}
|
|
|
|
std::vector<String> Context::selections_content() const
|
|
{
|
|
auto& buf = buffer();
|
|
std::vector<String> contents;
|
|
for (auto& sel : selections())
|
|
contents.push_back(buf.string(sel.min(), buf.char_next(sel.max())));
|
|
return contents;
|
|
}
|
|
|
|
void Context::set_selections(std::vector<Selection> sels)
|
|
{
|
|
*m_selections = std::move(sels);
|
|
(*m_selections).check_invariant();
|
|
}
|
|
|
|
void Context::begin_edition()
|
|
{
|
|
if (m_edition_level >= 0)
|
|
++m_edition_level;
|
|
}
|
|
|
|
void Context::end_edition()
|
|
{
|
|
if (m_edition_level < 0)
|
|
return;
|
|
|
|
kak_assert(m_edition_level != 0);
|
|
if (m_edition_level == 1)
|
|
buffer().commit_undo_group();
|
|
|
|
--m_edition_level;
|
|
}
|
|
|
|
StringView Context::main_sel_register_value(StringView reg) const
|
|
{
|
|
auto strings = RegisterManager::instance()[reg].values(*this);
|
|
size_t index = m_selections ? (*m_selections).main_index() : 0;
|
|
if (strings.size() <= index)
|
|
index = strings.size() - 1;
|
|
return strings[index];
|
|
}
|
|
|
|
}
|