mssh-window.c 9.3 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358
  1. #include <string.h>
  2. #include <stdlib.h>
  3. #include <gdk/gdkkeysyms.h>
  4. #include "mssh-terminal.h"
  5. #include "mssh-pref.h"
  6. #include "mssh-window.h"
  7. #include "config.h"
  8. static void mssh_window_sendhost(GtkWidget *widget, gpointer data);
  9. static void mssh_window_destroy(GtkWidget *widget, gpointer data);
  10. static void mssh_window_pref(GtkWidget *widget, gpointer data);
  11. static gboolean mssh_window_key_press(GtkWidget *widget,
  12. GdkEventKey *event, gpointer data);
  13. static gboolean mssh_window_entry_focused(GtkWidget *widget,
  14. GtkDirectionType dir, gpointer data);
  15. static gboolean mssh_window_session_close(gpointer data);
  16. static void mssh_window_session_closed(MSSHTerminal *terminal,
  17. gpointer data);
  18. static void mssh_window_session_focused(MSSHTerminal *terminal,
  19. gpointer data);
  20. static void mssh_window_insert(GtkWidget *widget, gchar *new_text,
  21. gint new_text_length, gint *position, gpointer data);
  22. static void mssh_window_relayout(MSSHWindow *window);
  23. static void mssh_window_add_session(MSSHWindow *window, char *hostname);
  24. static void mssh_window_init(MSSHWindow* window);
  25. static void mssh_window_class_init(MSSHWindowClass *klass);
  26. G_DEFINE_TYPE(MSSHWindow, mssh_window, GTK_TYPE_WINDOW)
  27. struct WinTermPair
  28. {
  29. MSSHWindow *window;
  30. MSSHTerminal *terminal;
  31. };
  32. GtkWidget* mssh_window_new(void)
  33. {
  34. return g_object_new(MSSH_TYPE_WINDOW, NULL);
  35. }
  36. static void mssh_window_sendhost(GtkWidget *widget, gpointer data)
  37. {
  38. int i;
  39. MSSHWindow *window = MSSH_WINDOW(data);
  40. for(i = 0; i < window->terminals->len; i++)
  41. {
  42. mssh_terminal_send_host(g_array_index(window->terminals,
  43. MSSHTerminal*, i));
  44. }
  45. }
  46. static void mssh_window_destroy(GtkWidget *widget, gpointer data)
  47. {
  48. int i;
  49. MSSHWindow *window = MSSH_WINDOW(data);
  50. if(window->terminals->len > 0)
  51. {
  52. GtkWidget *dialog = gtk_message_dialog_new(GTK_WINDOW(window),
  53. GTK_DIALOG_DESTROY_WITH_PARENT, GTK_MESSAGE_QUESTION,
  54. GTK_BUTTONS_YES_NO, "%s, %s",
  55. "You still have open sessions",
  56. "are you sure you wish to quit?");
  57. if(gtk_dialog_run(GTK_DIALOG(dialog)) == GTK_RESPONSE_YES)
  58. {
  59. for(i = 0; i < window->terminals->len; i++)
  60. {
  61. mssh_terminal_destroy(g_array_index(window->terminals,
  62. MSSHTerminal*, i));
  63. }
  64. g_array_free(window->terminals, TRUE);
  65. gtk_main_quit();
  66. }
  67. gtk_widget_destroy(dialog);
  68. }
  69. else
  70. {
  71. gtk_main_quit();
  72. }
  73. }
  74. static void mssh_window_pref(GtkWidget *widget, gpointer data)
  75. {
  76. MSSHWindow *window = MSSH_WINDOW(data);
  77. GtkWidget *pref = mssh_pref_new();
  78. gtk_window_set_transient_for(GTK_WINDOW(pref), GTK_WINDOW(window));
  79. gtk_window_set_modal(GTK_WINDOW(pref), TRUE);
  80. gtk_window_set_position(GTK_WINDOW(pref),
  81. GTK_WIN_POS_CENTER_ON_PARENT);
  82. gtk_widget_show_all(pref);
  83. }
  84. static void mssh_window_insert(GtkWidget *widget, gchar *new_text,
  85. gint new_text_length, gint *position, gpointer data)
  86. {
  87. int i;
  88. MSSHWindow *window = MSSH_WINDOW(data);
  89. for(i = 0; i < window->terminals->len; i++)
  90. {
  91. mssh_terminal_send_string(g_array_index(window->terminals,
  92. MSSHTerminal*, i), new_text);
  93. }
  94. g_signal_stop_emission_by_name(G_OBJECT(widget), "insert-text");
  95. }
  96. static gboolean mssh_window_key_press(GtkWidget *widget,
  97. GdkEventKey *event, gpointer data)
  98. {
  99. int i;
  100. MSSHWindow *window = MSSH_WINDOW(data);
  101. for(i = 0; i < window->terminals->len; i++)
  102. {
  103. mssh_terminal_send_data(g_array_index(window->terminals,
  104. MSSHTerminal*, i), event);
  105. }
  106. return TRUE;
  107. }
  108. static gboolean mssh_window_entry_focused(GtkWidget *widget,
  109. GtkDirectionType dir, gpointer data)
  110. {
  111. MSSHWindow *window = MSSH_WINDOW(data);
  112. gtk_window_set_title(GTK_WINDOW(window), PACKAGE_NAME" - All");
  113. return FALSE;
  114. }
  115. static gboolean mssh_window_session_close(gpointer data)
  116. {
  117. int i, idx = -1;
  118. struct WinTermPair *data_pair = (struct WinTermPair*)data;
  119. for(i = 0; i < data_pair->window->terminals->len; i++)
  120. {
  121. if(data_pair->terminal == g_array_index(
  122. data_pair->window->terminals, MSSHTerminal*, i))
  123. {
  124. idx = i;
  125. break;
  126. }
  127. }
  128. if(idx == -1)
  129. {
  130. fprintf(stderr,
  131. "mssh: Fatal Error: Can't find terminal to remove!\n");
  132. }
  133. else
  134. {
  135. gtk_widget_destroy(data_pair->terminal->menu_item);
  136. gtk_container_remove(GTK_CONTAINER(data_pair->window->table),
  137. GTK_WIDGET(data_pair->terminal));
  138. g_array_remove_index(data_pair->window->terminals, idx);
  139. mssh_window_relayout(data_pair->window);
  140. }
  141. free(data_pair);
  142. return FALSE;
  143. }
  144. static void mssh_window_session_closed(MSSHTerminal *terminal,
  145. gpointer data)
  146. {
  147. struct WinTermPair *data_pair = malloc(sizeof(struct WinTermPair));
  148. data_pair->terminal = terminal;
  149. data_pair->window = MSSH_WINDOW(data);
  150. g_timeout_add_seconds(2, mssh_window_session_close, data_pair);
  151. }
  152. static void mssh_window_session_focused(MSSHTerminal *terminal,
  153. gpointer data)
  154. {
  155. char *title;
  156. size_t len;
  157. MSSHWindow *window = MSSH_WINDOW(data);
  158. len = strlen(PACKAGE_NAME" - ") + strlen(terminal->hostname) + 1;
  159. title = malloc(len);
  160. snprintf(title, len, PACKAGE_NAME" - %s", terminal->hostname);
  161. gtk_window_set_title(GTK_WINDOW(window), title);
  162. free(title);
  163. }
  164. static void mssh_window_relayout(MSSHWindow *window)
  165. {
  166. int i, len = window->terminals->len;
  167. for(i = 0; i < len; i++)
  168. {
  169. MSSHTerminal *terminal = g_array_index(window->terminals,
  170. MSSHTerminal*, i);
  171. g_object_ref(terminal);
  172. if(GTK_WIDGET(terminal)->parent == GTK_WIDGET(window->table))
  173. {
  174. gtk_container_remove(GTK_CONTAINER(window->table),
  175. GTK_WIDGET(terminal));
  176. }
  177. gtk_table_attach(GTK_TABLE(window->table), GTK_WIDGET(terminal),
  178. (i % 2), (i == len - 1) ? 2 : (i % 2) + 1, i / 2, (i / 2) + 1,
  179. GTK_FILL | GTK_EXPAND, GTK_FILL | GTK_EXPAND, 2, 2);
  180. g_object_unref(terminal);
  181. if(!terminal->started)
  182. {
  183. mssh_terminal_start_session(terminal, window->env);
  184. terminal->started = 1;
  185. }
  186. }
  187. if(len > 0)
  188. {
  189. gtk_table_resize(GTK_TABLE(window->table), ((len + 1) / 2), 2);
  190. }
  191. }
  192. static void mssh_window_add_session(MSSHWindow *window, char *hostname)
  193. {
  194. MSSHTerminal *terminal = MSSH_TERMINAL(mssh_terminal_new());
  195. g_array_append_val(window->terminals, terminal);
  196. g_signal_connect(G_OBJECT(terminal), "session-closed",
  197. G_CALLBACK(mssh_window_session_closed), window);
  198. g_signal_connect(G_OBJECT(terminal), "session-focused",
  199. G_CALLBACK(mssh_window_session_focused), window);
  200. mssh_terminal_init_session(terminal, hostname);
  201. gtk_menu_shell_append(GTK_MENU_SHELL(window->server_menu),
  202. terminal->menu_item);
  203. }
  204. static void mssh_window_init(MSSHWindow* window)
  205. {
  206. GtkAccelGroup *accel_group = gtk_accel_group_new();
  207. GtkWidget *vbox = gtk_vbox_new(FALSE, 0);
  208. GtkWidget *entry = gtk_entry_new();
  209. GtkWidget *menu_bar = gtk_menu_bar_new();
  210. GtkWidget *file_menu = gtk_menu_new();
  211. GtkWidget *edit_menu = gtk_menu_new();
  212. GtkWidget *file_item = gtk_menu_item_new_with_label("File");
  213. GtkWidget *edit_item = gtk_menu_item_new_with_label("Edit");
  214. GtkWidget *server_item = gtk_menu_item_new_with_label("Servers");
  215. GtkWidget *file_quit = gtk_image_menu_item_new_from_stock(
  216. GTK_STOCK_QUIT, NULL);
  217. GtkWidget *file_sendhost = gtk_image_menu_item_new_with_label(
  218. "Send hostname");
  219. GtkWidget *file_add = gtk_image_menu_item_new_with_label(
  220. "Add session");
  221. GtkWidget *edit_pref = gtk_image_menu_item_new_from_stock(
  222. GTK_STOCK_PREFERENCES, NULL);
  223. window->server_menu = gtk_menu_new();
  224. gtk_menu_item_set_submenu(GTK_MENU_ITEM(file_item), file_menu);
  225. gtk_menu_item_set_submenu(GTK_MENU_ITEM(edit_item), edit_menu);
  226. gtk_menu_item_set_submenu(GTK_MENU_ITEM(server_item),
  227. window->server_menu);
  228. gtk_menu_shell_append(GTK_MENU_SHELL(file_menu), file_add);
  229. gtk_menu_shell_append(GTK_MENU_SHELL(file_menu), file_sendhost);
  230. gtk_menu_shell_append(GTK_MENU_SHELL(file_menu), file_quit);
  231. gtk_menu_shell_append(GTK_MENU_SHELL(edit_menu), edit_pref);
  232. g_signal_connect(G_OBJECT(file_sendhost), "activate",
  233. G_CALLBACK(mssh_window_sendhost), window);
  234. g_signal_connect(G_OBJECT(file_quit), "activate",
  235. G_CALLBACK(mssh_window_destroy), window);
  236. g_signal_connect(G_OBJECT(edit_pref), "activate",
  237. G_CALLBACK(mssh_window_pref), window);
  238. gtk_widget_add_accelerator(file_quit, "activate", accel_group,
  239. GDK_W, GDK_CONTROL_MASK, GTK_ACCEL_VISIBLE);
  240. gtk_window_add_accel_group(GTK_WINDOW(window), accel_group);
  241. gtk_menu_bar_append(GTK_MENU_BAR(menu_bar), file_item);
  242. gtk_menu_bar_append(GTK_MENU_BAR(menu_bar), edit_item);
  243. gtk_menu_bar_append(GTK_MENU_BAR(menu_bar), server_item);
  244. g_signal_connect(G_OBJECT(entry), "key-press-event",
  245. G_CALLBACK(mssh_window_key_press), window);
  246. g_signal_connect(G_OBJECT(entry), "insert-text",
  247. G_CALLBACK(mssh_window_insert), window);
  248. g_signal_connect(G_OBJECT(entry), "focus-in-event",
  249. G_CALLBACK(mssh_window_entry_focused), window);
  250. gtk_box_pack_start(GTK_BOX(vbox), menu_bar, FALSE, TRUE, 0);
  251. gtk_box_pack_start(GTK_BOX(vbox), entry, FALSE, TRUE, 2);
  252. window->table = gtk_table_new(1, 1, TRUE);
  253. gtk_box_pack_start(GTK_BOX(vbox), window->table, TRUE, TRUE, 0);
  254. gtk_container_add(GTK_CONTAINER(window), vbox);
  255. gtk_widget_set_size_request(GTK_WIDGET(window), 1024, 768);
  256. gtk_window_set_title(GTK_WINDOW(window), PACKAGE_NAME);
  257. }
  258. void mssh_window_start_session(MSSHWindow* window, char **env, int nhosts,
  259. char **servers)
  260. {
  261. int i, j, k;
  262. int rows = (nhosts / 2) + (nhosts % 2);
  263. window->env = env;
  264. window->terminals = g_array_new(FALSE, TRUE, sizeof(MSSHTerminal*));
  265. for(i = 0; i < rows; i++)
  266. {
  267. for(j = 0; j < 2; j++)
  268. {
  269. k = j + i*2;
  270. if(k < nhosts)
  271. {
  272. mssh_window_add_session(window, servers[k]);
  273. }
  274. }
  275. }
  276. mssh_window_relayout(window);
  277. }
  278. static void mssh_window_class_init(MSSHWindowClass *klass)
  279. {
  280. }