Now able to open multiple terminal tabs
This commit is contained in:
parent
85f147115e
commit
38559e6ee7
104
src/terminal.c
104
src/terminal.c
|
@ -20,6 +20,7 @@
|
||||||
|
|
||||||
#include <sys/wait.h>
|
#include <sys/wait.h>
|
||||||
#include <stdlib.h>
|
#include <stdlib.h>
|
||||||
|
#include <string.h>
|
||||||
#include <gtk/gtk.h>
|
#include <gtk/gtk.h>
|
||||||
#include <gdk/gdkkeysyms.h>
|
#include <gdk/gdkkeysyms.h>
|
||||||
#include <System.h>
|
#include <System.h>
|
||||||
|
@ -60,6 +61,9 @@ struct _TerminalTab
|
||||||
|
|
||||||
|
|
||||||
/* prototypes */
|
/* prototypes */
|
||||||
|
/* useful */
|
||||||
|
static int _terminal_open_tab(Terminal * terminal);
|
||||||
|
|
||||||
/* callbacks */
|
/* callbacks */
|
||||||
static void _terminal_on_child_watch(GPid pid, gint status, gpointer data);
|
static void _terminal_on_child_watch(GPid pid, gint status, gpointer data);
|
||||||
static gboolean _terminal_on_closex(gpointer data);
|
static gboolean _terminal_on_closex(gpointer data);
|
||||||
|
@ -67,11 +71,15 @@ static void _terminal_on_close(gpointer data);
|
||||||
static gboolean _terminal_on_closex(gpointer data);
|
static gboolean _terminal_on_closex(gpointer data);
|
||||||
|
|
||||||
static void _terminal_on_file_close(gpointer data);
|
static void _terminal_on_file_close(gpointer data);
|
||||||
|
static void _terminal_on_file_new_tab(gpointer data);
|
||||||
|
|
||||||
/* constants */
|
/* constants */
|
||||||
/* menubar */
|
/* menubar */
|
||||||
static const DesktopMenu _terminal_file_menu[] =
|
static const DesktopMenu _terminal_file_menu[] =
|
||||||
{
|
{
|
||||||
|
{ "_New tab", G_CALLBACK(_terminal_on_file_new_tab), "tab-new",
|
||||||
|
GDK_CONTROL_MASK, GDK_KEY_T },
|
||||||
|
{ "", NULL, NULL, 0, 0 },
|
||||||
{ "_Close", G_CALLBACK(_terminal_on_file_close), GTK_STOCK_CLOSE,
|
{ "_Close", G_CALLBACK(_terminal_on_file_close), GTK_STOCK_CLOSE,
|
||||||
GDK_CONTROL_MASK, GDK_KEY_W },
|
GDK_CONTROL_MASK, GDK_KEY_W },
|
||||||
{ NULL, NULL, NULL, 0, 0 }
|
{ NULL, NULL, NULL, 0, 0 }
|
||||||
|
@ -93,23 +101,12 @@ Terminal * terminal_new(void)
|
||||||
GtkAccelGroup * group;
|
GtkAccelGroup * group;
|
||||||
GtkWidget * vbox;
|
GtkWidget * vbox;
|
||||||
GtkWidget * widget;
|
GtkWidget * widget;
|
||||||
char * argv[] = { BINDIR "/xterm", "xterm", "-into", NULL, NULL };
|
|
||||||
char buf[16];
|
|
||||||
int flags = G_SPAWN_FILE_AND_ARGV_ZERO | G_SPAWN_DO_NOT_REAP_CHILD;
|
|
||||||
GError * error = NULL;
|
|
||||||
|
|
||||||
if((terminal = object_new(sizeof(*terminal))) == NULL)
|
if((terminal = object_new(sizeof(*terminal))) == NULL)
|
||||||
return NULL;
|
return NULL;
|
||||||
/* FIXME really implement */
|
terminal->tabs = NULL;
|
||||||
terminal->tabs = malloc(sizeof(*terminal->tabs));
|
terminal->tabs_cnt = 0;
|
||||||
terminal->tabs_cnt = 1;
|
|
||||||
terminal->window = NULL;
|
terminal->window = NULL;
|
||||||
/* check for errors */
|
|
||||||
if(terminal->tabs == NULL)
|
|
||||||
{
|
|
||||||
terminal_delete(terminal);
|
|
||||||
return NULL;
|
|
||||||
}
|
|
||||||
/* widgets */
|
/* widgets */
|
||||||
group = gtk_accel_group_new();
|
group = gtk_accel_group_new();
|
||||||
terminal->window = gtk_window_new(GTK_WINDOW_TOPLEVEL);
|
terminal->window = gtk_window_new(GTK_WINDOW_TOPLEVEL);
|
||||||
|
@ -124,29 +121,15 @@ Terminal * terminal_new(void)
|
||||||
gtk_box_pack_start(GTK_BOX(vbox), widget, FALSE, TRUE, 0);
|
gtk_box_pack_start(GTK_BOX(vbox), widget, FALSE, TRUE, 0);
|
||||||
/* view */
|
/* view */
|
||||||
terminal->notebook = gtk_notebook_new();
|
terminal->notebook = gtk_notebook_new();
|
||||||
|
gtk_notebook_set_scrollable(GTK_NOTEBOOK(terminal->notebook), TRUE);
|
||||||
gtk_box_pack_start(GTK_BOX(vbox), terminal->notebook, TRUE, TRUE, 0);
|
gtk_box_pack_start(GTK_BOX(vbox), terminal->notebook, TRUE, TRUE, 0);
|
||||||
/* first tab */
|
|
||||||
terminal->tabs->socket = gtk_socket_new();
|
|
||||||
terminal->tabs->label = gtk_label_new("xterm");
|
|
||||||
gtk_notebook_append_page(GTK_NOTEBOOK(terminal->notebook),
|
|
||||||
terminal->tabs->socket, terminal->tabs->label);
|
|
||||||
gtk_container_add(GTK_CONTAINER(terminal->window), vbox);
|
gtk_container_add(GTK_CONTAINER(terminal->window), vbox);
|
||||||
gtk_widget_show_all(vbox);
|
gtk_widget_show_all(vbox);
|
||||||
/* launch xterm */
|
if(_terminal_open_tab(terminal) != 0)
|
||||||
snprintf(buf, sizeof(buf), "%u", gtk_socket_get_id(
|
|
||||||
GTK_SOCKET(terminal->tabs->socket)));
|
|
||||||
argv[3] = buf;
|
|
||||||
if(g_spawn_async(NULL, argv, NULL, flags, NULL, NULL,
|
|
||||||
&terminal->tabs->pid, &error) == FALSE)
|
|
||||||
{
|
{
|
||||||
fprintf(stderr, "%s: %s: %s\n", "Terminal", argv[1],
|
|
||||||
error->message);
|
|
||||||
g_error_free(error);
|
|
||||||
terminal_delete(terminal);
|
terminal_delete(terminal);
|
||||||
return NULL;
|
return NULL;
|
||||||
}
|
}
|
||||||
g_child_watch_add(terminal->tabs->pid, _terminal_on_child_watch,
|
|
||||||
terminal);
|
|
||||||
gtk_widget_show(terminal->window);
|
gtk_widget_show(terminal->window);
|
||||||
return terminal;
|
return terminal;
|
||||||
}
|
}
|
||||||
|
@ -170,13 +153,55 @@ void terminal_delete(Terminal * terminal)
|
||||||
|
|
||||||
/* private */
|
/* private */
|
||||||
/* functions */
|
/* functions */
|
||||||
|
/* useful */
|
||||||
|
/* terminal_open_tab */
|
||||||
|
static int _terminal_open_tab(Terminal * terminal)
|
||||||
|
{
|
||||||
|
TerminalTab * p;
|
||||||
|
char * argv[] = { BINDIR "/xterm", "xterm", "-into", NULL, NULL };
|
||||||
|
char buf[16];
|
||||||
|
int flags = G_SPAWN_FILE_AND_ARGV_ZERO | G_SPAWN_DO_NOT_REAP_CHILD;
|
||||||
|
GError * error = NULL;
|
||||||
|
|
||||||
|
if((p = realloc(terminal->tabs, sizeof(*p) * (terminal->tabs_cnt + 1)))
|
||||||
|
== NULL)
|
||||||
|
return -1;
|
||||||
|
terminal->tabs = p;
|
||||||
|
p = &terminal->tabs[terminal->tabs_cnt++];
|
||||||
|
/* first tab */
|
||||||
|
p->socket = gtk_socket_new();
|
||||||
|
p->label = gtk_label_new("xterm");
|
||||||
|
gtk_notebook_append_page(GTK_NOTEBOOK(terminal->notebook), p->socket,
|
||||||
|
p->label);
|
||||||
|
/* launch xterm */
|
||||||
|
snprintf(buf, sizeof(buf), "%u", gtk_socket_get_id(
|
||||||
|
GTK_SOCKET(p->socket)));
|
||||||
|
argv[3] = buf;
|
||||||
|
if(g_spawn_async(NULL, argv, NULL, flags, NULL, NULL, &p->pid, &error)
|
||||||
|
== FALSE)
|
||||||
|
{
|
||||||
|
fprintf(stderr, "%s: %s: %s\n", "Terminal", argv[1],
|
||||||
|
error->message);
|
||||||
|
g_error_free(error);
|
||||||
|
return -1;
|
||||||
|
}
|
||||||
|
g_child_watch_add(p->pid, _terminal_on_child_watch, terminal);
|
||||||
|
gtk_widget_show(p->socket);
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
/* callbacks */
|
/* callbacks */
|
||||||
/* terminal_on_child_watch */
|
/* terminal_on_child_watch */
|
||||||
static void _terminal_on_child_watch(GPid pid, gint status, gpointer data)
|
static void _terminal_on_child_watch(GPid pid, gint status, gpointer data)
|
||||||
{
|
{
|
||||||
Terminal * terminal = data;
|
Terminal * terminal = data;
|
||||||
|
size_t i;
|
||||||
|
|
||||||
if(terminal->tabs->pid != pid)
|
for(i = 0; i < terminal->tabs_cnt; i++)
|
||||||
|
if(terminal->tabs[i].pid == pid)
|
||||||
|
break;
|
||||||
|
if(i == terminal->tabs_cnt)
|
||||||
return;
|
return;
|
||||||
if(WIFEXITED(status))
|
if(WIFEXITED(status))
|
||||||
{
|
{
|
||||||
|
@ -184,14 +209,20 @@ static void _terminal_on_child_watch(GPid pid, gint status, gpointer data)
|
||||||
fprintf(stderr, "%s: %s%u\n", "Terminal",
|
fprintf(stderr, "%s: %s%u\n", "Terminal",
|
||||||
"xterm exited with status ",
|
"xterm exited with status ",
|
||||||
WEXITSTATUS(status));
|
WEXITSTATUS(status));
|
||||||
gtk_main_quit();
|
|
||||||
}
|
}
|
||||||
else if(WIFSIGNALED(status))
|
else if(WIFSIGNALED(status))
|
||||||
{
|
{
|
||||||
fprintf(stderr, "%s: %s%u\n", "Terminal",
|
fprintf(stderr, "%s: %s%u\n", "Terminal",
|
||||||
"xterm exited with signal ", WTERMSIG(status));
|
"xterm exited with signal ", WTERMSIG(status));
|
||||||
gtk_main_quit();
|
|
||||||
}
|
}
|
||||||
|
else
|
||||||
|
return;
|
||||||
|
gtk_notebook_remove_page(GTK_NOTEBOOK(terminal->notebook), i);
|
||||||
|
memmove(&terminal->tabs[i], &terminal->tabs[i + 1],
|
||||||
|
(terminal->tabs_cnt - (i + 1))
|
||||||
|
* sizeof(*terminal->tabs));
|
||||||
|
if(--terminal->tabs_cnt == 0)
|
||||||
|
gtk_main_quit();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
@ -222,3 +253,12 @@ static void _terminal_on_file_close(gpointer data)
|
||||||
|
|
||||||
_terminal_on_close(terminal);
|
_terminal_on_close(terminal);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
/* terminal_on_file_new_tab */
|
||||||
|
static void _terminal_on_file_new_tab(gpointer data)
|
||||||
|
{
|
||||||
|
Terminal * terminal = data;
|
||||||
|
|
||||||
|
_terminal_open_tab(terminal);
|
||||||
|
}
|
||||||
|
|
Loading…
Reference in New Issue
Block a user