#define _POSIX_C_SOURCE 200809L #include #include #include #include #ifndef __GNUC__ # define __attribute__(x) #endif static const char *url_regex = "(ftp|http)s?://[-a-zA-Z0-9.?$%&/=_~#.,:;+()]*"; typedef enum overlay_mode { OVERLAY_HIDDEN = 0, OVERLAY_SEARCH, OVERLAY_RSEARCH, OVERLAY_COMPLETION } overlay_mode; typedef struct search_panel_info { GtkWidget *vte; GtkWidget *entry; GtkBin *panel; enum overlay_mode mode; } search_panel_info; static gchar *browser_cmd[3] = { NULL }; static void launch_browser(char *url); static void window_title_cb(VteTerminal *vte, GtkWindow *window); static gboolean key_press_cb(VteTerminal *vte, GdkEventKey *event, search_panel_info *info); static gboolean entry_key_press_cb(GtkEntry *entry, GdkEventKey *event, search_panel_info *info); static gboolean position_overlay_cb(GtkBin *overlay, GtkWidget *widget, GdkRectangle *alloc); static gboolean button_press_cb(VteTerminal *vte, GdkEventButton *event); static void beep_cb(GtkWindow *window); static gboolean focus_in_cb(GtkWindow *window); static gboolean add_to_list_store(char *key, void *value, GtkListStore *store); static GtkTreeModel *create_completion_model(VteTerminal *vte); static void search(VteTerminal *vte, const char *pattern, bool reverse); static void overlay_show(search_panel_info *info, overlay_mode mode, bool complete); static void get_vte_padding(VteTerminal *vte, int *w, int *h); static char *check_match(VteTerminal *vte, int event_x, int event_y); void launch_browser(char *url) { browser_cmd[1] = url; g_spawn_async(NULL, browser_cmd, NULL, G_SPAWN_SEARCH_PATH, NULL, NULL, NULL, NULL); } /* {{{ CALLBACKS */ void window_title_cb(VteTerminal *vte, GtkWindow *window) { const char *t = vte_terminal_get_window_title(vte); gtk_window_set_title(window, t ? t : "termite"); } gboolean key_press_cb(VteTerminal *vte, GdkEventKey *event, search_panel_info *info) { const guint modifiers = event->state & gtk_accelerator_get_default_mod_mask(); if (modifiers == (GDK_CONTROL_MASK|GDK_SHIFT_MASK)) { switch (gdk_keyval_to_lower(event->keyval)) { case GDK_KEY_c: vte_terminal_copy_clipboard(vte); return TRUE; case GDK_KEY_v: vte_terminal_paste_clipboard(vte); return TRUE; case GDK_KEY_p: vte_terminal_search_find_previous(vte); vte_terminal_copy_primary(vte); return TRUE; case GDK_KEY_n: vte_terminal_search_find_next(vte); vte_terminal_copy_primary(vte); return TRUE; case GDK_KEY_f: overlay_show(info, OVERLAY_SEARCH, true); return TRUE; case GDK_KEY_r: overlay_show(info, OVERLAY_RSEARCH, true); return TRUE; case GDK_KEY_j: search(vte, url_regex, false); return TRUE; case GDK_KEY_k: search(vte, url_regex, true); return TRUE; } } else if (modifiers == GDK_CONTROL_MASK && event->keyval == GDK_KEY_Tab) { overlay_show(info, OVERLAY_COMPLETION, true); return TRUE; } return FALSE; } gboolean entry_key_press_cb(GtkEntry *entry, GdkEventKey *event, search_panel_info *info) { gboolean ret = FALSE; if (event->keyval == GDK_KEY_Escape) { ret = TRUE; } else if (event->keyval == GDK_KEY_Return) { const gchar *text = gtk_entry_get_text(entry); switch (info->mode) { case OVERLAY_SEARCH: search(VTE_TERMINAL(info->vte), text, false); break; case OVERLAY_RSEARCH: search(VTE_TERMINAL(info->vte), text, true); break; case OVERLAY_COMPLETION: vte_terminal_feed_child(VTE_TERMINAL(info->vte), text, -1); break; case OVERLAY_HIDDEN: break; } ret = TRUE; } if (ret) { info->mode = OVERLAY_HIDDEN; gtk_widget_hide(GTK_WIDGET(info->panel)); gtk_widget_grab_focus(info->vte); } return ret; } gboolean position_overlay_cb(GtkBin *overlay, GtkWidget *widget, GdkRectangle *alloc) { GtkWidget *vte = gtk_bin_get_child(overlay); int width = gtk_widget_get_allocated_width(vte); int height = gtk_widget_get_allocated_height(vte); GtkRequisition req; gtk_widget_get_preferred_size(widget, NULL, &req); alloc->x = width - req.width - 40; alloc->y = 0; alloc->width = MIN(width, req.width); alloc->height = MIN(height, req.height); return TRUE; } gboolean button_press_cb(VteTerminal *vte, GdkEventButton *event) { char *match = check_match(vte, (int)event->x, (int)event->y); if (event->button == 1 && event->type == GDK_BUTTON_PRESS && match) { launch_browser(match); g_free(match); return TRUE; } return FALSE; } void beep_cb(GtkWindow *window) { gtk_window_set_urgency_hint(window, TRUE); } gboolean focus_in_cb(GtkWindow *window) { gtk_window_set_urgency_hint(window, FALSE); return FALSE; } /* }}} */ gboolean add_to_list_store(char *key, __attribute__((unused)) void *value, GtkListStore *store) { GtkTreeIter iter; gtk_list_store_append(store, &iter); gtk_list_store_set(store, &iter, 0, key, -1); return FALSE; } GtkTreeModel *create_completion_model(VteTerminal *vte) { GtkListStore *store = gtk_list_store_new(1, G_TYPE_STRING); glong end_row, end_col; vte_terminal_get_cursor_position(vte, &end_col, &end_row); gchar *content = vte_terminal_get_text_range(vte, 0, 0, end_row, end_col, NULL, NULL, NULL); if (!content) { g_printerr("no content returned for completion\n"); return GTK_TREE_MODEL(store); } char *s_ptr = content, *saveptr; GTree *tree = g_tree_new((GCompareFunc)strcmp); for (; ; s_ptr = NULL) { char *token = strtok_r(s_ptr, " \n\t", &saveptr); if (!token) { break; } g_tree_insert(tree, token, NULL); } g_tree_foreach(tree, (GTraverseFunc)add_to_list_store, store); g_tree_destroy(tree); g_free(content); return GTK_TREE_MODEL(store); } void search(VteTerminal *vte, const char *pattern, bool reverse) { GRegex *regex = vte_terminal_search_get_gregex(vte); if (regex) g_regex_unref(regex); regex = g_regex_new(pattern, (GRegexCompileFlags)0, (GRegexMatchFlags)0, NULL); vte_terminal_search_set_gregex(vte, regex); if (!reverse) { vte_terminal_search_find_next(vte); } else { vte_terminal_search_find_previous(vte); } vte_terminal_copy_primary(vte); } void overlay_show(search_panel_info *info, overlay_mode mode, bool complete) { if (complete) { GtkEntryCompletion *completion = gtk_entry_completion_new(); gtk_entry_set_completion(GTK_ENTRY(info->entry), completion); g_object_unref(completion); GtkTreeModel *completion_model = create_completion_model(VTE_TERMINAL(info->vte)); gtk_entry_completion_set_model(completion, completion_model); g_object_unref(completion_model); gtk_entry_completion_set_inline_selection(completion, TRUE); gtk_entry_completion_set_text_column(completion, 0); } gtk_entry_set_text(GTK_ENTRY(info->entry), ""); info->mode = mode; gtk_widget_show(GTK_WIDGET(info->panel)); gtk_widget_grab_focus(info->entry); } void get_vte_padding(VteTerminal *vte, int *w, int *h) { GtkBorder *border = NULL; gtk_widget_style_get(GTK_WIDGET(vte), "inner-border", &border, NULL); 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); } } char *check_match(VteTerminal *vte, int event_x, int event_y) { int xpad, ypad, tag; get_vte_padding(vte, &xpad, &ypad); 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); } /* {{{ CONFIG LOADING */ #define MAKE_GET_CONFIG_FUNCTION(NAME, TYPE) \ static bool get_config_ ## NAME (GKeyFile *config, const char *group, const char *key, TYPE *value) { \ GError *error = NULL; \ *value = g_key_file_get_ ## NAME (config, group, key, &error); \ if (error) { \ g_error_free(error); \ return false; \ } \ return true; \ } MAKE_GET_CONFIG_FUNCTION(boolean, gboolean) MAKE_GET_CONFIG_FUNCTION(integer, gint) MAKE_GET_CONFIG_FUNCTION(string, gchar *) MAKE_GET_CONFIG_FUNCTION(double, gdouble) static void load_config(GtkWindow *window, VteTerminal *vte, bool first_run, gboolean *dynamic_title, gboolean *urgent_on_bell, gboolean *clickable_url, double *transparency, const gchar **term) { static const char *filename = "termite.cfg"; const gchar *dir = g_get_user_config_dir(); gchar *path = g_build_filename(dir, filename, NULL); GKeyFile *config = g_key_file_new(); if ((g_key_file_load_from_file(config, path, G_KEY_FILE_NONE, NULL) || g_key_file_load_from_dirs(config, filename, (const char **)g_get_system_config_dirs(), NULL, G_KEY_FILE_NONE, NULL))) { gboolean cfgbool; gdouble cfgdouble; gint cfgint; gchar *cfgstr; if (first_run && get_config_string(config, "options", "term", &cfgstr)) { *term = cfgstr; } if (get_config_boolean(config, "options", "resize_grip", &cfgbool)) { gtk_window_set_has_resize_grip(window, cfgbool); } if (get_config_boolean(config, "options", "scroll_on_output", &cfgbool)) { vte_terminal_set_scroll_on_output(vte, cfgbool); } if (get_config_boolean(config, "options", "scroll_on_keystroke", &cfgbool)) { vte_terminal_set_scroll_on_keystroke(vte, cfgbool); } if (get_config_boolean(config, "options", "audible_bell", &cfgbool)) { vte_terminal_set_audible_bell(vte, cfgbool); } if (get_config_boolean(config, "options", "visible_bell", &cfgbool)) { vte_terminal_set_visible_bell(vte, cfgbool); } if (get_config_boolean(config, "options", "mouse_autohide", &cfgbool)) { vte_terminal_set_mouse_autohide(vte, cfgbool); } if (get_config_boolean(config, "options", "allow_bold", &cfgbool)) { vte_terminal_set_allow_bold(vte, cfgbool); } if (get_config_boolean(config, "options", "dynamic_title", &cfgbool)) { *dynamic_title = cfgbool; } if (get_config_boolean(config, "options", "urgent_on_bell", &cfgbool)) { *urgent_on_bell = cfgbool; } if (get_config_boolean(config, "options", "clickable_url", &cfgbool)) { *clickable_url = cfgbool; } if (get_config_string(config, "options", "browser", &cfgstr)) { browser_cmd[0] = cfgstr; } else { browser_cmd[0] = g_strdup(g_getenv("BROWSER")); if (!browser_cmd[0]) *clickable_url = false; } if (get_config_string(config, "options", "font", &cfgstr)) { vte_terminal_set_font_from_string(vte, cfgstr); g_free(cfgstr); } if (get_config_integer(config, "options", "scrollback_lines", &cfgint)) { vte_terminal_set_scrollback_lines(vte, cfgint); } if (get_config_string(config, "options", "cursor_blink", &cfgstr)) { if (!g_ascii_strcasecmp(cfgstr, "system")) { vte_terminal_set_cursor_blink_mode(vte, VTE_CURSOR_BLINK_SYSTEM); } else if (!g_ascii_strcasecmp(cfgstr, "on")) { vte_terminal_set_cursor_blink_mode(vte, VTE_CURSOR_BLINK_ON); } else if (!g_ascii_strcasecmp(cfgstr, "off")) { vte_terminal_set_cursor_blink_mode(vte, VTE_CURSOR_BLINK_OFF); } g_free(cfgstr); } if (get_config_string(config, "options", "cursor_shape", &cfgstr)) { if (!g_ascii_strcasecmp(cfgstr, "block")) { vte_terminal_set_cursor_shape(vte, VTE_CURSOR_SHAPE_BLOCK); } else if (!g_ascii_strcasecmp(cfgstr, "ibeam")) { vte_terminal_set_cursor_shape(vte, VTE_CURSOR_SHAPE_IBEAM); } else if (!g_ascii_strcasecmp(cfgstr, "underline")) { vte_terminal_set_cursor_shape(vte, VTE_CURSOR_SHAPE_UNDERLINE); } g_free(cfgstr); } if (get_config_string(config, "options", "icon_name", &cfgstr)) { gtk_window_set_icon_name(window, cfgstr); g_free(cfgstr); } if (get_config_double(config, "options", "transparency", &cfgdouble)) { *transparency = cfgdouble; } GdkColor foreground, background, cursor, palette[16]; static const char *colors[8] = {"black", "red", "green", "yellow", "blue", "magenta", "cyan", "white"}; bool success = true; for (unsigned i = 0; i < 8; i++) { GError *error = NULL; gsize length; gchar **pair = g_key_file_get_string_list(config, "colors", colors[i], &length, &error); if (error) { success = false; g_error_free(error); break; } if ((length != 2 || !gdk_color_parse(pair[0], &palette[i]) || !gdk_color_parse(pair[1], &palette[i+8]))) { success = false; g_strfreev(pair); break; } g_strfreev(pair); } if (success) { vte_terminal_set_colors(vte, NULL, NULL, palette, 16); } if (get_config_string(config, "colors", "foreground", &cfgstr)) { if (gdk_color_parse(cfgstr, &foreground)) { vte_terminal_set_color_foreground(vte, &foreground); } g_free(cfgstr); } if (get_config_string(config, "colors", "background", &cfgstr)) { if (gdk_color_parse(cfgstr, &background)) { vte_terminal_set_color_background(vte, &background); } g_free(cfgstr); } if (get_config_string(config, "colors", "cursor", &cfgstr)) { if (gdk_color_parse(cfgstr, &cursor)) { vte_terminal_set_color_cursor(vte, &cursor); } g_free(cfgstr); } } g_free(path); g_key_file_free(config); }/*}}}*/ int main(int argc, char **argv) { GError *error = NULL; const char *term = "vte-256color"; gboolean dynamic_title = FALSE, urgent_on_bell = FALSE, clickable_url = FALSE; double transparency = 0.0; GOptionContext *context = g_option_context_new("[COMMAND]"); 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); g_option_context_add_group(context, gtk_get_option_group(TRUE)); if (!g_option_context_parse(context, &argc, &argv, &error)) { g_printerr("option parsing failed: %s\n", error->message); return 1; } GtkWidget *window = gtk_window_new(GTK_WINDOW_TOPLEVEL); GtkWidget *overlay = gtk_overlay_new(); GtkWidget *vte = vte_terminal_new(); if (role) { gtk_window_set_role(GTK_WINDOW(window), role); g_free(role); } char **command_argv; char fallback[] = "/bin/sh"; char *default_argv[2] = {fallback, NULL}; if (argc > 1) { command_argv = &argv[1]; } else { char *shell = vte_get_user_shell(); if (shell) default_argv[0] = shell; command_argv = default_argv; } VtePty *pty = vte_terminal_pty_new(VTE_TERMINAL(vte), VTE_PTY_DEFAULT, &error); if (!pty) { g_printerr("Failed to create pty: %s\n", error->message); return 1; } load_config(GTK_WINDOW(window), VTE_TERMINAL(vte), true, &dynamic_title, &urgent_on_bell, &clickable_url, &transparency, &term); 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, (GSpawnFlags)(G_SPAWN_DO_NOT_REAP_CHILD | G_SPAWN_SEARCH_PATH), (GSpawnChildSetupFunc)vte_pty_child_setup, pty, &ppid, &error)) { vte_terminal_watch_child(VTE_TERMINAL(vte), ppid); } else { g_printerr("The new terminal's command failed to run: %s\n", error->message); return 1; } GtkWidget *alignment = gtk_alignment_new(0, 0, 1, 1); gtk_alignment_set_padding(GTK_ALIGNMENT(alignment), 5, 5, 5, 5); gtk_overlay_add_overlay(GTK_OVERLAY(overlay), alignment); GtkWidget *entry = gtk_entry_new(); gtk_widget_set_halign(entry, GTK_ALIGN_START); gtk_widget_set_valign(entry, GTK_ALIGN_END); gtk_container_add(GTK_CONTAINER(alignment), entry); gtk_container_add(GTK_CONTAINER(overlay), vte); gtk_container_add(GTK_CONTAINER(window), overlay); search_panel_info info = {vte, entry, GTK_BIN(alignment), OVERLAY_HIDDEN}; g_signal_connect(window, "destroy", G_CALLBACK(gtk_main_quit), NULL); g_signal_connect(vte, "child-exited", G_CALLBACK(gtk_main_quit), NULL); g_signal_connect(vte, "key-press-event", G_CALLBACK(key_press_cb), &info); g_signal_connect(entry, "key-press-event", G_CALLBACK(entry_key_press_cb), &info); g_signal_connect(overlay, "get-child-position", G_CALLBACK(position_overlay_cb), NULL); if (transparency > 0.0) { GdkScreen *screen = gtk_widget_get_screen(window); GdkVisual *visual = gdk_screen_get_rgba_visual(screen); if (!visual) { visual = gdk_screen_get_system_visual(screen); } gtk_widget_set_visual(window, visual); vte_terminal_set_background_saturation(VTE_TERMINAL(vte), transparency); vte_terminal_set_opacity(VTE_TERMINAL(vte), (guint16)(0xffff * (1 - transparency))); } if (clickable_url) { int tmp = vte_terminal_match_add_gregex(VTE_TERMINAL(vte), g_regex_new(url_regex, G_REGEX_CASELESS, G_REGEX_MATCH_NOTEMPTY, NULL), (GRegexMatchFlags)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); } if (urgent_on_bell) { g_signal_connect_swapped(vte, "beep", G_CALLBACK(beep_cb), window); g_signal_connect(window, "focus-in-event", G_CALLBACK(focus_in_cb), NULL); } if (dynamic_title) { window_title_cb(VTE_TERMINAL(vte), GTK_WINDOW(window)); g_signal_connect(vte, "window-title-changed", G_CALLBACK(window_title_cb), window); } gtk_widget_grab_focus(vte); gtk_widget_show_all(window); gtk_widget_hide(alignment); gtk_main(); return 0; } // vim: et:sts=4:sw=4:cino=(0