termite/termite.c

335 lines
11 KiB
C
Raw Normal View History

2012-05-30 10:22:09 +02:00
#include <stdbool.h>
2012-05-31 14:05:25 +02:00
#include <gdk/gdkkeysyms.h>
2012-05-22 06:34:35 +02:00
#include <gtk/gtk.h>
#include <vte/vte.h>
2012-05-22 21:21:54 +02:00
#include "config.h"
2012-05-31 15:38:13 +02:00
#define CONCAT(X, Y) X ## Y
#define KEY(X) CONCAT(GDK_KEY_, X)
2012-05-30 10:22:09 +02:00
#ifndef __GNUC__
# define __attribute__(x)
#endif
typedef struct search_dialog_info {
GtkWidget *vte;
GtkWidget *entry;
bool reverse;
2012-05-30 10:22:09 +02:00
bool open;
} search_dialog_info;
static void search(VteTerminal *vte, const char *pattern, bool reverse) {
2012-05-30 10:22:09 +02:00
GRegex *regex = vte_terminal_search_get_gregex(vte);
if (regex) g_regex_unref(regex);
regex = g_regex_new(pattern, 0, 0, NULL);
vte_terminal_search_set_gregex(vte, regex);
if (!reverse) {
2012-05-30 10:22:09 +02:00
vte_terminal_search_find_next(vte);
} else {
vte_terminal_search_find_previous(vte);
}
2012-05-30 11:27:27 +02:00
vte_terminal_copy_primary(vte);
2012-05-30 10:22:09 +02:00
}
2012-05-31 13:12:39 +02:00
static void search_response_cb(GtkWidget *dialog, gint response_id, search_dialog_info *info) {
2012-05-30 10:22:09 +02:00
if (response_id == GTK_RESPONSE_ACCEPT) {
search(VTE_TERMINAL(info->vte), gtk_entry_get_text(GTK_ENTRY(info->entry)), info->reverse);
2012-05-30 10:22:09 +02:00
}
2012-05-31 13:12:39 +02:00
gtk_widget_destroy(dialog);
2012-05-30 10:22:09 +02:00
info->open = false;
}
2012-05-31 13:49:59 +02:00
static gboolean search_key_press_cb(__attribute__((unused)) GtkEntry *entry, GdkEventKey *event, GtkDialog *dialog) {
if (event->keyval == GDK_KEY_Return) {
gtk_dialog_response(dialog, GTK_RESPONSE_ACCEPT);
return TRUE;
}
return FALSE;
}
static void open_search_dialog(GtkWidget *vte, bool reverse, search_dialog_info *info) {
info->reverse = reverse;
2012-05-30 10:22:09 +02:00
if (info->open) {
return;
}
info->open = true;
info->entry = gtk_entry_new();
GtkWidget *dialog, *content_area;
dialog = gtk_dialog_new_with_buttons("Search",
GTK_WINDOW(gtk_widget_get_toplevel(vte)),
GTK_DIALOG_DESTROY_WITH_PARENT,
2012-05-31 13:49:59 +02:00
NULL,
2012-05-30 10:22:09 +02:00
NULL);
g_signal_connect(dialog, "response", G_CALLBACK(search_response_cb), info);
g_signal_connect(info->entry, "key-press-event", G_CALLBACK(search_key_press_cb), dialog);
2012-05-30 10:22:09 +02:00
content_area = gtk_dialog_get_content_area(GTK_DIALOG(dialog));
2012-05-30 10:22:09 +02:00
gtk_container_add(GTK_CONTAINER(content_area), info->entry);
gtk_widget_show_all(dialog);
gtk_widget_grab_focus(GTK_WIDGET(info->entry));
}
static gboolean key_press_cb(GtkWidget *vte, GdkEventKey *event, search_dialog_info *info) {
const GdkModifierType modifiers = event->state & gtk_accelerator_get_default_mod_mask();
if (modifiers == (GDK_CONTROL_MASK|GDK_SHIFT_MASK)) {
switch (gdk_keyval_to_lower(event->keyval)) {
2012-05-31 15:38:13 +02:00
case KEY(KEY_COPY):
vte_terminal_copy_clipboard(VTE_TERMINAL(vte));
return TRUE;
2012-05-31 15:38:13 +02:00
case KEY(KEY_PASTE):
vte_terminal_paste_clipboard(VTE_TERMINAL(vte));
return TRUE;
2012-05-31 15:38:13 +02:00
case KEY(KEY_PREV):
2012-05-30 10:22:09 +02:00
vte_terminal_search_find_previous(VTE_TERMINAL(vte));
2012-05-30 11:36:14 +02:00
vte_terminal_copy_primary(VTE_TERMINAL(vte));
2012-05-30 10:22:09 +02:00
return TRUE;
2012-05-31 15:38:13 +02:00
case KEY(KEY_NEXT):
2012-05-30 10:22:09 +02:00
vte_terminal_search_find_next(VTE_TERMINAL(vte));
2012-05-30 11:36:14 +02:00
vte_terminal_copy_primary(VTE_TERMINAL(vte));
2012-05-30 10:22:09 +02:00
return TRUE;
2012-05-31 15:38:13 +02:00
case KEY(KEY_SEARCH):
open_search_dialog(vte, false, info);
2012-05-30 10:22:09 +02:00
return TRUE;
2012-05-31 15:38:13 +02:00
case KEY(KEY_RSEARCH):
open_search_dialog(vte, true, info);
2012-05-30 10:22:09 +02:00
return TRUE;
2012-05-31 15:38:13 +02:00
case KEY(KEY_URL):
2012-05-30 11:36:14 +02:00
search(VTE_TERMINAL(vte), url_regex, false);
2012-05-30 11:35:41 +02:00
return TRUE;
2012-05-31 15:38:13 +02:00
case KEY(KEY_RURL):
2012-05-30 11:36:14 +02:00
search(VTE_TERMINAL(vte), url_regex, true);
2012-05-30 11:35:41 +02:00
return TRUE;
}
}
return FALSE;
}
2012-05-24 05:41:55 +02:00
#ifdef CLICKABLE_URL
static void get_vte_padding(VteTerminal *vte, int *w, int *h) {
GtkBorder *border = NULL;
gtk_widget_style_get(GTK_WIDGET(vte), "inner-border", &border, NULL);
2012-05-31 11:32:51 +02:00
if (!border) {
g_warning("VTE's inner-border property unavailable");
*w = *h = 0;
} else {
*w = border->left + border->right;
*h = border->top + border->bottom;
gtk_border_free(border);
}
}
static char *check_match(VteTerminal *vte, int event_x, int event_y) {
int xpad, ypad, tag;
get_vte_padding(vte, &xpad, &ypad);
2012-05-22 13:17:27 +02:00
return vte_terminal_match_check(vte,
(event_x - ypad) / vte_terminal_get_char_width(vte),
(event_y - ypad) / vte_terminal_get_char_height(vte),
&tag);
}
static gboolean button_press_cb(VteTerminal *vte, GdkEventButton *event) {
char *match = check_match(vte, event->x, event->y);
if (event->button == 1 && event->type == GDK_BUTTON_PRESS && match != NULL) {
2012-05-30 19:56:40 +02:00
const char *argv[] = URL_COMMAND(match);
g_spawn_async(NULL, (char **)argv, NULL, 0, NULL, NULL, NULL, NULL);
g_free(match);
return TRUE;
}
return FALSE;
}
2012-05-24 05:41:55 +02:00
#endif
2012-05-22 13:47:51 +02:00
#ifdef URGENT_ON_BEEP
2012-05-22 19:36:51 +02:00
static void beep_handler(__attribute__((unused)) VteTerminal *vte, GtkWindow *window) {
2012-05-30 19:30:00 +02:00
gtk_window_set_urgency_hint(window, TRUE);
}
static gboolean focus_in_handler(GtkWindow *window) {
gtk_window_set_urgency_hint(window, FALSE);
return FALSE;
2012-05-22 13:47:51 +02:00
}
#endif
2012-05-22 14:22:31 +02:00
#ifdef DYNAMIC_TITLE
static void window_title_cb(VteTerminal *vte, GtkWindow *window) {
const char *t = vte_terminal_get_window_title(vte);
2012-05-22 15:50:21 +02:00
gtk_window_set_title(window, t ? t : "termite");
2012-05-22 14:22:31 +02:00
}
#endif
static gboolean position_overlay(GtkOverlay *overlay, GtkWidget *widget, GdkRectangle *alloc) {
GtkWidget *main_widget = gtk_bin_get_child(GTK_BIN(overlay));
GtkAllocation main_alloc;
main_alloc.x = 0;
main_alloc.y = 0;
main_alloc.width = gtk_widget_get_allocated_width(main_widget);
main_alloc.height = gtk_widget_get_allocated_height(main_widget);
GtkRequisition req;
gtk_widget_get_preferred_size(widget, NULL, &req);
alloc->x = 2 * main_alloc.width / 3 - req.width / 2;
alloc->y = 0;
alloc->width = MIN(main_alloc.width, req.width);
alloc->height = MIN(main_alloc.height, req.height);
return TRUE;
}
2012-05-22 06:34:35 +02:00
int main(int argc, char **argv) {
GError *error = NULL;
2012-05-31 15:25:22 +02:00
GOptionContext *context = g_option_context_new("[COMMAND]");
const gchar *role = NULL;
const GOptionEntry entries[] = {
{ "role", 'r', 0, G_OPTION_ARG_STRING, &role, "The role to use", "ROLE" },
{ NULL }
};
g_option_context_add_main_entries(context, entries, NULL);
2012-05-31 15:17:08 +02:00
g_option_context_add_group(context, gtk_get_option_group(TRUE));
2012-05-22 06:34:35 +02:00
if (!g_option_context_parse(context, &argc, &argv, &error)) {
g_print("option parsing failed: %s\n", error->message);
return 1;
}
2012-05-22 06:34:35 +02:00
2012-05-22 17:50:01 +02:00
GtkWidget *window = gtk_window_new(GTK_WINDOW_TOPLEVEL);
2012-05-22 06:34:35 +02:00
/*gtk_window_set_default_size(GTK_WINDOW(window), 400, 400);*/
2012-05-30 12:54:10 +02:00
if (role) {
gtk_window_set_role(GTK_WINDOW(window), role);
}
2012-05-30 12:54:10 +02:00
#ifdef ICON_NAME
GdkPixbuf *icon = gtk_icon_theme_load_icon(gtk_icon_theme_get_default(), ICON_NAME, 48, 0, NULL);
2012-05-30 12:40:38 +02:00
if (icon) {
2012-05-30 12:54:10 +02:00
gtk_window_set_icon(GTK_WINDOW(window), icon);
2012-05-30 12:40:38 +02:00
}
2012-05-30 12:54:10 +02:00
#endif
2012-05-22 06:34:35 +02:00
GtkWidget *overlay = gtk_overlay_new();
2012-05-22 06:34:35 +02:00
GtkWidget *vte = vte_terminal_new();
char **command_argv;
char *default_argv[2] = {NULL, NULL};
if (argc > 1) {
command_argv = &argv[1];
} else {
default_argv[0] = vte_get_user_shell();
if (!default_argv[0]) default_argv[0] = "/bin/sh";
command_argv = default_argv;
}
2012-05-22 06:34:35 +02:00
VtePty *pty = vte_terminal_pty_new(VTE_TERMINAL(vte), 0, &error);
if (!pty) {
2012-05-22 10:47:36 +02:00
fprintf(stderr, "Failed to create pty: %s\n", error->message);
2012-05-22 06:34:35 +02:00
return 1;
}
vte_terminal_set_pty_object(VTE_TERMINAL(vte), pty);
vte_pty_set_term(pty, term);
GPid ppid;
if (g_spawn_async(NULL, command_argv, NULL,
2012-05-24 05:44:37 +02:00
G_SPAWN_DO_NOT_REAP_CHILD | G_SPAWN_SEARCH_PATH,
2012-05-22 06:34:35 +02:00
(GSpawnChildSetupFunc)vte_pty_child_setup, pty,
&ppid, &error)) {
vte_terminal_watch_child(VTE_TERMINAL(vte), ppid);
} else {
2012-05-22 10:47:36 +02:00
fprintf(stderr, "The new terminal's command failed to run: %s\n", error->message);
2012-05-22 06:34:35 +02:00
return 1;
}
gtk_container_add(GTK_CONTAINER(overlay), vte);
gtk_container_add(GTK_CONTAINER(window), overlay);
GtkWidget *entry = gtk_entry_new();
gtk_widget_set_halign(entry, GTK_ALIGN_START);
gtk_widget_set_valign(entry, GTK_ALIGN_END);
gtk_overlay_add_overlay(GTK_OVERLAY(overlay), entry);
2012-05-22 06:34:35 +02:00
g_signal_connect(vte, "child-exited", G_CALLBACK(gtk_main_quit), NULL);
g_signal_connect(overlay, "get-child-position", G_CALLBACK(position_overlay), NULL);
2012-05-22 06:34:35 +02:00
vte_terminal_set_scrollback_lines(VTE_TERMINAL(vte), scrollback_lines);
vte_terminal_set_font_from_string(VTE_TERMINAL(vte), font);
vte_terminal_set_scroll_on_output(VTE_TERMINAL(vte), scroll_on_output);
vte_terminal_set_scroll_on_keystroke(VTE_TERMINAL(vte), scroll_on_keystroke);
2012-05-22 06:55:48 +02:00
vte_terminal_set_audible_bell(VTE_TERMINAL(vte), audible_bell);
vte_terminal_set_visible_bell(VTE_TERMINAL(vte), visible_bell);
vte_terminal_set_mouse_autohide(VTE_TERMINAL(vte), mouse_autohide);
2012-05-22 06:34:35 +02:00
2012-05-24 08:51:40 +02:00
#ifdef TRANSPARENCY
GdkScreen *screen = gtk_widget_get_screen(window);
2012-05-31 11:04:57 +02:00
#if GTK_CHECK_VERSION(2, 90, 7)
2012-05-31 09:36:36 +02:00
GdkVisual *visual = gdk_screen_get_rgba_visual(screen);
2012-05-31 11:32:51 +02:00
if (!visual) {
2012-05-31 09:36:36 +02:00
visual = gdk_screen_get_system_visual(screen);
2012-05-31 11:32:51 +02:00
}
2012-05-31 09:36:36 +02:00
gtk_widget_set_visual(GTK_WIDGET(window), visual);
#else
GdkColormap *colormap = gdk_screen_get_rgba_colormap(screen);
2012-05-31 11:32:51 +02:00
if (colormap) {
gtk_widget_set_colormap(window, colormap);
}
#endif
2012-05-25 14:05:08 +02:00
vte_terminal_set_background_saturation(VTE_TERMINAL(vte), TRANSPARENCY);
vte_terminal_set_opacity(VTE_TERMINAL(vte), (guint16)(0xffff * (1 - TRANSPARENCY)));
2012-05-24 08:51:40 +02:00
#endif
2012-05-22 06:34:35 +02:00
// set colors
2012-05-30 15:22:02 +02:00
GdkColor foreground, background, palette[16];
2012-05-22 06:34:35 +02:00
gdk_color_parse(foreground_color, &foreground);
gdk_color_parse(background_color, &background);
2012-05-30 15:22:02 +02:00
for (unsigned i = 0; i < 16; i++) {
2012-05-22 06:34:35 +02:00
gdk_color_parse(colors[i], &palette[i]);
}
vte_terminal_set_colors(VTE_TERMINAL(vte), &foreground, &background, palette, 16);
2012-05-30 10:22:09 +02:00
search_dialog_info info = { .vte = vte, .open = false };
2012-05-31 11:34:47 +02:00
g_signal_connect(window, "destroy", G_CALLBACK(gtk_main_quit), NULL);
2012-05-30 10:22:09 +02:00
g_signal_connect(vte, "key-press-event", G_CALLBACK(key_press_cb), &info);
2012-05-22 06:34:35 +02:00
2012-05-24 05:41:55 +02:00
#ifdef CLICKABLE_URL
int tmp = vte_terminal_match_add_gregex(VTE_TERMINAL(vte),
g_regex_new(url_regex,
2012-05-24 05:44:37 +02:00
G_REGEX_CASELESS,
G_REGEX_MATCH_NOTEMPTY,
NULL),
2012-05-24 05:41:55 +02:00
0);
vte_terminal_match_set_cursor_type(VTE_TERMINAL(vte), tmp, GDK_HAND2);
g_signal_connect(vte, "button-press-event", G_CALLBACK(button_press_cb), NULL);
#endif
2012-05-22 13:47:51 +02:00
#ifdef URGENT_ON_BEEP
2012-05-30 19:30:00 +02:00
g_signal_connect(vte, "beep", G_CALLBACK(beep_handler), window);
g_signal_connect(window, "focus-in-event", G_CALLBACK(focus_in_handler), NULL);
2012-05-22 13:47:51 +02:00
#endif
2012-05-22 14:22:31 +02:00
#ifdef DYNAMIC_TITLE
window_title_cb(VTE_TERMINAL(vte), GTK_WINDOW(window));
2012-05-22 14:22:31 +02:00
g_signal_connect(vte, "window-title-changed", G_CALLBACK(window_title_cb), window);
#endif
2012-05-22 17:50:01 +02:00
gtk_widget_grab_focus(vte);
2012-05-22 06:34:35 +02:00
gtk_widget_show_all(window);
gtk_main();
return 0;
}