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-02 14:46:46 +02:00
|
|
|
#include <string.h>
|
2012-05-30 10:22:09 +02:00
|
|
|
|
2012-07-01 19:44:30 +02:00
|
|
|
#include <gdk/gdkx.h>
|
2012-05-22 06:34:35 +02:00
|
|
|
#include <gtk/gtk.h>
|
|
|
|
#include <vte/vte.h>
|
|
|
|
|
2012-05-30 10:22:09 +02:00
|
|
|
#ifndef __GNUC__
|
|
|
|
# define __attribute__(x)
|
|
|
|
#endif
|
|
|
|
|
2012-07-06 04:12:32 +02:00
|
|
|
#define CSI "\x1b["
|
|
|
|
|
2012-06-25 09:00:18 +02:00
|
|
|
static const char * const url_regex = "(ftp|http)s?://[-a-zA-Z0-9.?$%&/=_~#.,:;+()]*";
|
2012-06-08 22:16:41 +02:00
|
|
|
|
2012-06-08 22:04:48 +02:00
|
|
|
typedef enum overlay_mode {
|
2012-06-04 05:39:19 +02:00
|
|
|
OVERLAY_HIDDEN = 0,
|
2012-06-04 05:17:18 +02:00
|
|
|
OVERLAY_SEARCH,
|
2012-06-04 05:37:15 +02:00
|
|
|
OVERLAY_RSEARCH,
|
2012-06-04 05:17:18 +02:00
|
|
|
OVERLAY_COMPLETION
|
2012-06-08 22:04:48 +02:00
|
|
|
} overlay_mode;
|
2012-06-04 05:17:18 +02:00
|
|
|
|
2012-07-05 08:56:28 +02:00
|
|
|
typedef enum select_mode {
|
|
|
|
SELECT_OFF = 0,
|
|
|
|
SELECT_ON,
|
2012-07-07 02:38:46 +02:00
|
|
|
SELECT_VISUAL,
|
2012-07-07 04:46:00 +02:00
|
|
|
SELECT_VISUAL_LINE,
|
|
|
|
SELECT_VISUAL_BLOCK
|
2012-07-05 08:56:28 +02:00
|
|
|
} select_mode;
|
|
|
|
|
2012-07-05 08:40:48 +02:00
|
|
|
typedef struct select_info {
|
2012-07-05 08:56:28 +02:00
|
|
|
select_mode mode;
|
2012-07-07 01:39:07 +02:00
|
|
|
long begin_col;
|
|
|
|
long begin_row;
|
2012-07-09 09:10:08 +02:00
|
|
|
long cursor_col;
|
|
|
|
long cursor_row;
|
2012-07-05 08:40:48 +02:00
|
|
|
} select_info;
|
|
|
|
|
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-20 17:25:04 +02:00
|
|
|
GtkWidget *panel;
|
2012-06-04 05:17:18 +02:00
|
|
|
enum overlay_mode mode;
|
2012-07-05 08:40:48 +02:00
|
|
|
select_info select;
|
2012-06-01 03:31:09 +02:00
|
|
|
} search_panel_info;
|
2012-05-30 10:22:09 +02:00
|
|
|
|
2012-06-16 06:48:16 +02:00
|
|
|
static char *browser_cmd[3] = {NULL};
|
2012-05-31 23:16:01 +02:00
|
|
|
|
2012-06-08 22:04:48 +02:00
|
|
|
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);
|
2012-06-16 09:41:43 +02:00
|
|
|
static gboolean focus_cb(GtkWindow *window);
|
2012-06-08 22:04:48 +02:00
|
|
|
|
|
|
|
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);
|
2012-06-11 20:48:51 +02:00
|
|
|
static void load_config(GtkWindow *window, VteTerminal *vte,
|
2012-06-11 20:45:38 +02:00
|
|
|
gboolean *dynamic_title, gboolean *urgent_on_bell,
|
2012-06-16 06:48:16 +02:00
|
|
|
gboolean *clickable_url, const char **term);
|
2012-06-08 22:04:48 +02:00
|
|
|
|
|
|
|
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) {
|
2012-07-08 07:46:44 +02:00
|
|
|
const char * const t = vte_terminal_get_window_title(vte);
|
2012-06-08 22:04:48 +02:00
|
|
|
gtk_window_set_title(window, t ? t : "termite");
|
|
|
|
}
|
|
|
|
|
2012-07-09 22:07:08 +02:00
|
|
|
static void update_selection(VteTerminal *vte, const select_info *select) {
|
2012-07-09 09:24:44 +02:00
|
|
|
if (select->mode == SELECT_ON) {
|
2012-07-05 08:56:28 +02:00
|
|
|
return; // not in visual mode
|
|
|
|
}
|
|
|
|
|
2012-07-06 01:11:52 +02:00
|
|
|
vte_terminal_select_none(vte);
|
2012-07-08 07:32:30 +02:00
|
|
|
vte_terminal_set_selection_block_mode(vte, select->mode == SELECT_VISUAL_BLOCK);
|
2012-07-05 08:40:48 +02:00
|
|
|
|
2012-07-08 07:46:44 +02:00
|
|
|
const long n_columns = vte_terminal_get_column_count(vte);
|
2012-07-05 23:35:19 +02:00
|
|
|
|
2012-07-07 02:38:46 +02:00
|
|
|
if (select->mode == SELECT_VISUAL) {
|
2012-07-08 07:46:44 +02:00
|
|
|
const long begin = select->begin_row * n_columns + select->begin_col;
|
2012-07-09 22:16:41 +02:00
|
|
|
const long end = select->cursor_row * n_columns + select->cursor_col;
|
2012-07-07 02:38:46 +02:00
|
|
|
if (begin < end) {
|
|
|
|
vte_terminal_select_text(vte, select->begin_col, select->begin_row,
|
2012-07-09 22:16:41 +02:00
|
|
|
select->cursor_col, select->cursor_row, 0, 0);
|
2012-07-07 02:38:46 +02:00
|
|
|
} else {
|
2012-07-09 22:16:41 +02:00
|
|
|
vte_terminal_select_text(vte, select->cursor_col, select->cursor_row,
|
2012-07-07 02:38:46 +02:00
|
|
|
select->begin_col, select->begin_row, 0, 0);
|
|
|
|
}
|
|
|
|
} else if (select->mode == SELECT_VISUAL_LINE) {
|
2012-07-08 07:15:00 +02:00
|
|
|
vte_terminal_select_text(vte, 0,
|
2012-07-09 22:16:41 +02:00
|
|
|
MIN(select->begin_row, select->cursor_row),
|
2012-07-08 07:15:00 +02:00
|
|
|
n_columns - 1,
|
2012-07-09 22:16:41 +02:00
|
|
|
MAX(select->begin_row, select->cursor_row),
|
2012-07-08 07:15:00 +02:00
|
|
|
0, 0);
|
2012-07-07 04:46:00 +02:00
|
|
|
} else if (select->mode == SELECT_VISUAL_BLOCK) {
|
2012-07-08 07:12:26 +02:00
|
|
|
vte_terminal_select_text(vte,
|
2012-07-09 22:16:41 +02:00
|
|
|
MIN(select->begin_col, select->cursor_col),
|
|
|
|
MIN(select->begin_row, select->cursor_row),
|
|
|
|
MAX(select->begin_col, select->cursor_col),
|
|
|
|
MAX(select->begin_row, select->cursor_row),
|
2012-07-08 07:12:26 +02:00
|
|
|
0, 0);
|
2012-07-07 02:16:48 +02:00
|
|
|
}
|
2012-07-05 08:40:48 +02:00
|
|
|
|
|
|
|
vte_terminal_copy_primary(vte);
|
|
|
|
}
|
|
|
|
|
2012-07-08 09:36:27 +02:00
|
|
|
static void feed_str(VteTerminal *vte, const char *s) {
|
|
|
|
vte_terminal_feed(vte, s, (long)strlen(s));
|
|
|
|
}
|
|
|
|
|
2012-07-05 23:35:19 +02:00
|
|
|
static void start_selection(VteTerminal *vte, select_info *select) {
|
2012-07-09 09:10:08 +02:00
|
|
|
feed_str(vte, CSI "?25l"); // hide cursor
|
2012-07-05 08:56:28 +02:00
|
|
|
select->mode = SELECT_ON;
|
2012-07-09 09:10:08 +02:00
|
|
|
vte_terminal_get_cursor_position(vte, &select->cursor_col, &select->cursor_row);
|
2012-07-05 23:35:19 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
static void end_selection(VteTerminal *vte, select_info *select) {
|
2012-07-09 09:10:08 +02:00
|
|
|
feed_str(vte, CSI "?25h"); // show cursor
|
2012-07-07 04:56:47 +02:00
|
|
|
vte_terminal_select_none(vte);
|
2012-07-05 23:35:19 +02:00
|
|
|
select->mode = SELECT_OFF;
|
2012-07-05 08:40:48 +02:00
|
|
|
}
|
|
|
|
|
2012-07-07 02:38:46 +02:00
|
|
|
static void toggle_visual(VteTerminal *vte, select_info *select, select_mode mode) {
|
|
|
|
if (select->mode == mode) {
|
2012-07-05 08:56:28 +02:00
|
|
|
select->mode = SELECT_ON;
|
2012-07-06 01:11:52 +02:00
|
|
|
vte_terminal_select_none(vte);
|
2012-07-05 08:56:28 +02:00
|
|
|
} else {
|
2012-07-09 09:19:02 +02:00
|
|
|
if (select->mode == SELECT_ON) {
|
|
|
|
select->begin_col = select->cursor_col;
|
|
|
|
select->begin_row = select->cursor_row;
|
|
|
|
}
|
2012-07-07 02:38:46 +02:00
|
|
|
select->mode = mode;
|
2012-07-09 09:19:02 +02:00
|
|
|
update_selection(vte, select);
|
2012-07-05 08:56:28 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2012-07-10 01:29:22 +02:00
|
|
|
static long first_row(VteTerminal *vte) {
|
|
|
|
GtkAdjustment *adjust = gtk_scrollable_get_vadjustment(GTK_SCROLLABLE(vte));
|
|
|
|
double scroll_lower = gtk_adjustment_get_lower(adjust);
|
|
|
|
return (long)scroll_lower;
|
|
|
|
}
|
|
|
|
|
|
|
|
static long last_row(VteTerminal *vte) {
|
|
|
|
GtkAdjustment *adjust = gtk_scrollable_get_vadjustment(GTK_SCROLLABLE(vte));
|
|
|
|
double scroll_upper = gtk_adjustment_get_upper(adjust);
|
|
|
|
return (long)scroll_upper - 1;
|
|
|
|
}
|
|
|
|
|
2012-07-10 01:38:33 +02:00
|
|
|
static void update_scroll(VteTerminal *vte, const select_info *select) {
|
2012-07-09 22:57:09 +02:00
|
|
|
GtkAdjustment *adjust = gtk_scrollable_get_vadjustment(GTK_SCROLLABLE(vte));
|
|
|
|
double scroll_row = gtk_adjustment_get_value(adjust);
|
|
|
|
long n_rows = vte_terminal_get_row_count(vte);
|
|
|
|
|
|
|
|
if (select->cursor_row < scroll_row) {
|
|
|
|
gtk_adjustment_set_value(adjust, (double)select->cursor_row);
|
|
|
|
} else if (select->cursor_row - n_rows >= (long)scroll_row) {
|
|
|
|
gtk_adjustment_set_value(adjust, (double)(select->cursor_row - n_rows + 1));
|
|
|
|
}
|
2012-07-10 01:38:33 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
static void move(VteTerminal *vte, select_info *select, long col, long row) {
|
|
|
|
const long end_col = vte_terminal_get_column_count(vte) - 1;
|
|
|
|
|
|
|
|
select->cursor_col = CLAMP(select->cursor_col + col, 0, end_col);
|
|
|
|
select->cursor_row = CLAMP(select->cursor_row + row, first_row(vte), last_row(vte));
|
|
|
|
|
|
|
|
update_scroll(vte, select);
|
|
|
|
update_selection(vte, select);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void move_to_row_start(VteTerminal *vte, select_info *select, long row) {
|
|
|
|
select->cursor_col = 0;
|
|
|
|
select->cursor_row = row;
|
|
|
|
update_scroll(vte, select);
|
2012-07-09 09:10:08 +02:00
|
|
|
update_selection(vte, select);
|
2012-07-08 11:39:37 +02:00
|
|
|
}
|
|
|
|
|
2012-06-08 22:04:48 +02:00
|
|
|
gboolean key_press_cb(VteTerminal *vte, GdkEventKey *event, search_panel_info *info) {
|
|
|
|
const guint modifiers = event->state & gtk_accelerator_get_default_mod_mask();
|
2012-06-11 20:45:38 +02:00
|
|
|
gboolean dynamic_title = FALSE, urgent_on_bell = FALSE, clickable_url = FALSE;
|
2012-07-05 08:40:48 +02:00
|
|
|
if (info->select.mode) {
|
2012-07-07 04:46:00 +02:00
|
|
|
if (modifiers == GDK_CONTROL_MASK) {
|
|
|
|
if (gdk_keyval_to_lower(event->keyval) == GDK_KEY_v) {
|
|
|
|
toggle_visual(vte, &info->select, SELECT_VISUAL_BLOCK);
|
|
|
|
}
|
|
|
|
return TRUE;
|
|
|
|
}
|
2012-07-05 08:40:48 +02:00
|
|
|
switch (event->keyval) {
|
2012-07-05 20:34:23 +02:00
|
|
|
case GDK_KEY_Left:
|
2012-07-05 08:40:48 +02:00
|
|
|
case GDK_KEY_h:
|
2012-07-09 09:10:08 +02:00
|
|
|
move(vte, &info->select, -1, 0);
|
2012-07-05 23:35:19 +02:00
|
|
|
break;
|
|
|
|
case GDK_KEY_Down:
|
|
|
|
case GDK_KEY_j:
|
2012-07-09 09:10:08 +02:00
|
|
|
move(vte, &info->select, 0, 1);
|
2012-07-05 23:35:19 +02:00
|
|
|
break;
|
|
|
|
case GDK_KEY_Up:
|
|
|
|
case GDK_KEY_k:
|
2012-07-09 09:10:08 +02:00
|
|
|
move(vte, &info->select, 0, -1);
|
2012-07-05 08:40:48 +02:00
|
|
|
break;
|
2012-07-05 20:34:23 +02:00
|
|
|
case GDK_KEY_Right:
|
2012-07-05 08:40:48 +02:00
|
|
|
case GDK_KEY_l:
|
2012-07-09 09:10:08 +02:00
|
|
|
move(vte, &info->select, 1, 0);
|
2012-07-05 08:40:48 +02:00
|
|
|
break;
|
2012-07-07 06:10:30 +02:00
|
|
|
case GDK_KEY_asciicircum:
|
2012-07-09 09:10:08 +02:00
|
|
|
info->select.cursor_col = 0;
|
|
|
|
update_selection(vte, &info->select);
|
2012-07-07 06:10:30 +02:00
|
|
|
break;
|
|
|
|
case GDK_KEY_dollar:
|
2012-07-09 09:10:08 +02:00
|
|
|
info->select.cursor_col = vte_terminal_get_column_count(vte) - 1;
|
|
|
|
update_selection(vte, &info->select);
|
2012-07-07 06:10:30 +02:00
|
|
|
break;
|
2012-07-10 01:38:33 +02:00
|
|
|
case GDK_KEY_g:
|
|
|
|
move_to_row_start(vte, &info->select, first_row(vte));
|
|
|
|
break;
|
|
|
|
case GDK_KEY_G:
|
|
|
|
move_to_row_start(vte, &info->select, last_row(vte));
|
|
|
|
break;
|
2012-07-05 08:56:28 +02:00
|
|
|
case GDK_KEY_v:
|
2012-07-07 02:38:46 +02:00
|
|
|
toggle_visual(vte, &info->select, SELECT_VISUAL);
|
|
|
|
break;
|
|
|
|
case GDK_KEY_V:
|
|
|
|
toggle_visual(vte, &info->select, SELECT_VISUAL_LINE);
|
2012-07-05 08:56:28 +02:00
|
|
|
break;
|
2012-07-05 08:40:48 +02:00
|
|
|
case GDK_KEY_Escape:
|
2012-07-05 23:35:19 +02:00
|
|
|
end_selection(vte, &info->select);
|
2012-07-05 08:40:48 +02:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
return TRUE;
|
|
|
|
}
|
2012-06-08 22:04:48 +02:00
|
|
|
if (modifiers == (GDK_CONTROL_MASK|GDK_SHIFT_MASK)) {
|
|
|
|
switch (gdk_keyval_to_lower(event->keyval)) {
|
2012-07-05 08:40:48 +02:00
|
|
|
case GDK_KEY_space:
|
2012-07-05 23:35:19 +02:00
|
|
|
start_selection(vte, &info->select);
|
2012-07-05 08:40:48 +02:00
|
|
|
return TRUE;
|
2012-06-08 22:16:41 +02:00
|
|
|
case GDK_KEY_c:
|
2012-06-08 22:04:48 +02:00
|
|
|
vte_terminal_copy_clipboard(vte);
|
|
|
|
return TRUE;
|
2012-06-08 22:16:41 +02:00
|
|
|
case GDK_KEY_v:
|
2012-06-08 22:04:48 +02:00
|
|
|
vte_terminal_paste_clipboard(vte);
|
|
|
|
return TRUE;
|
2012-06-08 22:16:41 +02:00
|
|
|
case GDK_KEY_p:
|
2012-06-08 22:04:48 +02:00
|
|
|
vte_terminal_search_find_previous(vte);
|
|
|
|
vte_terminal_copy_primary(vte);
|
|
|
|
return TRUE;
|
2012-06-08 22:16:41 +02:00
|
|
|
case GDK_KEY_n:
|
2012-06-08 22:04:48 +02:00
|
|
|
vte_terminal_search_find_next(vte);
|
|
|
|
vte_terminal_copy_primary(vte);
|
|
|
|
return TRUE;
|
2012-06-08 22:16:41 +02:00
|
|
|
case GDK_KEY_f:
|
2012-06-08 22:04:48 +02:00
|
|
|
overlay_show(info, OVERLAY_SEARCH, true);
|
|
|
|
return TRUE;
|
2012-06-08 22:16:41 +02:00
|
|
|
case GDK_KEY_r:
|
2012-06-08 22:04:48 +02:00
|
|
|
overlay_show(info, OVERLAY_RSEARCH, true);
|
|
|
|
return TRUE;
|
2012-06-08 22:16:41 +02:00
|
|
|
case GDK_KEY_j:
|
2012-06-08 22:04:48 +02:00
|
|
|
search(vte, url_regex, false);
|
|
|
|
return TRUE;
|
2012-06-08 22:16:41 +02:00
|
|
|
case GDK_KEY_k:
|
2012-06-08 22:04:48 +02:00
|
|
|
search(vte, url_regex, true);
|
|
|
|
return TRUE;
|
2012-06-11 20:45:38 +02:00
|
|
|
case GDK_KEY_Escape:
|
|
|
|
load_config(GTK_WINDOW(gtk_widget_get_toplevel(GTK_WIDGET(vte))),
|
2012-06-11 20:48:51 +02:00
|
|
|
vte, &dynamic_title, &urgent_on_bell,
|
2012-06-11 22:38:01 +02:00
|
|
|
&clickable_url, NULL);
|
2012-06-11 20:45:38 +02:00
|
|
|
return TRUE;
|
2012-06-08 22:04:48 +02:00
|
|
|
}
|
|
|
|
} 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) {
|
2012-06-16 06:48:16 +02:00
|
|
|
const char *text = gtk_entry_get_text(entry);
|
2012-06-08 22:04:48 +02:00
|
|
|
|
|
|
|
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;
|
2012-06-20 17:25:04 +02:00
|
|
|
gtk_widget_hide(info->panel);
|
2012-06-08 22:04:48 +02:00
|
|
|
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);
|
|
|
|
|
2012-07-08 07:46:44 +02:00
|
|
|
const int width = gtk_widget_get_allocated_width(vte);
|
|
|
|
const int height = gtk_widget_get_allocated_height(vte);
|
2012-06-08 22:04:48 +02:00
|
|
|
|
|
|
|
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);
|
2012-06-10 20:16:51 +02:00
|
|
|
if (event->button == 1 && event->type == GDK_BUTTON_PRESS && match) {
|
2012-06-08 22:04:48 +02:00
|
|
|
launch_browser(match);
|
|
|
|
g_free(match);
|
|
|
|
return TRUE;
|
|
|
|
}
|
|
|
|
return FALSE;
|
|
|
|
}
|
|
|
|
|
|
|
|
void beep_cb(GtkWindow *window) {
|
|
|
|
gtk_window_set_urgency_hint(window, TRUE);
|
|
|
|
}
|
|
|
|
|
2012-06-16 09:41:43 +02:00
|
|
|
gboolean focus_cb(GtkWindow *window) {
|
2012-06-15 22:45:39 +02:00
|
|
|
gtk_window_set_urgency_hint(window, FALSE);
|
|
|
|
return FALSE;
|
|
|
|
}
|
2012-06-08 22:04:48 +02:00
|
|
|
/* }}} */
|
|
|
|
|
|
|
|
gboolean add_to_list_store(char *key,
|
|
|
|
__attribute__((unused)) void *value,
|
|
|
|
GtkListStore *store) {
|
2012-06-03 03:31:28 +02:00
|
|
|
GtkTreeIter iter;
|
|
|
|
gtk_list_store_append(store, &iter);
|
|
|
|
gtk_list_store_set(store, &iter, 0, key, -1);
|
|
|
|
return FALSE;
|
|
|
|
}
|
|
|
|
|
2012-06-08 22:04:48 +02:00
|
|
|
GtkTreeModel *create_completion_model(VteTerminal *vte) {
|
2012-06-06 01:39:44 +02:00
|
|
|
GtkListStore *store = gtk_list_store_new(1, G_TYPE_STRING);
|
2012-06-03 02:28:06 +02:00
|
|
|
|
2012-06-16 06:48:16 +02:00
|
|
|
long end_row, end_col;
|
2012-06-07 03:22:26 +02:00
|
|
|
vte_terminal_get_cursor_position(vte, &end_col, &end_row);
|
2012-06-16 06:48:16 +02:00
|
|
|
char *content = vte_terminal_get_text_range(vte, 0, 0, end_row, end_col,
|
|
|
|
NULL, NULL, NULL);
|
2012-06-02 14:46:46 +02:00
|
|
|
|
|
|
|
if (!content) {
|
2012-06-07 17:42:17 +02:00
|
|
|
g_printerr("no content returned for completion\n");
|
2012-06-07 03:26:22 +02:00
|
|
|
return GTK_TREE_MODEL(store);
|
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-07 04:06:46 +02:00
|
|
|
for (; ; s_ptr = NULL) {
|
2012-06-05 21:24:05 +02:00
|
|
|
char *token = strtok_r(s_ptr, " \n\t", &saveptr);
|
2012-06-02 14:46:46 +02:00
|
|
|
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);
|
2012-06-07 02:54:33 +02:00
|
|
|
g_tree_destroy(tree);
|
2012-06-07 01:17:39 +02:00
|
|
|
g_free(content);
|
2012-06-03 02:28:06 +02:00
|
|
|
return GTK_TREE_MODEL(store);
|
|
|
|
}
|
|
|
|
|
2012-06-08 22:04:48 +02:00
|
|
|
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-08 22:04:48 +02:00
|
|
|
void overlay_show(search_panel_info *info, overlay_mode mode, bool complete) {
|
2012-06-04 05:37:15 +02:00
|
|
|
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);
|
|
|
|
|
2012-06-06 01:28:55 +02:00
|
|
|
gtk_entry_completion_set_inline_selection(completion, TRUE);
|
2012-06-04 05:37:15 +02:00
|
|
|
gtk_entry_completion_set_text_column(completion, 0);
|
|
|
|
}
|
|
|
|
|
2012-06-04 06:24:39 +02:00
|
|
|
gtk_entry_set_text(GTK_ENTRY(info->entry), "");
|
|
|
|
|
2012-06-04 05:37:15 +02:00
|
|
|
info->mode = mode;
|
2012-06-20 17:25:04 +02:00
|
|
|
gtk_widget_show(info->panel);
|
2012-06-04 05:37:15 +02:00
|
|
|
gtk_widget_grab_focus(info->entry);
|
|
|
|
}
|
|
|
|
|
2012-06-08 22:04:48 +02:00
|
|
|
void get_vte_padding(VteTerminal *vte, int *w, int *h) {
|
2012-05-22 13:15:28 +02:00
|
|
|
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);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2012-06-08 22:04:48 +02:00
|
|
|
char *check_match(VteTerminal *vte, int event_x, int event_y) {
|
2012-05-22 13:15:28 +02:00
|
|
|
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);
|
|
|
|
}
|
|
|
|
|
2012-06-08 22:04:48 +02:00
|
|
|
/* {{{ CONFIG LOADING */
|
2012-06-07 23:56:17 +02:00
|
|
|
#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)
|
2012-06-16 06:48:16 +02:00
|
|
|
MAKE_GET_CONFIG_FUNCTION(integer, int)
|
|
|
|
MAKE_GET_CONFIG_FUNCTION(string, char *)
|
|
|
|
MAKE_GET_CONFIG_FUNCTION(double, double)
|
2012-06-07 18:41:48 +02:00
|
|
|
|
2012-06-20 00:56:00 +02:00
|
|
|
static bool get_config_color(GKeyFile *config, const char *key, GdkColor *color) {
|
|
|
|
char *cfgstr;
|
|
|
|
bool success = false;
|
|
|
|
if (get_config_string(config, "colors", key, &cfgstr)) {
|
|
|
|
if (gdk_color_parse(cfgstr, color)) {
|
|
|
|
success = true;
|
|
|
|
} else {
|
|
|
|
g_printerr("invalid color string: %s\n", cfgstr);
|
|
|
|
}
|
|
|
|
g_free(cfgstr);
|
|
|
|
}
|
|
|
|
return success;
|
|
|
|
}
|
|
|
|
|
2012-06-11 20:48:51 +02:00
|
|
|
static void load_config(GtkWindow *window, VteTerminal *vte,
|
2012-06-07 20:15:31 +02:00
|
|
|
gboolean *dynamic_title, gboolean *urgent_on_bell,
|
2012-06-16 06:48:16 +02:00
|
|
|
gboolean *clickable_url, const char **term) {
|
2012-06-08 01:25:04 +02:00
|
|
|
|
2012-06-25 09:13:09 +02:00
|
|
|
static const char * const filename = "termite.cfg";
|
2012-06-16 06:48:16 +02:00
|
|
|
const char *dir = g_get_user_config_dir();
|
|
|
|
char *path = g_build_filename(dir, filename, NULL);
|
2012-06-07 18:22:24 +02:00
|
|
|
GKeyFile *config = g_key_file_new();
|
2012-06-08 02:08:34 +02:00
|
|
|
|
|
|
|
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))) {
|
2012-06-07 23:56:17 +02:00
|
|
|
gboolean cfgbool;
|
2012-06-16 06:48:16 +02:00
|
|
|
double cfgdouble;
|
|
|
|
int cfgint;
|
|
|
|
char *cfgstr;
|
2012-06-07 23:56:17 +02:00
|
|
|
|
2012-06-11 20:48:51 +02:00
|
|
|
if (term && get_config_string(config, "options", "term", &cfgstr)) {
|
2012-06-08 19:19:10 +02:00
|
|
|
*term = cfgstr;
|
|
|
|
}
|
2012-06-08 18:44:08 +02:00
|
|
|
if (get_config_boolean(config, "options", "resize_grip", &cfgbool)) {
|
2012-06-07 23:56:17 +02:00
|
|
|
gtk_window_set_has_resize_grip(window, cfgbool);
|
2012-06-08 18:44:08 +02:00
|
|
|
}
|
|
|
|
if (get_config_boolean(config, "options", "scroll_on_output", &cfgbool)) {
|
2012-06-07 23:56:17 +02:00
|
|
|
vte_terminal_set_scroll_on_output(vte, cfgbool);
|
2012-06-08 18:44:08 +02:00
|
|
|
}
|
|
|
|
if (get_config_boolean(config, "options", "scroll_on_keystroke", &cfgbool)) {
|
2012-06-07 23:56:17 +02:00
|
|
|
vte_terminal_set_scroll_on_keystroke(vte, cfgbool);
|
2012-06-08 18:44:08 +02:00
|
|
|
}
|
|
|
|
if (get_config_boolean(config, "options", "audible_bell", &cfgbool)) {
|
2012-06-07 23:56:17 +02:00
|
|
|
vte_terminal_set_audible_bell(vte, cfgbool);
|
2012-06-08 18:44:08 +02:00
|
|
|
}
|
|
|
|
if (get_config_boolean(config, "options", "visible_bell", &cfgbool)) {
|
2012-06-07 23:56:17 +02:00
|
|
|
vte_terminal_set_visible_bell(vte, cfgbool);
|
2012-06-08 18:44:08 +02:00
|
|
|
}
|
|
|
|
if (get_config_boolean(config, "options", "mouse_autohide", &cfgbool)) {
|
2012-06-07 23:56:17 +02:00
|
|
|
vte_terminal_set_mouse_autohide(vte, cfgbool);
|
2012-06-08 18:44:08 +02:00
|
|
|
}
|
|
|
|
if (get_config_boolean(config, "options", "allow_bold", &cfgbool)) {
|
2012-06-08 17:56:50 +02:00
|
|
|
vte_terminal_set_allow_bold(vte, cfgbool);
|
2012-06-08 18:44:08 +02:00
|
|
|
}
|
|
|
|
if (get_config_boolean(config, "options", "dynamic_title", &cfgbool)) {
|
2012-06-07 23:56:17 +02:00
|
|
|
*dynamic_title = cfgbool;
|
2012-06-08 18:44:08 +02:00
|
|
|
}
|
|
|
|
if (get_config_boolean(config, "options", "urgent_on_bell", &cfgbool)) {
|
2012-06-07 23:56:17 +02:00
|
|
|
*urgent_on_bell = cfgbool;
|
2012-06-08 18:44:08 +02:00
|
|
|
}
|
|
|
|
if (get_config_boolean(config, "options", "clickable_url", &cfgbool)) {
|
2012-06-07 23:56:17 +02:00
|
|
|
*clickable_url = cfgbool;
|
2012-06-08 18:44:08 +02:00
|
|
|
}
|
2012-07-05 05:41:53 +02:00
|
|
|
if (get_config_boolean(config, "options", "search_wrap", &cfgbool)) {
|
|
|
|
vte_terminal_search_set_wrap_around(vte, cfgbool);
|
|
|
|
}
|
2012-06-08 21:47:57 +02:00
|
|
|
|
2012-06-11 20:32:44 +02:00
|
|
|
g_free(browser_cmd[0]);
|
2012-06-08 01:31:07 +02:00
|
|
|
if (get_config_string(config, "options", "browser", &cfgstr)) {
|
|
|
|
browser_cmd[0] = cfgstr;
|
|
|
|
} else {
|
2012-06-08 21:47:57 +02:00
|
|
|
browser_cmd[0] = g_strdup(g_getenv("BROWSER"));
|
2012-06-08 01:31:07 +02:00
|
|
|
if (!browser_cmd[0]) *clickable_url = false;
|
|
|
|
}
|
|
|
|
|
2012-06-07 23:56:17 +02:00
|
|
|
if (get_config_string(config, "options", "font", &cfgstr)) {
|
|
|
|
vte_terminal_set_font_from_string(vte, cfgstr);
|
|
|
|
g_free(cfgstr);
|
2012-06-07 18:55:08 +02:00
|
|
|
}
|
2012-06-07 18:58:07 +02:00
|
|
|
|
2012-06-07 23:56:17 +02:00
|
|
|
if (get_config_integer(config, "options", "scrollback_lines", &cfgint)) {
|
|
|
|
vte_terminal_set_scrollback_lines(vte, cfgint);
|
2012-06-07 18:58:07 +02:00
|
|
|
}
|
2012-06-07 19:36:03 +02:00
|
|
|
|
2012-06-07 23:56:17 +02:00
|
|
|
if (get_config_string(config, "options", "cursor_blink", &cfgstr)) {
|
2012-06-08 20:52:17 +02:00
|
|
|
if (!g_ascii_strcasecmp(cfgstr, "system")) {
|
2012-06-07 19:36:03 +02:00
|
|
|
vte_terminal_set_cursor_blink_mode(vte, VTE_CURSOR_BLINK_SYSTEM);
|
2012-06-08 20:52:17 +02:00
|
|
|
} else if (!g_ascii_strcasecmp(cfgstr, "on")) {
|
2012-06-07 19:36:03 +02:00
|
|
|
vte_terminal_set_cursor_blink_mode(vte, VTE_CURSOR_BLINK_ON);
|
2012-06-08 20:52:17 +02:00
|
|
|
} else if (!g_ascii_strcasecmp(cfgstr, "off")) {
|
2012-06-07 19:36:03 +02:00
|
|
|
vte_terminal_set_cursor_blink_mode(vte, VTE_CURSOR_BLINK_OFF);
|
|
|
|
}
|
2012-06-07 23:56:17 +02:00
|
|
|
g_free(cfgstr);
|
2012-06-07 19:36:03 +02:00
|
|
|
}
|
|
|
|
|
2012-06-07 23:56:17 +02:00
|
|
|
if (get_config_string(config, "options", "cursor_shape", &cfgstr)) {
|
2012-06-08 20:52:17 +02:00
|
|
|
if (!g_ascii_strcasecmp(cfgstr, "block")) {
|
2012-06-07 19:36:03 +02:00
|
|
|
vte_terminal_set_cursor_shape(vte, VTE_CURSOR_SHAPE_BLOCK);
|
2012-06-08 20:52:17 +02:00
|
|
|
} else if (!g_ascii_strcasecmp(cfgstr, "ibeam")) {
|
2012-06-07 19:36:03 +02:00
|
|
|
vte_terminal_set_cursor_shape(vte, VTE_CURSOR_SHAPE_IBEAM);
|
2012-06-08 20:52:17 +02:00
|
|
|
} else if (!g_ascii_strcasecmp(cfgstr, "underline")) {
|
2012-06-07 19:36:03 +02:00
|
|
|
vte_terminal_set_cursor_shape(vte, VTE_CURSOR_SHAPE_UNDERLINE);
|
|
|
|
}
|
2012-06-07 23:56:17 +02:00
|
|
|
g_free(cfgstr);
|
2012-06-07 19:36:03 +02:00
|
|
|
}
|
2012-06-07 20:10:16 +02:00
|
|
|
|
2012-06-07 23:56:17 +02:00
|
|
|
if (get_config_string(config, "options", "icon_name", &cfgstr)) {
|
|
|
|
gtk_window_set_icon_name(window, cfgstr);
|
|
|
|
g_free(cfgstr);
|
2012-06-07 20:10:16 +02:00
|
|
|
}
|
2012-06-07 20:36:28 +02:00
|
|
|
|
2012-06-08 03:05:31 +02:00
|
|
|
if (get_config_double(config, "options", "transparency", &cfgdouble)) {
|
2012-06-19 22:11:48 +02:00
|
|
|
vte_terminal_set_background_saturation(vte, cfgdouble);
|
|
|
|
vte_terminal_set_opacity(vte, (guint16)(0xffff * (1 - cfgdouble)));
|
2012-06-08 03:05:31 +02:00
|
|
|
}
|
|
|
|
|
2012-06-25 22:05:23 +02:00
|
|
|
static const long palette_size = 255;
|
2012-06-25 05:15:50 +02:00
|
|
|
GdkColor color, palette[palette_size];
|
2012-06-07 21:07:42 +02:00
|
|
|
|
2012-06-25 22:05:23 +02:00
|
|
|
char color_key[] = "color000";
|
2012-06-07 21:22:28 +02:00
|
|
|
|
2012-06-07 22:50:52 +02:00
|
|
|
bool success = true;
|
2012-06-25 22:05:23 +02:00
|
|
|
for (unsigned i = 0; success && i < palette_size; i++) {
|
|
|
|
snprintf(color_key, sizeof color_key, "color%u", i);
|
|
|
|
if (get_config_string(config, "colors", color_key, &cfgstr)) {
|
|
|
|
if (!gdk_color_parse(cfgstr, &palette[i])) {
|
|
|
|
g_printerr("invalid color string: %s\n", cfgstr);
|
|
|
|
success = false;
|
|
|
|
}
|
2012-06-19 21:59:36 +02:00
|
|
|
} else {
|
2012-06-25 22:05:23 +02:00
|
|
|
if (i < 16) {
|
|
|
|
palette[i].blue = (i & 4) ? 0xc000 : 0;
|
|
|
|
palette[i].green = (i & 2) ? 0xc000 : 0;
|
|
|
|
palette[i].red = (i & 1) ? 0xc000 : 0;
|
|
|
|
if (i > 7) {
|
2012-06-27 03:30:34 +02:00
|
|
|
palette[i].blue = (guint16)(palette[i].blue + 0x3fff);
|
|
|
|
palette[i].green = (guint16)(palette[i].green + 0x3fff);
|
|
|
|
palette[i].red = (guint16)(palette[i].red + 0x3fff);
|
2012-06-25 22:05:23 +02:00
|
|
|
}
|
2012-06-25 22:08:53 +02:00
|
|
|
} else if (i < 232) {
|
2012-07-08 07:46:44 +02:00
|
|
|
const unsigned j = i - 16;
|
|
|
|
const unsigned r = j / 36, g = (j / 6) % 6, b = j % 6;
|
|
|
|
const unsigned red = (r == 0) ? 0 : r * 40 + 55;
|
|
|
|
const unsigned green = (g == 0) ? 0 : g * 40 + 55;
|
|
|
|
const unsigned blue = (b == 0) ? 0 : b * 40 + 55;
|
2012-06-27 03:30:34 +02:00
|
|
|
palette[i].red = (guint16)(red | red << 8);
|
|
|
|
palette[i].green = (guint16)(green | green << 8);
|
|
|
|
palette[i].blue = (guint16)(blue | blue << 8);
|
2012-06-25 22:05:23 +02:00
|
|
|
} else if (i < 256) {
|
2012-07-08 07:46:44 +02:00
|
|
|
const unsigned shade = 8 + (i - 232) * 10;
|
2012-06-27 03:30:34 +02:00
|
|
|
palette[i].red = palette[i].green = palette[i].blue = (guint16)(shade | shade << 8);
|
2012-06-25 22:05:23 +02:00
|
|
|
}
|
2012-06-07 21:22:28 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2012-06-25 08:18:12 +02:00
|
|
|
if (success) {
|
|
|
|
vte_terminal_set_colors(vte, NULL, NULL, palette, palette_size);
|
|
|
|
}
|
|
|
|
|
2012-06-20 00:56:00 +02:00
|
|
|
if (get_config_color(config, "foreground", &color)) {
|
|
|
|
vte_terminal_set_color_foreground(vte, &color);
|
2012-06-07 20:43:41 +02:00
|
|
|
}
|
|
|
|
|
2012-06-25 21:27:21 +02:00
|
|
|
if (get_config_color(config, "foreground_bold", &color)) {
|
|
|
|
vte_terminal_set_color_bold(vte, &color);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (get_config_color(config, "foreground_dim", &color)) {
|
|
|
|
vte_terminal_set_color_dim(vte, &color);
|
|
|
|
}
|
|
|
|
|
2012-06-20 00:56:00 +02:00
|
|
|
if (get_config_color(config, "background", &color)) {
|
|
|
|
vte_terminal_set_color_background(vte, &color);
|
2012-06-07 20:43:41 +02:00
|
|
|
}
|
2012-06-07 20:39:02 +02:00
|
|
|
|
2012-06-20 00:56:00 +02:00
|
|
|
if (get_config_color(config, "cursor", &color)) {
|
|
|
|
vte_terminal_set_color_cursor(vte, &color);
|
2012-06-07 20:39:02 +02:00
|
|
|
}
|
2012-06-07 18:22:24 +02:00
|
|
|
}
|
2012-06-08 01:25:04 +02:00
|
|
|
g_free(path);
|
2012-06-07 18:22:24 +02:00
|
|
|
g_key_file_free(config);
|
2012-06-08 22:04:48 +02:00
|
|
|
}/*}}}*/
|
2012-06-07 18:22:24 +02:00
|
|
|
|
2012-05-22 06:34:35 +02:00
|
|
|
int main(int argc, char **argv) {
|
|
|
|
GError *error = NULL;
|
2012-06-25 06:39:32 +02:00
|
|
|
const char *term = "termite";
|
2012-06-08 22:04:48 +02:00
|
|
|
gboolean dynamic_title = FALSE, urgent_on_bell = FALSE, clickable_url = FALSE;
|
2012-06-20 00:55:56 +02:00
|
|
|
gboolean version = FALSE;
|
2012-05-22 06:34:35 +02:00
|
|
|
|
2012-06-16 06:31:50 +02:00
|
|
|
GOptionContext *context = g_option_context_new(NULL);
|
2012-06-16 06:33:19 +02:00
|
|
|
char *role = NULL, *geometry = NULL, *execute = NULL;
|
2012-05-31 15:22:13 +02:00
|
|
|
const GOptionEntry entries[] = {
|
2012-06-11 20:26:51 +02:00
|
|
|
{"role", 'r', 0, G_OPTION_ARG_STRING, &role, "The role to use", "ROLE"},
|
2012-06-12 05:50:45 +02:00
|
|
|
{"geometry", 0, 0, G_OPTION_ARG_STRING, &geometry, "Window geometry", "GEOMETRY"},
|
2012-06-16 06:31:50 +02:00
|
|
|
{"exec", 'e', 0, G_OPTION_ARG_STRING, &execute, "Command to execute", "COMMAND"},
|
2012-06-20 01:01:27 +02:00
|
|
|
{"version", 'v', 0, G_OPTION_ARG_NONE, &version, "Version info", NULL},
|
2012-06-11 20:26:51 +02:00
|
|
|
{NULL}
|
2012-05-31 15:22:13 +02:00
|
|
|
};
|
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-06-20 00:54:38 +02:00
|
|
|
if (version) {
|
|
|
|
g_print("termite %s\n", TERMITE_VERSION);
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2012-05-22 17:50:01 +02:00
|
|
|
GtkWidget *window = gtk_window_new(GTK_WINDOW_TOPLEVEL);
|
2012-06-08 22:04:48 +02:00
|
|
|
GtkWidget *overlay = gtk_overlay_new();
|
|
|
|
GtkWidget *vte = vte_terminal_new();
|
2012-05-30 12:54:10 +02:00
|
|
|
|
2012-06-11 22:38:01 +02:00
|
|
|
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);
|
|
|
|
|
2012-05-31 11:43:18 +02:00
|
|
|
if (role) {
|
|
|
|
gtk_window_set_role(GTK_WINDOW(window), role);
|
2012-06-11 06:03:19 +02:00
|
|
|
g_free(role);
|
2012-05-31 11:43:18 +02:00
|
|
|
}
|
|
|
|
|
2012-05-24 04:50:01 +02:00
|
|
|
char **command_argv;
|
2012-06-08 22:01:36 +02:00
|
|
|
char fallback[] = "/bin/sh";
|
|
|
|
char *default_argv[2] = {fallback, NULL};
|
2012-05-24 04:50:01 +02:00
|
|
|
|
2012-06-16 06:31:50 +02:00
|
|
|
if (execute) {
|
2012-06-16 06:48:16 +02:00
|
|
|
int argcp;
|
|
|
|
char **argvp;
|
2012-06-16 06:31:50 +02:00
|
|
|
g_shell_parse_argv(execute, &argcp, &argvp, &error);
|
|
|
|
if (error) {
|
|
|
|
g_printerr("Failed to parse command: %s\n", error->message);
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
command_argv = argvp;
|
2012-05-24 04:50:01 +02:00
|
|
|
} else {
|
2012-06-08 22:01:36 +02:00
|
|
|
char *shell = vte_get_user_shell();
|
|
|
|
if (shell) default_argv[0] = shell;
|
2012-05-24 04:50:01 +02:00
|
|
|
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;
|
|
|
|
}
|
|
|
|
|
2012-06-11 20:48:51 +02:00
|
|
|
load_config(GTK_WINDOW(window), VTE_TERMINAL(vte), &dynamic_title,
|
2012-06-11 22:38:01 +02:00
|
|
|
&urgent_on_bell, &clickable_url, &term);
|
2012-06-08 19:19:10 +02:00
|
|
|
|
2012-05-22 06:34:35 +02:00
|
|
|
vte_terminal_set_pty_object(VTE_TERMINAL(vte), pty);
|
|
|
|
vte_pty_set_term(pty, term);
|
|
|
|
|
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-07-09 09:10:08 +02:00
|
|
|
select_info select = {SELECT_OFF, 0, 0, 0, 0};
|
2012-07-05 08:40:48 +02:00
|
|
|
search_panel_info info = {vte, entry, alignment, OVERLAY_HIDDEN, select};
|
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);
|
2012-06-04 05:17:18 +02:00
|
|
|
g_signal_connect(entry, "key-press-event", G_CALLBACK(entry_key_press_cb), &info);
|
2012-06-01 03:31:09 +02:00
|
|
|
g_signal_connect(overlay, "get-child-position", G_CALLBACK(position_overlay_cb), NULL);
|
2012-05-22 06:34:35 +02:00
|
|
|
|
2012-06-07 20:15:31 +02:00
|
|
|
if (clickable_url) {
|
2012-06-20 01:13:55 +02:00
|
|
|
int tag = vte_terminal_match_add_gregex(VTE_TERMINAL(vte),
|
2012-06-07 20:15:31 +02:00
|
|
|
g_regex_new(url_regex,
|
|
|
|
G_REGEX_CASELESS,
|
|
|
|
G_REGEX_MATCH_NOTEMPTY,
|
|
|
|
NULL),
|
|
|
|
(GRegexMatchFlags)0);
|
2012-06-20 01:13:55 +02:00
|
|
|
vte_terminal_match_set_cursor_type(VTE_TERMINAL(vte), tag, GDK_HAND2);
|
2012-06-07 20:15:31 +02:00
|
|
|
g_signal_connect(vte, "button-press-event", G_CALLBACK(button_press_cb), NULL);
|
|
|
|
}
|
2012-05-24 05:41:55 +02:00
|
|
|
|
2012-06-07 19:54:15 +02:00
|
|
|
if (urgent_on_bell) {
|
2012-06-08 22:04:48 +02:00
|
|
|
g_signal_connect_swapped(vte, "beep", G_CALLBACK(beep_cb), window);
|
2012-06-16 09:41:43 +02:00
|
|
|
g_signal_connect(window, "focus-in-event", G_CALLBACK(focus_cb), NULL);
|
|
|
|
g_signal_connect(window, "focus-out-event", G_CALLBACK(focus_cb), NULL);
|
2012-06-07 19:54:15 +02:00
|
|
|
}
|
2012-05-22 13:47:51 +02:00
|
|
|
|
2012-06-07 19:54:15 +02:00
|
|
|
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);
|
|
|
|
}
|
2012-05-22 14:22:31 +02:00
|
|
|
|
2012-06-12 05:50:45 +02:00
|
|
|
if (geometry) {
|
|
|
|
gtk_widget_show_all(overlay);
|
|
|
|
gtk_widget_show_all(alignment);
|
|
|
|
if (!gtk_window_parse_geometry(GTK_WINDOW(window), geometry)) {
|
|
|
|
g_printerr("Invalid geometry string: %s\n", geometry);
|
|
|
|
}
|
|
|
|
g_free(geometry);
|
|
|
|
}
|
|
|
|
|
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-07-01 19:44:30 +02:00
|
|
|
|
|
|
|
GdkWindow *gdk_window = gtk_widget_get_window(window);
|
|
|
|
if (!gdk_window) {
|
|
|
|
g_printerr("no window");
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
char *xid_s = g_strdup_printf("%lu", GDK_WINDOW_XID(gdk_window));
|
|
|
|
char **env = g_get_environ();
|
|
|
|
env = g_environ_setenv(env, "WINDOWID", xid_s, TRUE);
|
|
|
|
env = g_environ_setenv(env, "TERM", term, TRUE);
|
|
|
|
g_free(xid_s);
|
|
|
|
|
|
|
|
GPid ppid;
|
|
|
|
if (g_spawn_async(NULL, command_argv, env,
|
|
|
|
(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;
|
|
|
|
}
|
|
|
|
|
|
|
|
g_strfreev(env);
|
|
|
|
|
2012-05-22 06:34:35 +02:00
|
|
|
gtk_main();
|
2012-06-20 01:20:49 +02:00
|
|
|
return vte_terminal_get_child_exit_status(VTE_TERMINAL(vte));
|
2012-05-22 06:34:35 +02:00
|
|
|
}
|
2012-06-08 18:56:46 +02:00
|
|
|
|
|
|
|
// vim: et:sts=4:sw=4:cino=(0
|