2012-09-03 09:32:38 +00:00
|
|
|
/* $OpenBSD$ */
|
|
|
|
|
|
|
|
/*
|
2016-01-19 15:59:12 +00:00
|
|
|
* Copyright (c) 2012 Nicholas Marriott <nicholas.marriott@gmail.com>
|
2012-09-03 09:32:38 +00:00
|
|
|
* Copyright (c) 2012 George Nachman <tmux@georgester.com>
|
|
|
|
*
|
|
|
|
* Permission to use, copy, modify, and distribute this software for any
|
|
|
|
* purpose with or without fee is hereby granted, provided that the above
|
|
|
|
* copyright notice and this permission notice appear in all copies.
|
|
|
|
*
|
|
|
|
* THE SOFTWARE IS PROVIDED "AS IS" AND THE AUTHOR DISCLAIMS ALL WARRANTIES
|
|
|
|
* WITH REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF
|
|
|
|
* MERCHANTABILITY AND FITNESS. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR
|
|
|
|
* ANY SPECIAL, DIRECT, INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES
|
|
|
|
* WHATSOEVER RESULTING FROM LOSS OF MIND, USE, DATA OR PROFITS, WHETHER
|
|
|
|
* IN AN ACTION OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING
|
|
|
|
* OUT OF OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE.
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include <sys/types.h>
|
|
|
|
|
2015-11-13 10:00:26 +00:00
|
|
|
#include <stdlib.h>
|
|
|
|
|
2012-09-03 09:32:38 +00:00
|
|
|
#include "tmux.h"
|
|
|
|
|
|
|
|
#define CONTROL_SHOULD_NOTIFY_CLIENT(c) \
|
|
|
|
((c) != NULL && ((c)->flags & CLIENT_CONTROL))
|
|
|
|
|
2012-09-25 07:41:22 +00:00
|
|
|
void
|
|
|
|
control_notify_input(struct client *c, struct window_pane *wp,
|
2019-05-07 10:25:15 +00:00
|
|
|
const u_char *buf, size_t len)
|
2012-09-25 07:41:22 +00:00
|
|
|
{
|
|
|
|
struct evbuffer *message;
|
|
|
|
u_int i;
|
|
|
|
|
|
|
|
if (c->session == NULL)
|
|
|
|
return;
|
|
|
|
|
2019-07-10 11:20:10 +00:00
|
|
|
if (c->flags & CLIENT_CONTROL_NOOUTPUT)
|
|
|
|
return;
|
|
|
|
|
2012-09-25 07:41:22 +00:00
|
|
|
/*
|
|
|
|
* Only write input if the window pane is linked to a window belonging
|
|
|
|
* to the client's session.
|
|
|
|
*/
|
|
|
|
if (winlink_find_by_window(&c->session->windows, wp->window) != NULL) {
|
|
|
|
message = evbuffer_new();
|
2018-11-19 13:35:40 +00:00
|
|
|
if (message == NULL)
|
|
|
|
fatalx("out of memory");
|
2012-09-25 07:41:22 +00:00
|
|
|
evbuffer_add_printf(message, "%%output %%%u ", wp->id);
|
2013-03-25 11:40:54 +00:00
|
|
|
for (i = 0; i < len; i++) {
|
|
|
|
if (buf[i] < ' ' || buf[i] == '\\')
|
|
|
|
evbuffer_add_printf(message, "\\%03o", buf[i]);
|
|
|
|
else
|
|
|
|
evbuffer_add_printf(message, "%c", buf[i]);
|
|
|
|
}
|
2020-03-16 09:12:44 +00:00
|
|
|
evbuffer_add(message, "", 1);
|
2019-12-12 11:39:56 +00:00
|
|
|
control_write(c, "%s", EVBUFFER_DATA(message));
|
2012-09-25 07:41:22 +00:00
|
|
|
evbuffer_free(message);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-05-04 07:16:43 +00:00
|
|
|
void
|
|
|
|
control_notify_pane_mode_changed(int pane)
|
|
|
|
{
|
|
|
|
struct client *c;
|
|
|
|
|
|
|
|
TAILQ_FOREACH(c, &clients, entry) {
|
|
|
|
if (!CONTROL_SHOULD_NOTIFY_CLIENT(c))
|
|
|
|
continue;
|
|
|
|
|
|
|
|
control_write(c, "%%pane-mode-changed %%%u", pane);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2012-09-03 09:32:38 +00:00
|
|
|
void
|
|
|
|
control_notify_window_layout_changed(struct window *w)
|
|
|
|
{
|
2017-05-04 07:16:43 +00:00
|
|
|
struct client *c;
|
|
|
|
struct session *s;
|
|
|
|
struct winlink *wl;
|
|
|
|
const char *template;
|
|
|
|
char *cp;
|
2015-11-13 10:00:26 +00:00
|
|
|
|
|
|
|
template = "%layout-change #{window_id} #{window_layout} "
|
|
|
|
"#{window_visible_layout} #{window_flags}";
|
2012-09-03 09:32:38 +00:00
|
|
|
|
2015-04-24 23:17:11 +00:00
|
|
|
TAILQ_FOREACH(c, &clients, entry) {
|
2012-09-03 09:32:38 +00:00
|
|
|
if (!CONTROL_SHOULD_NOTIFY_CLIENT(c) || c->session == NULL)
|
|
|
|
continue;
|
|
|
|
s = c->session;
|
|
|
|
|
|
|
|
if (winlink_find_by_window_id(&s->windows, w->id) == NULL)
|
|
|
|
continue;
|
|
|
|
|
|
|
|
/*
|
|
|
|
* When the last pane in a window is closed it won't have a
|
|
|
|
* layout root and we don't need to inform the client about the
|
|
|
|
* layout change because the whole window will go away soon.
|
|
|
|
*/
|
|
|
|
if (w->layout_root == NULL)
|
|
|
|
continue;
|
|
|
|
|
|
|
|
wl = winlink_find_by_window(&s->windows, w);
|
|
|
|
if (wl != NULL) {
|
2017-03-08 13:36:12 +00:00
|
|
|
cp = format_single(NULL, template, c, NULL, wl, NULL);
|
|
|
|
control_write(c, "%s", cp);
|
|
|
|
free(cp);
|
2012-09-03 09:32:38 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-05-04 07:16:43 +00:00
|
|
|
void
|
|
|
|
control_notify_window_pane_changed(struct window *w)
|
|
|
|
{
|
|
|
|
struct client *c;
|
|
|
|
|
|
|
|
TAILQ_FOREACH(c, &clients, entry) {
|
|
|
|
if (!CONTROL_SHOULD_NOTIFY_CLIENT(c))
|
|
|
|
continue;
|
|
|
|
|
|
|
|
control_write(c, "%%window-pane-changed @%u %%%u", w->id,
|
|
|
|
w->active->id);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2012-09-03 09:32:38 +00:00
|
|
|
void
|
2015-11-18 14:27:44 +00:00
|
|
|
control_notify_window_unlinked(__unused struct session *s, struct window *w)
|
2012-09-03 09:32:38 +00:00
|
|
|
{
|
|
|
|
struct client *c;
|
2014-04-23 10:14:29 +00:00
|
|
|
struct session *cs;
|
2012-09-03 09:32:38 +00:00
|
|
|
|
2015-04-24 23:17:11 +00:00
|
|
|
TAILQ_FOREACH(c, &clients, entry) {
|
2012-09-03 09:32:38 +00:00
|
|
|
if (!CONTROL_SHOULD_NOTIFY_CLIENT(c) || c->session == NULL)
|
|
|
|
continue;
|
2014-04-23 10:14:29 +00:00
|
|
|
cs = c->session;
|
2012-09-03 09:32:38 +00:00
|
|
|
|
2014-04-23 10:14:29 +00:00
|
|
|
if (winlink_find_by_window_id(&cs->windows, w->id) != NULL)
|
|
|
|
control_write(c, "%%window-close @%u", w->id);
|
|
|
|
else
|
|
|
|
control_write(c, "%%unlinked-window-close @%u", w->id);
|
2012-09-03 09:32:38 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
2015-11-18 14:27:44 +00:00
|
|
|
control_notify_window_linked(__unused struct session *s, struct window *w)
|
2012-09-03 09:32:38 +00:00
|
|
|
{
|
|
|
|
struct client *c;
|
|
|
|
struct session *cs;
|
|
|
|
|
2015-04-24 23:17:11 +00:00
|
|
|
TAILQ_FOREACH(c, &clients, entry) {
|
2012-09-03 09:32:38 +00:00
|
|
|
if (!CONTROL_SHOULD_NOTIFY_CLIENT(c) || c->session == NULL)
|
|
|
|
continue;
|
|
|
|
cs = c->session;
|
|
|
|
|
|
|
|
if (winlink_find_by_window_id(&cs->windows, w->id) != NULL)
|
2013-03-25 11:40:54 +00:00
|
|
|
control_write(c, "%%window-add @%u", w->id);
|
2012-09-03 09:32:38 +00:00
|
|
|
else
|
2013-03-25 11:40:54 +00:00
|
|
|
control_write(c, "%%unlinked-window-add @%u", w->id);
|
2012-09-03 09:32:38 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
control_notify_window_renamed(struct window *w)
|
|
|
|
{
|
|
|
|
struct client *c;
|
2014-04-23 10:14:29 +00:00
|
|
|
struct session *cs;
|
2012-09-03 09:32:38 +00:00
|
|
|
|
2015-04-24 23:17:11 +00:00
|
|
|
TAILQ_FOREACH(c, &clients, entry) {
|
2012-09-03 09:32:38 +00:00
|
|
|
if (!CONTROL_SHOULD_NOTIFY_CLIENT(c) || c->session == NULL)
|
|
|
|
continue;
|
2014-04-23 10:14:29 +00:00
|
|
|
cs = c->session;
|
2012-09-03 09:32:38 +00:00
|
|
|
|
2014-04-23 10:14:29 +00:00
|
|
|
if (winlink_find_by_window_id(&cs->windows, w->id) != NULL) {
|
|
|
|
control_write(c, "%%window-renamed @%u %s", w->id,
|
|
|
|
w->name);
|
|
|
|
} else {
|
|
|
|
control_write(c, "%%unlinked-window-renamed @%u %s",
|
|
|
|
w->id, w->name);
|
|
|
|
}
|
2012-09-03 09:32:38 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
2017-05-04 07:16:43 +00:00
|
|
|
control_notify_client_session_changed(struct client *cc)
|
2012-09-03 09:32:38 +00:00
|
|
|
{
|
2017-05-04 07:16:43 +00:00
|
|
|
struct client *c;
|
2012-09-03 09:32:38 +00:00
|
|
|
struct session *s;
|
|
|
|
|
2017-05-04 07:16:43 +00:00
|
|
|
if (cc->session == NULL)
|
2012-09-03 09:32:38 +00:00
|
|
|
return;
|
2017-05-04 07:16:43 +00:00
|
|
|
s = cc->session;
|
2012-09-03 09:32:38 +00:00
|
|
|
|
2017-05-04 07:16:43 +00:00
|
|
|
TAILQ_FOREACH(c, &clients, entry) {
|
|
|
|
if (!CONTROL_SHOULD_NOTIFY_CLIENT(c) || c->session == NULL)
|
|
|
|
continue;
|
|
|
|
|
|
|
|
if (cc == c) {
|
|
|
|
control_write(c, "%%session-changed $%u %s", s->id,
|
|
|
|
s->name);
|
|
|
|
} else {
|
|
|
|
control_write(c, "%%client-session-changed %s $%u %s",
|
|
|
|
cc->name, s->id, s->name);
|
|
|
|
}
|
|
|
|
}
|
2012-09-03 09:32:38 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
control_notify_session_renamed(struct session *s)
|
|
|
|
{
|
|
|
|
struct client *c;
|
|
|
|
|
2015-04-24 23:17:11 +00:00
|
|
|
TAILQ_FOREACH(c, &clients, entry) {
|
2013-03-25 11:40:54 +00:00
|
|
|
if (!CONTROL_SHOULD_NOTIFY_CLIENT(c))
|
2012-09-03 09:32:38 +00:00
|
|
|
continue;
|
|
|
|
|
2013-03-25 11:40:54 +00:00
|
|
|
control_write(c, "%%session-renamed $%u %s", s->id, s->name);
|
2012-09-03 09:32:38 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
2015-11-18 14:27:44 +00:00
|
|
|
control_notify_session_created(__unused struct session *s)
|
2012-09-03 09:32:38 +00:00
|
|
|
{
|
|
|
|
struct client *c;
|
|
|
|
|
2015-04-24 23:17:11 +00:00
|
|
|
TAILQ_FOREACH(c, &clients, entry) {
|
2013-03-25 11:40:54 +00:00
|
|
|
if (!CONTROL_SHOULD_NOTIFY_CLIENT(c))
|
2012-09-03 09:32:38 +00:00
|
|
|
continue;
|
|
|
|
|
|
|
|
control_write(c, "%%sessions-changed");
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
2016-10-15 00:12:58 +00:00
|
|
|
control_notify_session_closed(__unused struct session *s)
|
2012-09-03 09:32:38 +00:00
|
|
|
{
|
|
|
|
struct client *c;
|
|
|
|
|
2015-04-24 23:17:11 +00:00
|
|
|
TAILQ_FOREACH(c, &clients, entry) {
|
2013-03-25 11:40:54 +00:00
|
|
|
if (!CONTROL_SHOULD_NOTIFY_CLIENT(c))
|
2012-09-03 09:32:38 +00:00
|
|
|
continue;
|
|
|
|
|
|
|
|
control_write(c, "%%sessions-changed");
|
|
|
|
}
|
|
|
|
}
|
2017-05-04 07:16:43 +00:00
|
|
|
|
|
|
|
void
|
|
|
|
control_notify_session_window_changed(struct session *s)
|
|
|
|
{
|
|
|
|
struct client *c;
|
|
|
|
|
|
|
|
TAILQ_FOREACH(c, &clients, entry) {
|
|
|
|
if (!CONTROL_SHOULD_NOTIFY_CLIENT(c))
|
|
|
|
continue;
|
|
|
|
|
|
|
|
control_write(c, "%%session-window-changed $%u @%u", s->id,
|
|
|
|
s->curw->window->id);
|
|
|
|
}
|
|
|
|
}
|