2011-11-26 19:32:57 +01:00
|
|
|
#ifndef context_hh_INCLUDED
|
|
|
|
#define context_hh_INCLUDED
|
|
|
|
|
2012-02-13 22:32:54 +01:00
|
|
|
#include "window.hh"
|
2012-09-26 14:13:04 +02:00
|
|
|
#include "user_interface.hh"
|
2012-02-13 22:32:54 +01:00
|
|
|
|
2011-11-26 19:32:57 +01:00
|
|
|
namespace Kakoune
|
|
|
|
{
|
|
|
|
|
2013-01-28 13:48:34 +01:00
|
|
|
class InputHandler;
|
|
|
|
|
2012-10-10 13:57:15 +02:00
|
|
|
// A Context is used to access non singleton objects for various services
|
|
|
|
// in commands.
|
|
|
|
//
|
2012-10-17 13:14:03 +02:00
|
|
|
// The Context object links an InputHandler, an Editor (which may be a Window),
|
2012-10-10 13:57:15 +02:00
|
|
|
// and a UserInterface. It may represent an interactive user window, or
|
|
|
|
// a hook execution or a macro replay.
|
2011-11-26 19:32:57 +01:00
|
|
|
struct Context
|
|
|
|
{
|
2012-08-05 18:23:37 +02:00
|
|
|
Context() {}
|
2012-10-10 13:57:15 +02:00
|
|
|
explicit Context(Editor& editor)
|
2012-08-15 17:32:46 +02:00
|
|
|
: m_editor(&editor) {}
|
2013-01-28 13:48:34 +01:00
|
|
|
Context(InputHandler& input_handler, UserInterface& ui)
|
|
|
|
: m_input_handler(&input_handler), m_ui(&ui) {}
|
2012-10-17 13:14:03 +02:00
|
|
|
Context(InputHandler& input_handler, Editor& editor, UserInterface& ui)
|
|
|
|
: m_input_handler(&input_handler), m_editor(&editor), m_ui(&ui) {}
|
2012-08-15 22:36:45 +02:00
|
|
|
|
2012-08-05 20:12:43 +02:00
|
|
|
// to allow func(Context(Editor(...)))
|
2012-10-10 13:57:15 +02:00
|
|
|
// make sure the context will not survive the next ';'
|
|
|
|
explicit Context(Editor&& editor)
|
2012-08-15 17:32:46 +02:00
|
|
|
: m_editor(&editor) {}
|
2012-01-15 22:33:35 +01:00
|
|
|
|
2012-09-26 14:27:23 +02:00
|
|
|
Context(const Context&) = delete;
|
2012-10-16 17:15:09 +02:00
|
|
|
Context(Context&&) = default;
|
2012-08-15 22:36:45 +02:00
|
|
|
Context& operator=(const Context&) = delete;
|
|
|
|
|
2012-01-15 22:33:35 +01:00
|
|
|
Buffer& buffer() const
|
|
|
|
{
|
2012-08-05 18:23:37 +02:00
|
|
|
if (not has_buffer())
|
2012-01-15 22:33:35 +01:00
|
|
|
throw runtime_error("no buffer in context");
|
2012-08-15 17:32:46 +02:00
|
|
|
return m_editor->buffer();
|
2012-01-15 22:33:35 +01:00
|
|
|
}
|
2012-10-02 14:18:34 +02:00
|
|
|
bool has_buffer() const { return (bool)m_editor; }
|
2012-01-31 14:38:06 +01:00
|
|
|
|
2012-08-05 18:23:37 +02:00
|
|
|
Editor& editor() const
|
|
|
|
{
|
|
|
|
if (not has_editor())
|
|
|
|
throw runtime_error("no editor in context");
|
|
|
|
return *m_editor.get();
|
|
|
|
}
|
2012-10-02 14:18:34 +02:00
|
|
|
bool has_editor() const { return (bool)m_editor; }
|
2012-08-05 18:23:37 +02:00
|
|
|
|
2012-01-15 22:33:35 +01:00
|
|
|
Window& window() const
|
|
|
|
{
|
2012-08-05 18:23:37 +02:00
|
|
|
if (not has_window())
|
2012-01-15 22:33:35 +01:00
|
|
|
throw runtime_error("no window in context");
|
2012-08-05 18:23:37 +02:00
|
|
|
return *dynamic_cast<Window*>(m_editor.get());
|
2012-01-15 22:33:35 +01:00
|
|
|
}
|
2012-10-02 14:18:34 +02:00
|
|
|
bool has_window() const { return (bool)m_editor and dynamic_cast<Window*>(m_editor.get()); }
|
2012-06-28 14:01:37 +02:00
|
|
|
|
2012-10-17 13:14:03 +02:00
|
|
|
InputHandler& input_handler() const
|
2012-08-15 22:36:45 +02:00
|
|
|
{
|
2012-10-17 13:14:03 +02:00
|
|
|
if (not has_input_handler())
|
|
|
|
throw runtime_error("no input handler in context");
|
|
|
|
return *m_input_handler;
|
2012-08-15 22:36:45 +02:00
|
|
|
}
|
2012-10-17 13:14:03 +02:00
|
|
|
bool has_input_handler() const { return (bool)m_input_handler; }
|
2012-08-15 22:36:45 +02:00
|
|
|
|
2012-09-26 14:13:04 +02:00
|
|
|
UserInterface& ui() const
|
|
|
|
{
|
|
|
|
if (not has_ui())
|
|
|
|
throw runtime_error("no user interface in context");
|
|
|
|
return *m_ui;
|
|
|
|
}
|
2012-10-02 14:18:34 +02:00
|
|
|
bool has_ui() const { return (bool)m_ui; }
|
2012-09-26 14:13:04 +02:00
|
|
|
|
2012-08-15 22:36:45 +02:00
|
|
|
void change_editor(Editor& editor)
|
|
|
|
{
|
|
|
|
m_editor.reset(&editor);
|
2012-11-19 13:37:38 +01:00
|
|
|
if (has_window() && has_ui())
|
|
|
|
window().set_dimensions(ui().dimensions());
|
2012-08-15 22:36:45 +02:00
|
|
|
}
|
|
|
|
|
2012-11-22 13:50:29 +01:00
|
|
|
OptionManager& options() const
|
2012-06-28 14:01:37 +02:00
|
|
|
{
|
2012-08-05 18:23:37 +02:00
|
|
|
if (has_window())
|
2012-11-22 13:50:29 +01:00
|
|
|
return window().options();
|
2012-08-05 18:23:37 +02:00
|
|
|
if (has_buffer())
|
2012-11-22 13:50:29 +01:00
|
|
|
return buffer().options();
|
|
|
|
return GlobalOptions::instance();
|
2012-06-28 14:01:37 +02:00
|
|
|
}
|
|
|
|
|
2013-01-14 19:26:44 +01:00
|
|
|
HookManager& hooks() const
|
|
|
|
{
|
|
|
|
if (has_window())
|
|
|
|
return window().hooks();
|
|
|
|
if (has_buffer())
|
|
|
|
return buffer().hooks();
|
|
|
|
return GlobalHooks::instance();
|
|
|
|
}
|
|
|
|
|
2012-08-15 22:36:45 +02:00
|
|
|
void print_status(const String& status) const
|
|
|
|
{
|
2012-09-26 14:13:04 +02:00
|
|
|
if (has_ui())
|
|
|
|
ui().print_status(status);
|
2012-08-15 22:36:45 +02:00
|
|
|
}
|
|
|
|
|
2012-09-26 20:07:06 +02:00
|
|
|
using Insertion = std::pair<InsertMode, std::vector<Key>>;
|
|
|
|
Insertion& last_insert() { return m_last_insert; }
|
2012-08-05 16:46:10 +02:00
|
|
|
|
2012-11-12 19:59:25 +01:00
|
|
|
void push_jump()
|
|
|
|
{
|
2012-12-11 19:51:59 +01:00
|
|
|
const SelectionList& jump = editor().selections();
|
2012-11-12 19:59:25 +01:00
|
|
|
if (m_current_jump != m_jump_list.end())
|
|
|
|
{
|
2012-11-12 20:41:03 +01:00
|
|
|
auto begin = m_current_jump;
|
2012-12-11 19:51:59 +01:00
|
|
|
if (&editor().buffer() != &begin->buffer() or
|
|
|
|
(const SelectionList&)(*begin) != jump)
|
2012-11-12 20:41:03 +01:00
|
|
|
++begin;
|
|
|
|
m_jump_list.erase(begin, m_jump_list.end());
|
2012-11-12 19:59:25 +01:00
|
|
|
}
|
|
|
|
|
2012-12-11 19:51:59 +01:00
|
|
|
m_jump_list.push_back({editor().buffer(), jump});
|
2012-11-12 19:59:25 +01:00
|
|
|
m_current_jump = m_jump_list.end();
|
|
|
|
}
|
|
|
|
|
2012-12-11 19:51:59 +01:00
|
|
|
const SelectionList& jump_forward()
|
2012-11-12 19:59:25 +01:00
|
|
|
{
|
|
|
|
if (m_current_jump != m_jump_list.end() and
|
|
|
|
m_current_jump + 1 != m_jump_list.end())
|
|
|
|
return *++m_current_jump;
|
|
|
|
throw runtime_error("no next jump");
|
|
|
|
}
|
|
|
|
|
2012-12-11 19:51:59 +01:00
|
|
|
const SelectionList& jump_backward()
|
2012-11-12 19:59:25 +01:00
|
|
|
{
|
|
|
|
if (m_current_jump != m_jump_list.begin())
|
|
|
|
{
|
|
|
|
if (m_current_jump == m_jump_list.end())
|
|
|
|
{
|
|
|
|
push_jump();
|
|
|
|
--m_current_jump;
|
|
|
|
}
|
|
|
|
return *--m_current_jump;
|
|
|
|
}
|
|
|
|
throw runtime_error("no previous jump");
|
|
|
|
}
|
|
|
|
|
|
|
|
void forget_jumps_to_buffer(Buffer& buffer)
|
|
|
|
{
|
|
|
|
for (auto it = m_jump_list.begin(); it != m_jump_list.end();)
|
|
|
|
{
|
2012-12-11 19:51:59 +01:00
|
|
|
if (&it->buffer() == &buffer)
|
2012-11-12 19:59:25 +01:00
|
|
|
{
|
|
|
|
if (it < m_current_jump)
|
|
|
|
--m_current_jump;
|
|
|
|
else if (it == m_current_jump)
|
2012-11-12 20:41:03 +01:00
|
|
|
m_current_jump = m_jump_list.end()-1;
|
2012-11-12 19:59:25 +01:00
|
|
|
|
|
|
|
it = m_jump_list.erase(it);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
++it;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2012-09-26 20:07:06 +02:00
|
|
|
int& numeric_param() { return m_numeric_param; }
|
|
|
|
private:
|
2012-09-26 14:13:04 +02:00
|
|
|
safe_ptr<Editor> m_editor;
|
2013-01-28 13:48:34 +01:00
|
|
|
InputHandler* m_input_handler;
|
2012-09-26 14:13:04 +02:00
|
|
|
safe_ptr<UserInterface> m_ui;
|
2012-09-26 20:07:06 +02:00
|
|
|
|
|
|
|
Insertion m_last_insert = {InsertMode::Insert, {}};
|
2012-08-05 16:46:10 +02:00
|
|
|
int m_numeric_param = 0;
|
2012-11-12 19:59:25 +01:00
|
|
|
|
2012-12-11 19:51:59 +01:00
|
|
|
using JumpList = std::vector<DynamicSelectionList>;
|
|
|
|
JumpList m_jump_list;
|
|
|
|
JumpList::iterator m_current_jump = m_jump_list.begin();
|
2011-11-26 19:32:57 +01:00
|
|
|
};
|
|
|
|
|
|
|
|
}
|
|
|
|
#endif // context_hh_INCLUDED
|