123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209 |
- /* $OpenBSD$ */
- /*
- * Copyright (c) 2012 Nicholas Marriott <nicholas.marriott@gmail.com>
- * 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>
- #include <stdlib.h>
- #include "tmux.h"
- #define CONTROL_SHOULD_NOTIFY_CLIENT(c) \
- ((c) != NULL && ((c)->flags & CLIENT_CONTROL))
- void
- control_notify_input(struct client *c, struct window_pane *wp,
- struct evbuffer *input)
- {
- u_char *buf;
- size_t len;
- struct evbuffer *message;
- u_int i;
- if (c->session == NULL)
- return;
- buf = EVBUFFER_DATA(input);
- len = EVBUFFER_LENGTH(input);
- /*
- * 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();
- evbuffer_add_printf(message, "%%output %%%u ", wp->id);
- 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]);
- }
- control_write_buffer(c, message);
- evbuffer_free(message);
- }
- }
- void
- control_notify_window_layout_changed(struct window *w)
- {
- struct client *c;
- struct session *s;
- struct format_tree *ft;
- struct winlink *wl;
- const char *template;
- char *expanded;
- template = "%layout-change #{window_id} #{window_layout} "
- "#{window_visible_layout} #{window_flags}";
- TAILQ_FOREACH(c, &clients, entry) {
- 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;
- ft = format_create(NULL, 0);
- wl = winlink_find_by_window(&s->windows, w);
- if (wl != NULL) {
- format_defaults(ft, c, NULL, wl, NULL);
- expanded = format_expand(ft, template);
- control_write(c, "%s", expanded);
- free(expanded);
- }
- format_free(ft);
- }
- }
- void
- control_notify_window_unlinked(__unused struct session *s, struct window *w)
- {
- struct client *c;
- struct session *cs;
- TAILQ_FOREACH(c, &clients, entry) {
- if (!CONTROL_SHOULD_NOTIFY_CLIENT(c) || c->session == NULL)
- continue;
- cs = c->session;
- 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);
- }
- }
- void
- control_notify_window_linked(__unused struct session *s, struct window *w)
- {
- struct client *c;
- struct session *cs;
- TAILQ_FOREACH(c, &clients, entry) {
- if (!CONTROL_SHOULD_NOTIFY_CLIENT(c) || c->session == NULL)
- continue;
- cs = c->session;
- if (winlink_find_by_window_id(&cs->windows, w->id) != NULL)
- control_write(c, "%%window-add @%u", w->id);
- else
- control_write(c, "%%unlinked-window-add @%u", w->id);
- }
- }
- void
- control_notify_window_renamed(struct window *w)
- {
- struct client *c;
- struct session *cs;
- TAILQ_FOREACH(c, &clients, entry) {
- if (!CONTROL_SHOULD_NOTIFY_CLIENT(c) || c->session == NULL)
- continue;
- cs = c->session;
- 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);
- }
- }
- }
- void
- control_notify_attached_session_changed(struct client *c)
- {
- struct session *s;
- if (!CONTROL_SHOULD_NOTIFY_CLIENT(c) || c->session == NULL)
- return;
- s = c->session;
- control_write(c, "%%session-changed $%u %s", s->id, s->name);
- }
- void
- control_notify_session_renamed(struct session *s)
- {
- struct client *c;
- TAILQ_FOREACH(c, &clients, entry) {
- if (!CONTROL_SHOULD_NOTIFY_CLIENT(c))
- continue;
- control_write(c, "%%session-renamed $%u %s", s->id, s->name);
- }
- }
- void
- control_notify_session_created(__unused struct session *s)
- {
- struct client *c;
- TAILQ_FOREACH(c, &clients, entry) {
- if (!CONTROL_SHOULD_NOTIFY_CLIENT(c))
- continue;
- control_write(c, "%%sessions-changed");
- }
- }
- void
- control_notify_session_close(__unused struct session *s)
- {
- struct client *c;
- TAILQ_FOREACH(c, &clients, entry) {
- if (!CONTROL_SHOULD_NOTIFY_CLIENT(c))
- continue;
- control_write(c, "%%sessions-changed");
- }
- }
|