2012-06-02 14:46:46 +02:00
|
|
|
#define _POSIX_C_SOURCE 200809L
|
|
|
|
|
2012-05-30 10:22:09 +02:00
|
|
|
#include <stdbool.h>
|
2012-06-03 02:28:06 +02:00
|
|
|
#include <stdlib.h>
|
2012-06-02 14:46:46 +02:00
|
|
|
#include <string.h>
|
2012-05-30 10:22:09 +02:00
|
|
|
|
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
|
2012-06-01 12:10:17 +02:00
|
|
|
#define CONCAT2(X, Y) CONCAT(X, Y)
|
2012-05-31 15:38:13 +02:00
|
|
|
#define KEY(X) CONCAT(GDK_KEY_, X)
|
|
|
|
|
2012-05-30 10:22:09 +02:00
|
|
|
#ifndef __GNUC__
|
|
|
|
# define __attribute__(x)
|
|
|
|
#endif
|
|
|
|
|
2012-06-01 03:31:09 +02:00
|
|
|
typedef struct search_panel_info {
|
2012-05-30 10:22:09 +02:00
|
|
|
GtkWidget *vte;
|
2012-06-01 03:49:06 +02:00
|
|
|
GtkWidget *entry;
|
2012-06-01 03:31:09 +02:00
|
|
|
GtkBin *panel;
|
2012-05-30 10:29:25 +02:00
|
|
|
bool reverse;
|
2012-06-01 03:31:09 +02:00
|
|
|
} search_panel_info;
|
2012-05-30 10:22:09 +02:00
|
|
|
|
2012-06-03 02:28:06 +02:00
|
|
|
static gboolean always_selected(__attribute__((unused)) VteTerminal *vte,
|
|
|
|
__attribute__((unused)) glong column,
|
|
|
|
__attribute__((unused)) glong row) {
|
2012-06-02 14:46:46 +02:00
|
|
|
return TRUE;
|
|
|
|
}
|
|
|
|
|
2012-06-03 03:31:28 +02:00
|
|
|
static 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;
|
|
|
|
}
|
|
|
|
|
2012-06-03 02:28:06 +02:00
|
|
|
static GtkTreeModel *create_completion_model(VteTerminal *vte) {
|
|
|
|
GtkListStore *store;
|
|
|
|
|
|
|
|
store = gtk_list_store_new(1, G_TYPE_STRING);
|
|
|
|
|
2012-06-02 14:46:46 +02:00
|
|
|
// TODO: get the full buffer
|
|
|
|
gchar *content = vte_terminal_get_text(vte,
|
|
|
|
(VteSelectionFunc)always_selected,
|
|
|
|
NULL,
|
|
|
|
NULL);
|
|
|
|
|
|
|
|
if (!content) {
|
|
|
|
fputs("no content", stderr);
|
2012-06-03 02:28:06 +02:00
|
|
|
exit(EXIT_FAILURE);
|
2012-06-02 14:46:46 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
char *s_ptr = content, *saveptr;
|
|
|
|
|
2012-06-03 03:31:28 +02:00
|
|
|
GTree *tree = g_tree_new((GCompareFunc)strcmp);
|
|
|
|
|
2012-06-02 14:46:46 +02:00
|
|
|
for (int j = 1; ; j++, s_ptr = NULL) {
|
|
|
|
char *token = strtok_r(s_ptr, " \n", &saveptr);
|
|
|
|
if (!token) {
|
|
|
|
break;
|
|
|
|
}
|
2012-06-03 03:31:28 +02:00
|
|
|
g_tree_insert(tree, token, NULL);
|
2012-06-02 14:46:46 +02:00
|
|
|
}
|
|
|
|
|
2012-06-03 03:31:28 +02:00
|
|
|
g_tree_foreach(tree, (GTraverseFunc)add_to_list_store, store);
|
|
|
|
g_tree_destroy(tree);
|
2012-06-02 14:46:46 +02:00
|
|
|
g_free(content);
|
2012-06-03 02:28:06 +02:00
|
|
|
|
|
|
|
return GTK_TREE_MODEL(store);
|
|
|
|
}
|
|
|
|
|
|
|
|
static GtkWidget *test_window = NULL;
|
|
|
|
|
|
|
|
static GtkWidget *do_entry_completion(VteTerminal *do_widget) {
|
|
|
|
GtkWidget *window = gtk_widget_get_toplevel(GTK_WIDGET(do_widget));
|
|
|
|
|
|
|
|
if (!test_window) {
|
|
|
|
test_window = gtk_dialog_new_with_buttons("GtkEntryCompletion",
|
|
|
|
GTK_WINDOW(window),
|
|
|
|
(GtkDialogFlags)0,
|
|
|
|
NULL,
|
|
|
|
NULL);
|
|
|
|
gtk_window_set_resizable(GTK_WINDOW(test_window), FALSE);
|
|
|
|
|
|
|
|
g_signal_connect(test_window, "response", G_CALLBACK(gtk_widget_destroy), NULL);
|
|
|
|
g_signal_connect(test_window, "destroy", G_CALLBACK(gtk_widget_destroyed), &test_window);
|
|
|
|
|
|
|
|
GtkWidget *content_area = gtk_dialog_get_content_area(GTK_DIALOG(test_window));
|
|
|
|
|
|
|
|
// Create our entry
|
|
|
|
GtkWidget *entry = gtk_entry_new();
|
|
|
|
gtk_container_add(GTK_CONTAINER(content_area), entry);
|
|
|
|
|
|
|
|
// Create the completion object
|
|
|
|
GtkEntryCompletion *completion = gtk_entry_completion_new();
|
|
|
|
|
|
|
|
// Assign the completion to the entry
|
|
|
|
gtk_entry_set_completion(GTK_ENTRY(entry), completion);
|
|
|
|
g_object_unref(completion);
|
|
|
|
|
|
|
|
// Create a tree model and use it as the completion model
|
|
|
|
GtkTreeModel *completion_model = create_completion_model(do_widget);
|
|
|
|
gtk_entry_completion_set_model(completion, completion_model);
|
|
|
|
g_object_unref(completion_model);
|
|
|
|
|
|
|
|
// Use model column 0 as the text column
|
|
|
|
gtk_entry_completion_set_text_column(completion, 0);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!gtk_widget_get_visible(test_window))
|
|
|
|
gtk_widget_show_all(test_window);
|
|
|
|
else
|
|
|
|
gtk_widget_destroy(test_window);
|
|
|
|
|
|
|
|
return test_window;
|
2012-06-02 14:46:46 +02:00
|
|
|
}
|
|
|
|
|
2012-05-30 10:29:25 +02:00
|
|
|
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);
|
2012-06-02 12:13:59 +02:00
|
|
|
regex = g_regex_new(pattern, (GRegexCompileFlags)0, (GRegexMatchFlags)0, NULL);
|
2012-05-30 10:22:09 +02:00
|
|
|
vte_terminal_search_set_gregex(vte, regex);
|
|
|
|
|
2012-05-30 10:29:25 +02:00
|
|
|
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-06-01 03:31:09 +02:00
|
|
|
static gboolean search_key_press_cb(GtkEntry *entry, GdkEventKey *event, search_panel_info *info) {
|
2012-06-01 04:27:10 +02:00
|
|
|
gboolean ret = FALSE;
|
|
|
|
|
|
|
|
if (event->keyval == GDK_KEY_Escape) {
|
|
|
|
ret = TRUE;
|
|
|
|
} else if (event->keyval == GDK_KEY_Return) {
|
2012-06-02 12:22:07 +02:00
|
|
|
search(VTE_TERMINAL(info->vte), gtk_entry_get_text(entry), info->reverse);
|
2012-06-01 04:27:10 +02:00
|
|
|
ret = TRUE;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (ret) {
|
2012-06-01 03:31:09 +02:00
|
|
|
gtk_widget_hide(GTK_WIDGET(info->panel));
|
2012-06-01 03:49:06 +02:00
|
|
|
gtk_widget_grab_focus(info->vte);
|
2012-05-31 13:06:31 +02:00
|
|
|
}
|
2012-06-01 04:27:10 +02:00
|
|
|
return ret;
|
2012-05-31 13:06:31 +02:00
|
|
|
}
|
|
|
|
|
2012-06-02 12:53:31 +02:00
|
|
|
static gboolean key_press_cb(VteTerminal *vte, GdkEventKey *event, search_panel_info *info) {
|
2012-06-02 12:13:59 +02:00
|
|
|
const guint modifiers = event->state & gtk_accelerator_get_default_mod_mask();
|
2012-05-22 10:20:55 +02:00
|
|
|
if (modifiers == (GDK_CONTROL_MASK|GDK_SHIFT_MASK)) {
|
2012-05-22 08:58:38 +02:00
|
|
|
switch (gdk_keyval_to_lower(event->keyval)) {
|
2012-05-31 15:38:13 +02:00
|
|
|
case KEY(KEY_COPY):
|
2012-06-02 12:53:31 +02:00
|
|
|
vte_terminal_copy_clipboard(vte);
|
2012-05-23 11:40:04 +02:00
|
|
|
return TRUE;
|
2012-05-31 15:38:13 +02:00
|
|
|
case KEY(KEY_PASTE):
|
2012-06-02 12:53:31 +02:00
|
|
|
vte_terminal_paste_clipboard(vte);
|
2012-05-23 11:40:04 +02:00
|
|
|
return TRUE;
|
2012-05-31 15:38:13 +02:00
|
|
|
case KEY(KEY_PREV):
|
2012-06-02 12:53:31 +02:00
|
|
|
vte_terminal_search_find_previous(vte);
|
|
|
|
vte_terminal_copy_primary(vte);
|
2012-05-30 10:22:09 +02:00
|
|
|
return TRUE;
|
2012-05-31 15:38:13 +02:00
|
|
|
case KEY(KEY_NEXT):
|
2012-06-02 12:53:31 +02:00
|
|
|
vte_terminal_search_find_next(vte);
|
|
|
|
vte_terminal_copy_primary(vte);
|
2012-05-30 10:22:09 +02:00
|
|
|
return TRUE;
|
2012-05-31 15:38:13 +02:00
|
|
|
case KEY(KEY_SEARCH):
|
2012-06-01 03:31:09 +02:00
|
|
|
info->reverse = false;
|
|
|
|
gtk_widget_show(GTK_WIDGET(info->panel));
|
2012-06-01 03:49:06 +02:00
|
|
|
gtk_widget_grab_focus(info->entry);
|
2012-05-30 10:22:09 +02:00
|
|
|
return TRUE;
|
2012-05-31 15:38:13 +02:00
|
|
|
case KEY(KEY_RSEARCH):
|
2012-06-01 03:31:09 +02:00
|
|
|
info->reverse = true;
|
|
|
|
gtk_widget_show(GTK_WIDGET(info->panel));
|
2012-06-01 03:49:06 +02:00
|
|
|
gtk_widget_grab_focus(info->entry);
|
2012-05-30 10:22:09 +02:00
|
|
|
return TRUE;
|
2012-05-31 15:38:13 +02:00
|
|
|
case KEY(KEY_URL):
|
2012-06-02 12:53:31 +02:00
|
|
|
search(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-06-02 12:53:31 +02:00
|
|
|
search(vte, url_regex, true);
|
2012-05-30 11:35:41 +02:00
|
|
|
return TRUE;
|
2012-05-22 08:58:38 +02:00
|
|
|
}
|
|
|
|
}
|
2012-06-02 14:46:46 +02:00
|
|
|
if (modifiers == GDK_CONTROL_MASK && event->keyval == GDK_KEY_Tab) {
|
2012-06-03 02:28:06 +02:00
|
|
|
do_entry_completion(vte);
|
2012-06-02 14:46:46 +02:00
|
|
|
return TRUE;
|
|
|
|
}
|
2012-05-22 08:58:38 +02:00
|
|
|
return FALSE;
|
|
|
|
}
|
|
|
|
|
2012-05-24 05:41:55 +02:00
|
|
|
#ifdef CLICKABLE_URL
|
2012-05-22 13:15:28 +02:00
|
|
|
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) {
|
2012-05-22 13:15:28 +02:00
|
|
|
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,
|
2012-05-22 13:15:28 +02:00
|
|
|
(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) {
|
2012-06-01 11:42:05 +02:00
|
|
|
char *match = check_match(vte, (int)event->x, (int)event->y);
|
2012-05-22 13:15:28 +02:00
|
|
|
if (event->button == 1 && event->type == GDK_BUTTON_PRESS && match != NULL) {
|
2012-06-01 14:29:18 +02:00
|
|
|
char *argv[] = URL_COMMAND(match);
|
2012-06-02 12:13:59 +02:00
|
|
|
g_spawn_async(NULL, argv, NULL, (GSpawnFlags)0, NULL, NULL, NULL, NULL);
|
2012-05-22 13:15:28 +02:00
|
|
|
g_free(match);
|
|
|
|
return TRUE;
|
|
|
|
}
|
|
|
|
return FALSE;
|
|
|
|
}
|
2012-05-24 05:41:55 +02:00
|
|
|
#endif
|
2012-05-22 13:15:28 +02:00
|
|
|
|
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
|
|
|
|
|
2012-06-02 11:28:56 +02:00
|
|
|
static gboolean position_overlay_cb(GtkBin *overlay, GtkWidget *widget, GdkRectangle *alloc) {
|
|
|
|
GtkWidget *vte = gtk_bin_get_child(overlay);
|
2012-05-31 18:39:59 +02:00
|
|
|
|
2012-06-01 04:31:15 +02:00
|
|
|
int width = gtk_widget_get_allocated_width(vte);
|
|
|
|
int height = gtk_widget_get_allocated_height(vte);
|
2012-05-31 18:39:59 +02:00
|
|
|
|
|
|
|
GtkRequisition req;
|
|
|
|
gtk_widget_get_preferred_size(widget, NULL, &req);
|
|
|
|
|
2012-06-01 04:31:15 +02:00
|
|
|
alloc->x = width - req.width - 40;
|
2012-05-31 18:39:59 +02:00
|
|
|
alloc->y = 0;
|
2012-06-01 04:31:15 +02:00
|
|
|
alloc->width = MIN(width, req.width);
|
|
|
|
alloc->height = MIN(height, req.height);
|
2012-05-31 18:39:59 +02:00
|
|
|
|
|
|
|
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]");
|
2012-05-31 15:22:13 +02:00
|
|
|
const gchar *role = NULL;
|
|
|
|
const GOptionEntry entries[] = {
|
|
|
|
{ "role", 'r', 0, G_OPTION_ARG_STRING, &role, "The role to use", "ROLE" },
|
|
|
|
{ NULL }
|
|
|
|
};
|
2012-05-31 11:43:18 +02:00
|
|
|
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
|
|
|
|
2012-05-31 11:43:18 +02:00
|
|
|
if (!g_option_context_parse(context, &argc, &argv, &error)) {
|
2012-06-01 21:53:23 +02:00
|
|
|
g_printerr("option parsing failed: %s\n", error->message);
|
2012-05-31 11:43:18 +02:00
|
|
|
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
|
|
|
|
2012-05-31 11:43:18 +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
|
|
|
|
2012-05-31 18:39:59 +02:00
|
|
|
GtkWidget *overlay = gtk_overlay_new();
|
2012-05-22 06:34:35 +02:00
|
|
|
GtkWidget *vte = vte_terminal_new();
|
|
|
|
|
2012-05-24 04:50:01 +02:00
|
|
|
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
|
|
|
|
2012-06-02 17:42:20 +02:00
|
|
|
VtePty *pty = vte_terminal_pty_new(VTE_TERMINAL(vte), VTE_PTY_DEFAULT, &error);
|
2012-05-22 06:34:35 +02:00
|
|
|
|
|
|
|
if (!pty) {
|
2012-06-01 21:53:23 +02:00
|
|
|
g_printerr("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-06-02 12:13:59 +02:00
|
|
|
(GSpawnFlags)(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-06-01 21:53:23 +02:00
|
|
|
g_printerr("The new terminal's command failed to run: %s\n", error->message);
|
2012-05-22 06:34:35 +02:00
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
2012-06-01 03:31:09 +02:00
|
|
|
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);
|
2012-05-31 18:39:59 +02:00
|
|
|
|
|
|
|
GtkWidget *entry = gtk_entry_new();
|
|
|
|
gtk_widget_set_halign(entry, GTK_ALIGN_START);
|
|
|
|
gtk_widget_set_valign(entry, GTK_ALIGN_END);
|
2012-05-22 06:34:35 +02:00
|
|
|
|
2012-06-01 03:31:09 +02:00
|
|
|
gtk_container_add(GTK_CONTAINER(alignment), entry);
|
|
|
|
gtk_container_add(GTK_CONTAINER(overlay), vte);
|
|
|
|
gtk_container_add(GTK_CONTAINER(window), overlay);
|
|
|
|
|
2012-06-02 11:38:01 +02:00
|
|
|
search_panel_info info = {vte, entry, GTK_BIN(alignment), false};
|
2012-06-01 03:31:09 +02:00
|
|
|
|
|
|
|
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(search_key_press_cb), &info);
|
|
|
|
g_signal_connect(overlay, "get-child-position", G_CALLBACK(position_overlay_cb), 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-06-02 02:43:00 +02:00
|
|
|
vte_terminal_set_cursor_shape(VTE_TERMINAL(vte), CONCAT2(VTE_CURSOR_SHAPE_, CURSOR_SHAPE));
|
2012-06-01 12:10:17 +02:00
|
|
|
vte_terminal_set_cursor_blink_mode(VTE_TERMINAL(vte), CONCAT2(VTE_CURSOR_BLINK_, CURSOR_BLINK));
|
2012-06-01 11:14:44 +02:00
|
|
|
gtk_window_set_has_resize_grip(GTK_WINDOW(window), resize_grip);
|
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 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-06-03 03:53:03 +02:00
|
|
|
gtk_widget_set_visual(window, visual);
|
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-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-06-02 12:13:59 +02:00
|
|
|
(GRegexMatchFlags)0);
|
2012-05-24 05:41:55 +02:00
|
|
|
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
|
2012-05-22 14:26:53 +02:00
|
|
|
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);
|
2012-06-01 03:31:09 +02:00
|
|
|
gtk_widget_hide(alignment);
|
2012-05-22 06:34:35 +02:00
|
|
|
gtk_main();
|
|
|
|
return 0;
|
|
|
|
}
|