2012-10-30 14:00:44 +01:00
|
|
|
#include "client_manager.hh"
|
|
|
|
|
2012-11-07 14:02:23 +01:00
|
|
|
#include "buffer_manager.hh"
|
2013-04-09 19:39:03 +02:00
|
|
|
#include "color_registry.hh"
|
2012-12-18 21:20:36 +01:00
|
|
|
#include "command_manager.hh"
|
2013-04-09 19:39:03 +02:00
|
|
|
#include "event_manager.hh"
|
2013-03-25 19:11:26 +01:00
|
|
|
#include "file.hh"
|
2013-04-09 19:39:03 +02:00
|
|
|
#include "user_interface.hh"
|
|
|
|
#include "window.hh"
|
2012-10-31 14:23:44 +01:00
|
|
|
|
2012-10-30 14:00:44 +01:00
|
|
|
namespace Kakoune
|
|
|
|
{
|
|
|
|
|
2013-02-07 19:25:42 +01:00
|
|
|
ClientManager::ClientManager() = default;
|
|
|
|
ClientManager::~ClientManager() = default;
|
|
|
|
|
2013-01-07 13:59:09 +01:00
|
|
|
String ClientManager::generate_name() const
|
|
|
|
{
|
|
|
|
for (int i = 0; true; ++i)
|
|
|
|
{
|
2013-05-13 14:23:07 +02:00
|
|
|
String name = "unnamed" + to_string(i);
|
2013-01-07 13:59:09 +01:00
|
|
|
bool found = false;
|
|
|
|
for (auto& client : m_clients)
|
|
|
|
{
|
2013-04-15 14:28:21 +02:00
|
|
|
if (client->m_name == name)
|
2013-01-07 13:59:09 +01:00
|
|
|
{
|
|
|
|
found = true;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if (not found)
|
|
|
|
return name;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2013-09-12 23:47:23 +02:00
|
|
|
Client* ClientManager::create_client(std::unique_ptr<UserInterface>&& ui,
|
|
|
|
const String& init_commands)
|
2012-10-30 14:00:44 +01:00
|
|
|
{
|
2012-12-28 13:51:14 +01:00
|
|
|
Buffer& buffer = **BufferManager::instance().begin();
|
2013-09-12 23:47:23 +02:00
|
|
|
Client* client = new Client{std::move(ui), get_unused_window_for_buffer(buffer),
|
2013-09-12 23:39:34 +02:00
|
|
|
generate_name()};
|
2013-04-15 14:28:21 +02:00
|
|
|
m_clients.emplace_back(client);
|
2012-12-19 18:56:47 +01:00
|
|
|
try
|
|
|
|
{
|
2013-04-15 14:28:21 +02:00
|
|
|
CommandManager::instance().execute(init_commands, client->context());
|
2012-12-19 18:56:47 +01:00
|
|
|
}
|
|
|
|
catch (Kakoune::runtime_error& error)
|
|
|
|
{
|
2013-04-15 14:28:21 +02:00
|
|
|
client->context().print_status({ error.what(), get_color("Error") });
|
|
|
|
client->context().hooks().run_hook("RuntimeError", error.what(),
|
|
|
|
client->context());
|
2012-12-19 18:56:47 +01:00
|
|
|
}
|
|
|
|
catch (Kakoune::client_removed&)
|
|
|
|
{
|
|
|
|
m_clients.pop_back();
|
2013-04-15 14:28:21 +02:00
|
|
|
return nullptr;
|
2012-12-19 18:56:47 +01:00
|
|
|
}
|
|
|
|
|
2013-09-12 23:39:34 +02:00
|
|
|
client->ui().set_input_callback([client, this]() {
|
2012-10-31 14:23:44 +01:00
|
|
|
try
|
|
|
|
{
|
2013-09-12 23:39:34 +02:00
|
|
|
while (client->ui().is_key_available())
|
|
|
|
client->handle_key(client->ui().get_key());
|
2013-04-15 14:28:21 +02:00
|
|
|
client->context().window().forget_timestamp();
|
2012-10-31 14:23:44 +01:00
|
|
|
}
|
|
|
|
catch (Kakoune::runtime_error& error)
|
|
|
|
{
|
2013-04-15 14:28:21 +02:00
|
|
|
client->context().print_status({ error.what(), get_color("Error") });
|
|
|
|
client->context().hooks().run_hook("RuntimeError", error.what(),
|
|
|
|
client->context());
|
2012-10-31 14:23:44 +01:00
|
|
|
}
|
|
|
|
catch (Kakoune::client_removed&)
|
|
|
|
{
|
2013-04-15 14:28:21 +02:00
|
|
|
ClientManager::instance().remove_client(*client);
|
2012-10-31 14:23:44 +01:00
|
|
|
}
|
2012-11-05 19:58:04 +01:00
|
|
|
ClientManager::instance().redraw_clients();
|
2012-10-31 14:23:44 +01:00
|
|
|
});
|
2012-11-05 19:58:04 +01:00
|
|
|
redraw_clients();
|
2013-04-15 14:28:21 +02:00
|
|
|
|
|
|
|
return client;
|
2012-10-30 14:00:44 +01:00
|
|
|
}
|
|
|
|
|
2013-09-12 23:47:23 +02:00
|
|
|
void ClientManager::remove_client(Client& client)
|
2012-10-30 14:00:44 +01:00
|
|
|
{
|
|
|
|
for (auto it = m_clients.begin(); it != m_clients.end(); ++it)
|
|
|
|
{
|
2013-04-15 14:28:21 +02:00
|
|
|
if (it->get() == &client)
|
2012-10-30 14:00:44 +01:00
|
|
|
{
|
|
|
|
m_clients.erase(it);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
}
|
2013-04-09 20:04:11 +02:00
|
|
|
kak_assert(false);
|
2012-10-30 14:00:44 +01:00
|
|
|
}
|
|
|
|
|
2012-11-22 14:08:55 +01:00
|
|
|
Window& ClientManager::get_unused_window_for_buffer(Buffer& buffer)
|
2012-11-05 19:15:42 +01:00
|
|
|
{
|
2012-11-22 14:08:55 +01:00
|
|
|
for (auto& w : m_windows)
|
2012-11-05 19:15:42 +01:00
|
|
|
{
|
2012-11-22 14:08:55 +01:00
|
|
|
if (&w->buffer() != &buffer)
|
|
|
|
continue;
|
|
|
|
|
2012-11-05 19:15:42 +01:00
|
|
|
auto it = std::find_if(m_clients.begin(), m_clients.end(),
|
2013-09-12 23:47:23 +02:00
|
|
|
[&](const std::unique_ptr<Client>& client)
|
2013-01-28 13:48:34 +01:00
|
|
|
{ return &client->context().window() == w.get(); });
|
2012-11-05 19:15:42 +01:00
|
|
|
if (it == m_clients.end())
|
2012-11-05 19:58:04 +01:00
|
|
|
{
|
|
|
|
w->forget_timestamp();
|
2012-11-05 19:15:42 +01:00
|
|
|
return *w;
|
2012-11-05 19:58:04 +01:00
|
|
|
}
|
2012-11-05 19:15:42 +01:00
|
|
|
}
|
2012-11-22 14:08:55 +01:00
|
|
|
m_windows.emplace_back(new Window(buffer));
|
|
|
|
return *m_windows.back();
|
2012-11-05 19:15:42 +01:00
|
|
|
}
|
|
|
|
|
2012-11-07 14:02:23 +01:00
|
|
|
void ClientManager::ensure_no_client_uses_buffer(Buffer& buffer)
|
|
|
|
{
|
|
|
|
for (auto& client : m_clients)
|
|
|
|
{
|
2013-01-28 13:48:34 +01:00
|
|
|
client->context().forget_jumps_to_buffer(buffer);
|
2012-11-12 19:59:25 +01:00
|
|
|
|
2013-01-28 13:48:34 +01:00
|
|
|
if (&client->context().buffer() != &buffer)
|
2012-11-07 14:02:23 +01:00
|
|
|
continue;
|
|
|
|
|
2013-04-10 18:54:01 +02:00
|
|
|
if (client->context().editor().is_editing())
|
2013-04-15 14:28:21 +02:00
|
|
|
throw runtime_error("client '" + client->m_name + "' is inserting in '" +
|
2013-04-10 18:54:01 +02:00
|
|
|
buffer.display_name() + '\'');
|
|
|
|
|
2012-11-07 14:02:23 +01:00
|
|
|
// change client context to edit the first buffer which is not the
|
|
|
|
// specified one. As BufferManager stores buffer according to last
|
|
|
|
// access, this selects a sensible buffer to display.
|
|
|
|
for (auto& buf : BufferManager::instance())
|
|
|
|
{
|
|
|
|
if (buf != &buffer)
|
|
|
|
{
|
|
|
|
Window& w = get_unused_window_for_buffer(*buf);
|
2013-01-28 13:48:34 +01:00
|
|
|
client->context().change_editor(w);
|
2012-11-07 14:02:23 +01:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2012-11-22 14:08:55 +01:00
|
|
|
auto end = std::remove_if(m_windows.begin(), m_windows.end(),
|
2012-11-22 14:17:46 +01:00
|
|
|
[&buffer](const std::unique_ptr<Window>& w)
|
2012-11-22 14:08:55 +01:00
|
|
|
{ return &w->buffer() == &buffer; });
|
|
|
|
m_windows.erase(end, m_windows.end());
|
2012-11-07 14:02:23 +01:00
|
|
|
}
|
|
|
|
|
2013-09-12 23:47:23 +02:00
|
|
|
void ClientManager::set_client_name(Client& client, String name)
|
2012-12-03 18:56:53 +01:00
|
|
|
{
|
2013-09-12 23:47:23 +02:00
|
|
|
auto it = find_if(m_clients, [&name](std::unique_ptr<Client>& client)
|
2013-04-15 14:28:21 +02:00
|
|
|
{ return client->m_name == name; });
|
|
|
|
if (it != m_clients.end() and it->get() != &client)
|
2012-12-03 18:56:53 +01:00
|
|
|
throw runtime_error("name not unique: " + name);
|
2013-04-15 14:28:21 +02:00
|
|
|
client.m_name = std::move(name);
|
|
|
|
}
|
2012-12-03 18:56:53 +01:00
|
|
|
|
2013-09-12 23:47:23 +02:00
|
|
|
Client& ClientManager::get_client(const Context& context)
|
2013-04-15 14:28:21 +02:00
|
|
|
{
|
2012-12-03 18:56:53 +01:00
|
|
|
for (auto& client : m_clients)
|
|
|
|
{
|
2013-01-28 13:48:34 +01:00
|
|
|
if (&client->context() == &context)
|
2013-04-15 14:28:21 +02:00
|
|
|
return *client;
|
2012-12-03 18:56:53 +01:00
|
|
|
}
|
|
|
|
throw runtime_error("no client for current context");
|
|
|
|
}
|
|
|
|
|
2013-09-12 23:47:23 +02:00
|
|
|
Client& ClientManager::get_client(const String& name)
|
2013-01-07 13:59:09 +01:00
|
|
|
{
|
|
|
|
for (auto& client : m_clients)
|
|
|
|
{
|
2013-04-15 14:28:21 +02:00
|
|
|
if (client->m_name == name)
|
|
|
|
return *client;
|
2013-01-07 13:59:09 +01:00
|
|
|
}
|
2012-12-03 18:56:53 +01:00
|
|
|
throw runtime_error("no client named: " + name);
|
|
|
|
}
|
|
|
|
|
2013-09-12 23:47:23 +02:00
|
|
|
static DisplayLine generate_status_line(Client& client)
|
2013-02-18 19:03:39 +01:00
|
|
|
{
|
2013-04-18 14:28:35 +02:00
|
|
|
auto& context = client.context();
|
2013-04-11 13:45:28 +02:00
|
|
|
auto pos = context.editor().main_selection().last();
|
2013-06-05 18:47:39 +02:00
|
|
|
auto col = context.buffer()[pos.line].char_count_to(pos.column);
|
2013-03-25 19:11:26 +01:00
|
|
|
|
2013-04-11 13:45:28 +02:00
|
|
|
std::ostringstream oss;
|
2013-03-25 19:58:23 +01:00
|
|
|
oss << context.buffer().display_name()
|
2013-06-03 18:58:09 +02:00
|
|
|
<< " " << (int)pos.line+1 << ":" << (int)col+1;
|
2013-02-18 19:03:39 +01:00
|
|
|
if (context.buffer().is_modified())
|
|
|
|
oss << " [+]";
|
2013-09-12 23:47:23 +02:00
|
|
|
if (context.client().is_recording())
|
2013-02-18 19:03:39 +01:00
|
|
|
oss << " [recording]";
|
|
|
|
if (context.buffer().flags() & Buffer::Flags::New)
|
|
|
|
oss << " [new file]";
|
|
|
|
oss << " [" << context.editor().selections().size() << " sel]";
|
|
|
|
if (context.editor().is_editing())
|
|
|
|
oss << " [insert]";
|
2013-04-18 14:28:35 +02:00
|
|
|
oss << " - " << client.name();
|
2013-04-04 18:50:00 +02:00
|
|
|
return { oss.str(), get_color("StatusLine") };
|
2013-02-18 19:03:39 +01:00
|
|
|
}
|
|
|
|
|
2012-11-05 19:58:04 +01:00
|
|
|
void ClientManager::redraw_clients() const
|
|
|
|
{
|
|
|
|
for (auto& client : m_clients)
|
|
|
|
{
|
2013-01-28 13:48:34 +01:00
|
|
|
Context& context = client->context();
|
2012-11-05 19:58:04 +01:00
|
|
|
if (context.window().timestamp() != context.buffer().timestamp())
|
|
|
|
{
|
|
|
|
DisplayCoord dimensions = context.ui().dimensions();
|
|
|
|
if (dimensions == DisplayCoord{0,0})
|
|
|
|
return;
|
|
|
|
context.window().set_dimensions(dimensions);
|
|
|
|
context.window().update_display_buffer();;
|
2013-02-18 19:03:39 +01:00
|
|
|
|
2012-11-05 19:58:04 +01:00
|
|
|
context.ui().draw(context.window().display_buffer(),
|
2013-04-18 14:28:35 +02:00
|
|
|
generate_status_line(*client));
|
2012-11-05 19:58:04 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2012-10-30 14:00:44 +01:00
|
|
|
}
|