File:  [ELWIX - Embedded LightWeight unIX -] / embedaddon / mtr / ui / gtk.c
Revision 1.1.1.3 (vendor branch): download - view: text, annotated - select for diffs - revision graph
Wed Sep 27 11:18:58 2023 UTC (18 months, 1 week ago) by misho
Branches: mtr, MAIN
CVS tags: v0_95, HEAD
Version 0.95

    1: /*
    2:     mtr  --  a network diagnostic tool
    3:     Copyright (C) 1997,1998  Matt Kimball
    4:     Changes/additions Copyright (C) 1998 R.E.Wolff@BitWizard.nl
    5: 
    6:     This program is free software; you can redistribute it and/or modify
    7:     it under the terms of the GNU General Public License version 2 as
    8:     published by the Free Software Foundation.
    9: 
   10:     This program is distributed in the hope that it will be useful,
   11:     but WITHOUT ANY WARRANTY; without even the implied warranty of
   12:     MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
   13:     GNU General Public License for more details.
   14: 
   15:     You should have received a copy of the GNU General Public License along
   16:     with this program; if not, write to the Free Software Foundation, Inc.,
   17:     51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA.
   18: */
   19: 
   20: #include "config.h"
   21: 
   22: #include <stdio.h>
   23: #include <stdlib.h>
   24: #include <unistd.h>
   25: #include <sys/time.h>
   26: #include <sys/types.h>
   27: 
   28: #ifdef HAVE_GTK
   29: #include <string.h>
   30: #include <sys/types.h>
   31: #include <gtk/gtk.h>
   32: 
   33: #include "mtr.h"
   34: #include "net.h"
   35: #include "dns.h"
   36: #include "asn.h"
   37: #include "mtr-gtk.h"
   38: #include "utils.h"
   39: 
   40: #include "img/mtr_icon.xpm"
   41: #endif
   42: 
   43: static gint gtk_ping(
   44:     gpointer data);
   45: static gint Copy_activate(
   46:     GtkWidget * widget,
   47:     gpointer data);
   48: static gint NewDestination_activate(
   49:     GtkWidget * widget,
   50:     gpointer data);
   51: static gboolean ReportTreeView_clicked(
   52:     GtkWidget * Tree,
   53:     GdkEventButton * event,
   54:     gpointer data);
   55: static gchar *getSelectedHost(
   56:     GtkTreePath * path);
   57: 
   58: static int ping_timeout_timer;
   59: static GtkWidget *Pause_Button;
   60: static GtkWidget *Entry;
   61: static GtkWidget *main_window;
   62: 
   63: static void gtk_add_ping_timeout(
   64:     struct mtr_ctl *ctl)
   65: {
   66:     int dt;
   67: 
   68:     if (gtk_toggle_button_get_active((GtkToggleButton *) Pause_Button)) {
   69:         return;
   70:     }
   71:     dt = calc_deltatime(ctl->WaitTime);
   72:     gtk_redraw(ctl);
   73:     ping_timeout_timer = g_timeout_add(dt / 1000, gtk_ping, ctl);
   74: }
   75: 
   76: 
   77: static void gtk_do_init(
   78:     int *argc,
   79:     char ***argv)
   80: {
   81:     static int done = 0;
   82: 
   83:     if (!done) {
   84:         gtk_init(argc, argv);
   85: 
   86:         done = 1;
   87:     }
   88: }
   89: 
   90: 
   91: int gtk_detect(
   92:     ATTRIBUTE_UNUSED int *argc,
   93:     ATTRIBUTE_UNUSED char ***argv)
   94: {
   95:     if (getenv("DISPLAY") != NULL) {
   96:         /* If we do this here, gtk_init exits on an error. This happens
   97:            BEFORE the user has had a chance to tell us not to use the
   98:            display... */
   99:         return TRUE;
  100:     } else {
  101:         return FALSE;
  102:     }
  103: }
  104: 
  105: 
  106: static gint Window_destroy(
  107:     ATTRIBUTE_UNUSED GtkWidget * Window,
  108:     ATTRIBUTE_UNUSED gpointer data)
  109: {
  110:     gtk_main_quit();
  111: 
  112:     return FALSE;
  113: }
  114: 
  115: 
  116: static gint Restart_clicked(
  117:     ATTRIBUTE_UNUSED GtkWidget * Button,
  118:     gpointer data)
  119: {
  120:     struct mtr_ctl *ctl = (struct mtr_ctl *) data;
  121: 
  122:     net_reset(ctl);
  123:     gtk_redraw(ctl);
  124: 
  125:     return FALSE;
  126: }
  127: 
  128: 
  129: static gint Pause_clicked(
  130:     ATTRIBUTE_UNUSED GtkWidget * Button,
  131:     gpointer data)
  132: {
  133:     struct mtr_ctl *ctl = (struct mtr_ctl *) data;
  134: 
  135:     static int paused = 0;
  136: 
  137:     if (paused) {
  138:         gtk_add_ping_timeout(ctl);
  139:     } else {
  140:         g_source_remove(ping_timeout_timer);
  141:     }
  142:     paused = !paused;
  143:     gtk_redraw(ctl);
  144: 
  145:     return FALSE;
  146: }
  147: 
  148: static gint About_clicked(
  149:     ATTRIBUTE_UNUSED GtkWidget * Button,
  150:     ATTRIBUTE_UNUSED gpointer data)
  151: {
  152:     static const gchar *authors[] = {
  153:         "Matt Kimball <matt.kimball@gmail.com>",
  154:         "Roger Wolff <R.E.Wolff@BitWizard.nl>",
  155:         "Bohdan Vlasyuk <bohdan@cec.vstu.vinnica.ua>",
  156:         "Evgeniy Tretyak <evtr@ukr.net>",
  157:         "John Thacker <thacker@math.cornell.edu>",
  158:         "Juha Takala",
  159:         "David Sward <sward@clark.net>",
  160:         "David Stone <stone@AsIf.com>",
  161:         "Andrew Stesin",
  162:         "Greg Stark <gsstark@mit.edu>",
  163:         "Robert Sparks <rjsparks@nostrum.com>",
  164:         "Mike Simons <msimons@moria.simons-clan.com>",
  165:         "Aaron Scarisbrick,",
  166:         "Craig Milo Rogers <Rogers@ISI.EDU>",
  167:         "Antonio Querubin <tony@lavanauts.org>",
  168:         "Russell Nelson <rn-mtr@crynwr.com>",
  169:         "Davin Milun <milun@acm.org>",
  170:         "Josh Martin <jmartin@columbiaservices.net>",
  171:         "Alexander V. Lukyanov <lav@yars.free.net>",
  172:         "Charles Levert <charles@comm.polymtl.ca> ",
  173:         "Bertrand Leconte <B.Leconte@mail.dotcom.fr>",
  174:         "Anand Kumria",
  175:         "Olav Kvittem <Olav.Kvittem@uninett.no>",
  176:         "Adam Kramer <l3zqc@qcunix1.acc.qc.edu> ",
  177:         "Philip Kizer <pckizer@nostrum.com>",
  178:         "Simon Kirby",
  179:         "Sami Kerola <kerolasa@iki.fi>",
  180:         "Christophe Kalt",
  181:         "Steve Kann <stevek@spheara.horizonlive.com>",
  182:         "Brett Johnson <brett@jdacareers.com>",
  183:         "Roland Illig <roland.illig@gmx.de>",
  184:         "Damian Gryski <dgryski@uwaterloo.ca>",
  185:         "Rob Foehl <rwf@loonybin.net>",
  186:         "Mircea Damian",
  187:         "Cougar <cougar@random.ee>",
  188:         "Travis Cross <tc@traviscross.com>",
  189:         "Brian Casey",
  190:         "Andrew Brown <atatat@atatdot.net>",
  191:         "Bill Bogstad <bogstad@pobox.com> ",
  192:         "Marc Bejarano <marc.bejarano@openwave.com>",
  193:         "Moritz Barsnick <barsnick@gmx.net>",
  194:         "Thomas Klausner <wiz@NetBSD.org>",
  195:         NULL
  196:     };
  197: 
  198:     gtk_show_about_dialog(GTK_WINDOW(main_window)
  199:                           , "version", PACKAGE_VERSION, "copyright",
  200:                           "Copyright \xc2\xa9 1997,1998  Matt Kimball",
  201:                           "website", "https://www.bitwizard.nl/mtr/",
  202:                           "authors", authors, "comments",
  203:                           "The 'traceroute' and 'ping' programs in a single network diagnostic tool.",
  204:                           "license",
  205:                           "This program is free software; you can redistribute it and/or modify\n"
  206:                           "it under the terms of the GNU General Public License version 2 as\n"
  207:                           "published by the Free Software Foundation.\n"
  208:                           "\n"
  209:                           "This program is distributed in the hope that it will be useful,\n"
  210:                           "but WITHOUT ANY WARRANTY; without even the implied warranty of\n"
  211:                           "MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the\n"
  212:                           "GNU General Public License for more details.",
  213:                           NULL);
  214:     return TRUE;
  215: }
  216: 
  217: /*
  218:  * There is a small problem with the following code:
  219:  * The timeout is canceled and removed in order to ensure that
  220:  * it takes effect (consider what happens if you set the timeout to 999,
  221:  * then try to undo the change); is a better approach possible?
  222:  *
  223:  * What's the problem with this? (-> "I don't think so)
  224:  */
  225: 
  226: static gint WaitTime_changed(
  227:     ATTRIBUTE_UNUSED GtkAdjustment * Adj,
  228:     GtkWidget * data)
  229: {
  230:     struct mtr_ctl *ctl = (struct mtr_ctl *) data;
  231:     GtkWidget *Button = (GtkWidget *) ctl->gtk_data;
  232: 
  233:     ctl->WaitTime = gtk_spin_button_get_value(GTK_SPIN_BUTTON(Button));
  234:     g_source_remove(ping_timeout_timer);
  235:     gtk_add_ping_timeout(ctl);
  236:     gtk_redraw(ctl);
  237: 
  238:     return FALSE;
  239: }
  240: 
  241: 
  242: static gint Host_activate(
  243:     GtkWidget * entry,
  244:     gpointer data)
  245: {
  246:     struct mtr_ctl *ctl = (struct mtr_ctl *) data;
  247:     struct addrinfo *res = NULL;
  248: 
  249:     ctl->af = DEFAULT_AF;  // should this obey the cmd line option?
  250:     ctl->Hostname = gtk_entry_get_text(GTK_ENTRY(entry));
  251:     if (get_addrinfo_from_name(ctl, &res, ctl->Hostname) == 0) {
  252:         net_reopen(ctl, res);
  253:         freeaddrinfo(res);
  254:         net_send_batch(ctl);
  255:         /* If we are "Paused" at this point it is usually because someone
  256:            entered a non-existing host. Therefore do the go-ahead... */
  257:         gtk_toggle_button_set_active(GTK_TOGGLE_BUTTON(Pause_Button), 0);
  258:     } else {
  259:         int pos = strlen(gtk_entry_get_text(GTK_ENTRY(entry)));
  260:         gtk_toggle_button_set_active(GTK_TOGGLE_BUTTON(Pause_Button), 1);
  261:         gtk_editable_insert_text(GTK_EDITABLE(entry), ": not found", -1,
  262:                                  &pos);
  263:     }
  264: 
  265:     return FALSE;
  266: }
  267: 
  268: 
  269: 
  270: static void Toolbar_fill(
  271:     struct mtr_ctl *ctl,
  272:     GtkWidget * Toolbar)
  273: {
  274:     GtkWidget *Button;
  275:     GtkWidget *Label;
  276:     GtkAdjustment *Adjustment;
  277: 
  278:     Button = gtk_button_new_with_label("Quit");
  279:     gtk_box_pack_end(GTK_BOX(Toolbar), Button, FALSE, FALSE, 0);
  280:     g_signal_connect(G_OBJECT(Button), "clicked",
  281:                      G_CALLBACK(Window_destroy), NULL);
  282: 
  283:     Button = gtk_button_new_with_label("About");
  284:     gtk_box_pack_end(GTK_BOX(Toolbar), Button, FALSE, FALSE, 0);
  285:     g_signal_connect(G_OBJECT(Button), "clicked",
  286:                      G_CALLBACK(About_clicked), NULL);
  287: 
  288:     Button = gtk_button_new_with_mnemonic("_Restart");
  289:     gtk_box_pack_end(GTK_BOX(Toolbar), Button, FALSE, FALSE, 0);
  290:     g_signal_connect(G_OBJECT(Button), "clicked",
  291:                      G_CALLBACK(Restart_clicked), ctl);
  292: 
  293:     Pause_Button = gtk_toggle_button_new_with_mnemonic("_Pause");
  294:     gtk_box_pack_end(GTK_BOX(Toolbar), Pause_Button, FALSE, FALSE, 0);
  295:     g_signal_connect(G_OBJECT(Pause_Button), "clicked",
  296:                      G_CALLBACK(Pause_clicked), ctl);
  297: 
  298:     /* allow root only to set zero delay */
  299:     Adjustment = (GtkAdjustment *) gtk_adjustment_new(ctl->WaitTime,
  300:                                                       running_as_root() ? 0.01 : 1.00,
  301:                                                       999.99, 1.0, 10.0,
  302:                                                       0.0);
  303:     Button = gtk_spin_button_new(Adjustment, 0.5, 2);
  304:     gtk_spin_button_set_numeric(GTK_SPIN_BUTTON(Button), TRUE);
  305:     gtk_box_pack_end(GTK_BOX(Toolbar), Button, FALSE, FALSE, 0);
  306:     ctl->gtk_data = Button;
  307:     g_signal_connect(G_OBJECT(Adjustment), "value_changed",
  308:                      G_CALLBACK(WaitTime_changed), ctl);
  309: 
  310:     Label = gtk_label_new_with_mnemonic("_Hostname:");
  311:     gtk_box_pack_start(GTK_BOX(Toolbar), Label, FALSE, FALSE, 0);
  312: 
  313:     Entry = gtk_entry_new();
  314:     gtk_entry_set_text(GTK_ENTRY(Entry), ctl->Hostname);
  315:     g_signal_connect(G_OBJECT(Entry), "activate",
  316:                      G_CALLBACK(Host_activate), ctl);
  317:     gtk_box_pack_start(GTK_BOX(Toolbar), Entry, TRUE, TRUE, 0);
  318: 
  319:     gtk_label_set_mnemonic_widget(GTK_LABEL(Label), Entry);
  320: }
  321: 
  322: static GtkWidget *ReportTreeView;
  323: static GtkListStore *ReportStore;
  324: 
  325: enum {
  326: #ifdef HAVE_IPINFO
  327:     COL_ASN,
  328: #endif
  329:     COL_HOSTNAME,
  330:     COL_LOSS,
  331:     COL_RCV,
  332:     COL_SNT,
  333:     COL_LAST,
  334:     COL_BEST,
  335:     COL_AVG,
  336:     COL_WORST,
  337:     COL_STDEV,
  338:     COL_COLOR,
  339:     N_COLS
  340: };
  341: 
  342: /* Trick to cast a pointer to integer.  We are mis-using a pointer as a
  343:    single integer.  On 64-bit architectures, the pointer is 64 bits and the
  344:    integer only 32.  The compiler warns us of loss of precision.  However we
  345:    know we casted a normal 32-bit integer into this pointer a few
  346:    microseconds earlier, so it is ok.  Nothing to worry about.  */
  347: #define POINTER_TO_INT(p) ((int)(long)(p))
  348: 
  349: static void float_formatter(
  350:     GtkTreeViewColumn * tree_column ATTRIBUTE_UNUSED,
  351:     GtkCellRenderer * cell,
  352:     GtkTreeModel * tree_model,
  353:     GtkTreeIter * iter,
  354:     gpointer data)
  355: {
  356:     gfloat f;
  357:     gchar text[64];
  358:     gtk_tree_model_get(tree_model, iter, POINTER_TO_INT(data), &f, -1);
  359:     sprintf(text, "%.2f", f);
  360:     g_object_set(cell, "text", text, NULL);
  361: }
  362: 
  363: static void percent_formatter(
  364:     GtkTreeViewColumn * tree_column ATTRIBUTE_UNUSED,
  365:     GtkCellRenderer * cell,
  366:     GtkTreeModel * tree_model,
  367:     GtkTreeIter * iter,
  368:     gpointer data)
  369: {
  370:     gfloat f;
  371:     gchar text[64];
  372:     gtk_tree_model_get(tree_model, iter, POINTER_TO_INT(data), &f, -1);
  373:     sprintf(text, "%.1f%%", f);
  374:     g_object_set(cell, "text", text, NULL);
  375: }
  376: 
  377: static void TreeViewCreate(
  378:     struct mtr_ctl *ctl)
  379: {
  380:     GtkCellRenderer *renderer;
  381:     GtkTreeViewColumn *column;
  382: 
  383:     ReportStore = gtk_list_store_new(N_COLS,
  384: #ifdef HAVE_IPINFO
  385:                                      G_TYPE_STRING,
  386: #endif
  387:                                      G_TYPE_STRING,
  388:                                      G_TYPE_FLOAT,
  389:                                      G_TYPE_INT,
  390:                                      G_TYPE_INT,
  391:                                      G_TYPE_INT,
  392:                                      G_TYPE_INT,
  393:                                      G_TYPE_INT,
  394:                                      G_TYPE_INT,
  395:                                      G_TYPE_FLOAT, G_TYPE_STRING);
  396: 
  397:     ReportTreeView =
  398:         gtk_tree_view_new_with_model(GTK_TREE_MODEL(ReportStore));
  399: 
  400:     g_signal_connect(G_OBJECT(ReportTreeView), "button_press_event",
  401:                      G_CALLBACK(ReportTreeView_clicked), ctl);
  402: 
  403: #ifdef HAVE_IPINFO
  404:     if (is_printii(ctl)) {
  405:         renderer = gtk_cell_renderer_text_new();
  406:         column = gtk_tree_view_column_new_with_attributes("ASN",
  407:                                                           renderer,
  408:                                                           "text", COL_ASN,
  409:                                                           "foreground",
  410:                                                           COL_COLOR, NULL);
  411:         gtk_tree_view_column_set_resizable(column, TRUE);
  412:         gtk_tree_view_append_column(GTK_TREE_VIEW(ReportTreeView), column);
  413:     }
  414: #endif
  415: 
  416:     renderer = gtk_cell_renderer_text_new();
  417:     column = gtk_tree_view_column_new_with_attributes("Hostname",
  418:                                                       renderer,
  419:                                                       "text", COL_HOSTNAME,
  420:                                                       "foreground",
  421:                                                       COL_COLOR, NULL);
  422:     gtk_tree_view_column_set_expand(column, TRUE);
  423:     gtk_tree_view_column_set_resizable(column, TRUE);
  424:     gtk_tree_view_append_column(GTK_TREE_VIEW(ReportTreeView), column);
  425: 
  426:     renderer = gtk_cell_renderer_text_new();
  427:     g_object_set(G_OBJECT(renderer), "xalign", 1.0, NULL);
  428:     column = gtk_tree_view_column_new_with_attributes("Loss",
  429:                                                       renderer,
  430:                                                       "text", COL_LOSS,
  431:                                                       "foreground",
  432:                                                       COL_COLOR, NULL);
  433:     gtk_tree_view_column_set_resizable(column, TRUE);
  434:     gtk_tree_view_column_set_cell_data_func(column, renderer,
  435:                                             percent_formatter,
  436:                                             (void *) COL_LOSS, NULL);
  437:     gtk_tree_view_append_column(GTK_TREE_VIEW(ReportTreeView), column);
  438: 
  439:     renderer = gtk_cell_renderer_text_new();
  440:     g_object_set(G_OBJECT(renderer), "xalign", 1.0, NULL);
  441:     column = gtk_tree_view_column_new_with_attributes("Snt",
  442:                                                       renderer,
  443:                                                       "text", COL_SNT,
  444:                                                       "foreground",
  445:                                                       COL_COLOR, NULL);
  446:     gtk_tree_view_column_set_resizable(column, TRUE);
  447:     gtk_tree_view_append_column(GTK_TREE_VIEW(ReportTreeView), column);
  448: 
  449:     renderer = gtk_cell_renderer_text_new();
  450:     g_object_set(G_OBJECT(renderer), "xalign", 1.0, NULL);
  451:     column = gtk_tree_view_column_new_with_attributes("Last",
  452:                                                       renderer,
  453:                                                       "text", COL_LAST,
  454:                                                       "foreground",
  455:                                                       COL_COLOR, NULL);
  456:     gtk_tree_view_column_set_resizable(column, TRUE);
  457:     gtk_tree_view_append_column(GTK_TREE_VIEW(ReportTreeView), column);
  458: 
  459:     renderer = gtk_cell_renderer_text_new();
  460:     g_object_set(G_OBJECT(renderer), "xalign", 1.0, NULL);
  461:     column = gtk_tree_view_column_new_with_attributes("Avg",
  462:                                                       renderer,
  463:                                                       "text", COL_AVG,
  464:                                                       "foreground",
  465:                                                       COL_COLOR, NULL);
  466:     gtk_tree_view_column_set_resizable(column, TRUE);
  467:     gtk_tree_view_append_column(GTK_TREE_VIEW(ReportTreeView), column);
  468: 
  469:     renderer = gtk_cell_renderer_text_new();
  470:     g_object_set(G_OBJECT(renderer), "xalign", 1.0, NULL);
  471:     column = gtk_tree_view_column_new_with_attributes("Best",
  472:                                                       renderer,
  473:                                                       "text", COL_BEST,
  474:                                                       "foreground",
  475:                                                       COL_COLOR, NULL);
  476:     gtk_tree_view_column_set_resizable(column, TRUE);
  477:     gtk_tree_view_append_column(GTK_TREE_VIEW(ReportTreeView), column);
  478: 
  479:     renderer = gtk_cell_renderer_text_new();
  480:     g_object_set(G_OBJECT(renderer), "xalign", 1.0, NULL);
  481:     column = gtk_tree_view_column_new_with_attributes("Worst",
  482:                                                       renderer,
  483:                                                       "text", COL_WORST,
  484:                                                       "foreground",
  485:                                                       COL_COLOR, NULL);
  486:     gtk_tree_view_column_set_resizable(column, TRUE);
  487:     gtk_tree_view_append_column(GTK_TREE_VIEW(ReportTreeView), column);
  488: 
  489:     renderer = gtk_cell_renderer_text_new();
  490:     g_object_set(G_OBJECT(renderer), "xalign", 1.0, NULL);
  491:     column = gtk_tree_view_column_new_with_attributes("StDev",
  492:                                                       renderer,
  493:                                                       "text", COL_STDEV,
  494:                                                       "foreground",
  495:                                                       COL_COLOR, NULL);
  496:     gtk_tree_view_column_set_resizable(column, TRUE);
  497:     gtk_tree_view_column_set_cell_data_func(column, renderer,
  498:                                             float_formatter,
  499:                                             (void *) COL_STDEV, NULL);
  500:     gtk_tree_view_append_column(GTK_TREE_VIEW(ReportTreeView), column);
  501: 
  502: }
  503: 
  504: static void update_tree_row(
  505:     struct mtr_ctl *ctl,
  506:     int row,
  507:     GtkTreeIter * iter)
  508: {
  509:     ip_t *addr;
  510:     char str[256] = "???", *name = str;
  511: 
  512:     addr = net_addr(row);
  513:     if (addrcmp(addr, &ctl->unspec_addr, ctl->af)) {
  514:         if ((name = dns_lookup(ctl, addr))) {
  515:             if (ctl->show_ips) {
  516:                 snprintf(str, sizeof(str), "%s (%s)", name,
  517:                          strlongip(ctl->af, addr));
  518:                 name = str;
  519:             }
  520:         } else
  521:             name = strlongip(ctl->af, addr);
  522:     }
  523: 
  524:     gtk_list_store_set(ReportStore, iter,
  525:                        COL_HOSTNAME, name,
  526:                        COL_LOSS, (float) (net_loss(row) / 1000.0),
  527:                        COL_RCV, net_returned(row),
  528:                        COL_SNT, net_xmit(row),
  529:                        COL_LAST, net_last(row) / 1000,
  530:                        COL_BEST, net_best(row) / 1000,
  531:                        COL_AVG, net_avg(row) / 1000,
  532:                        COL_WORST, net_worst(row) / 1000,
  533:                        COL_STDEV, (float) (net_stdev(row) / 1000.0),
  534:                        COL_COLOR, net_up(row) ? NULL : "red", -1);
  535: #ifdef HAVE_IPINFO
  536:     if (is_printii(ctl))
  537:         gtk_list_store_set(ReportStore, iter, COL_ASN,
  538:                            fmt_ipinfo(ctl, addr), -1);
  539: #endif
  540: }
  541: 
  542: void gtk_redraw(
  543:     struct mtr_ctl *ctl)
  544: {
  545:     int max = net_max(ctl);
  546: 
  547:     GtkTreeIter iter;
  548:     int row = net_min(ctl);
  549:     gboolean valid;
  550: 
  551:     valid =
  552:         gtk_tree_model_get_iter_first(GTK_TREE_MODEL(ReportStore), &iter);
  553: 
  554:     while (valid) {
  555:         if (row < max) {
  556:             update_tree_row(ctl, row++, &iter);
  557:             valid =
  558:                 gtk_tree_model_iter_next(GTK_TREE_MODEL(ReportStore),
  559:                                          &iter);
  560:         } else {
  561:             valid = gtk_list_store_remove(ReportStore, &iter);
  562:         }
  563:     }
  564:     while (row < max) {
  565:         gtk_list_store_append(ReportStore, &iter);
  566:         update_tree_row(ctl, row++, &iter);
  567:     }
  568: }
  569: 
  570: // GTK 3 has changed the interface a bit. Here a few defines so that we can
  571: // work with GTK2 or GTK3 as required.
  572: #ifdef HAVE_GTK3
  573: #define gtk_vbox_new_(orientation,sz) gtk_box_new(orientation, sz)
  574: #define gtk_hbox_new_(orientation,sz) gtk_box_new(orientation, sz)
  575: #else
  576: #define gtk_vbox_new_(orientation,sz) gtk_vbox_new(FALSE, sz)
  577: #define gtk_hbox_new_(orientation,sz) gtk_hbox_new(FALSE, sz)
  578: #endif
  579: 
  580: static void Window_fill(
  581:     struct mtr_ctl *ctl,
  582:     GtkWidget * Window)
  583: {
  584:     GtkWidget *VBox;
  585:     GtkWidget *Toolbar;
  586:     GtkWidget *scroll;
  587: 
  588:     gtk_window_set_title(GTK_WINDOW(Window), "My traceroute");
  589:     gtk_window_set_default_size(GTK_WINDOW(Window), 650, 400);
  590:     gtk_container_set_border_width(GTK_CONTAINER(Window), 10);
  591: 
  592:     VBox    = gtk_vbox_new_(GTK_ORIENTATION_VERTICAL,   10);
  593:     Toolbar = gtk_hbox_new_(GTK_ORIENTATION_HORIZONTAL, 10);
  594: 
  595:     Toolbar_fill(ctl, Toolbar);
  596:     gtk_box_pack_start(GTK_BOX(VBox), Toolbar, FALSE, FALSE, 0);
  597: 
  598:     TreeViewCreate(ctl);
  599:     scroll = gtk_scrolled_window_new(NULL, NULL);
  600:     gtk_scrolled_window_set_policy(GTK_SCROLLED_WINDOW(scroll),
  601:                                    GTK_POLICY_AUTOMATIC,
  602:                                    GTK_POLICY_AUTOMATIC);
  603:     gtk_scrolled_window_set_shadow_type(GTK_SCROLLED_WINDOW(scroll),
  604:                                         GTK_SHADOW_IN);
  605:     gtk_container_add(GTK_CONTAINER(scroll), ReportTreeView);
  606:     gtk_box_pack_start(GTK_BOX(VBox), scroll, TRUE, TRUE, 0);
  607: 
  608:     gtk_container_add(GTK_CONTAINER(Window), VBox);
  609: }
  610: 
  611: 
  612: void gtk_open(
  613:     struct mtr_ctl *ctl)
  614: {
  615:     GdkPixbuf *icon;
  616:     int argc = 1;
  617:     char *args[2];
  618:     char **argv;
  619: 
  620:     argv = args;
  621:     argv[0] = xstrdup("");
  622:     argv[1] = NULL;
  623:     gtk_do_init(&argc, &argv);
  624:     free(argv[0]);
  625: 
  626:     icon = gdk_pixbuf_new_from_xpm_data((const char **) mtr_icon);
  627:     gtk_window_set_default_icon(icon);
  628: 
  629:     main_window = gtk_window_new(GTK_WINDOW_TOPLEVEL);
  630: 
  631:     g_set_application_name("My traceroute");
  632: 
  633:     Window_fill(ctl, main_window);
  634: 
  635:     g_signal_connect(G_OBJECT(main_window), "delete_event",
  636:                      G_CALLBACK(Window_destroy), NULL);
  637:     g_signal_connect(G_OBJECT(main_window), "destroy",
  638:                      G_CALLBACK(Window_destroy), NULL);
  639: 
  640:     gtk_widget_show_all(main_window);
  641: }
  642: 
  643: 
  644: void gtk_close(
  645:     void)
  646: {
  647: }
  648: 
  649: 
  650: int gtk_keyaction(
  651:     void)
  652: {
  653:     return 0;
  654: }
  655: 
  656: 
  657: static gint gtk_ping(
  658:     gpointer data)
  659: {
  660:     struct mtr_ctl *ctl = (struct mtr_ctl *) data;
  661: 
  662:     gtk_redraw(ctl);
  663:     net_send_batch(ctl);
  664:     net_harvest_fds(ctl);
  665:     g_source_remove(ping_timeout_timer);
  666:     gtk_add_ping_timeout(ctl);
  667:     return TRUE;
  668: }
  669: 
  670: 
  671: static gboolean gtk_net_data(
  672:     ATTRIBUTE_UNUSED GIOChannel * channel,
  673:     ATTRIBUTE_UNUSED GIOCondition cond,
  674:     gpointer data)
  675: {
  676:     struct mtr_ctl *ctl = (struct mtr_ctl *) data;
  677: 
  678:     net_process_return(ctl);
  679:     return TRUE;
  680: }
  681: 
  682: 
  683: static gboolean gtk_dns_data(
  684:     ATTRIBUTE_UNUSED GIOChannel * channel,
  685:     ATTRIBUTE_UNUSED GIOCondition cond,
  686:     gpointer data)
  687: {
  688:     struct mtr_ctl *ctl = (struct mtr_ctl *) data;
  689: 
  690:     dns_ack(ctl);
  691:     gtk_redraw(ctl);
  692:     return TRUE;
  693: }
  694: 
  695: #ifdef ENABLE_IPV6
  696: static gboolean gtk_dns_data6(
  697:     ATTRIBUTE_UNUSED GIOChannel * channel,
  698:     ATTRIBUTE_UNUSED GIOCondition cond,
  699:     gpointer data)
  700: {
  701:     struct mtr_ctl *ctl = (struct mtr_ctl *) data;
  702: 
  703:     dns_ack6();
  704:     gtk_redraw(ctl);
  705:     return TRUE;
  706: }
  707: #endif
  708: 
  709: 
  710: void gtk_loop(
  711:     struct mtr_ctl *ctl)
  712: {
  713:     GIOChannel *net_iochannel, *dns_iochannel;
  714: 
  715:     gtk_add_ping_timeout(ctl);
  716: 
  717:     net_iochannel = g_io_channel_unix_new(net_waitfd());
  718:     g_io_add_watch(net_iochannel, G_IO_IN, gtk_net_data, ctl);
  719: #ifdef ENABLE_IPV6
  720:     if (dns_waitfd6() > 0) {
  721:         dns_iochannel = g_io_channel_unix_new(dns_waitfd6());
  722:         g_io_add_watch(dns_iochannel, G_IO_IN, gtk_dns_data6, ctl);
  723:     }
  724: #endif
  725:     dns_iochannel = g_io_channel_unix_new(dns_waitfd());
  726:     g_io_add_watch(dns_iochannel, G_IO_IN, gtk_dns_data, ctl);
  727: 
  728:     gtk_main();
  729: }
  730: 
  731: static gboolean NewDestination_activate(
  732:     GtkWidget * widget ATTRIBUTE_UNUSED,
  733:     gpointer data)
  734: {
  735:     gchar *hostname;
  736:     struct mtr_ctl *ctl = (struct mtr_ctl *) data;
  737:     GtkTreePath *path = (GtkTreePath *) ctl->gtk_data;
  738: 
  739:     hostname = getSelectedHost(path);
  740:     if (hostname) {
  741:         ctl->gtk_data = hostname;
  742:         gtk_entry_set_text(GTK_ENTRY(Entry), hostname);
  743:         Host_activate(Entry, ctl);
  744:         g_free(hostname);
  745:     }
  746:     return TRUE;
  747: }
  748: 
  749: 
  750: static gboolean Copy_activate(
  751:     GtkWidget * widget ATTRIBUTE_UNUSED,
  752:     gpointer data)
  753: {
  754:     gchar *hostname;
  755:     GtkTreePath *path = (GtkTreePath *) data;
  756: 
  757:     hostname = getSelectedHost(path);
  758:     if (hostname != NULL) {
  759:         GtkClipboard *clipboard;
  760: 
  761:         clipboard = gtk_clipboard_get(GDK_SELECTION_CLIPBOARD);
  762:         gtk_clipboard_set_text(clipboard, hostname, -1);
  763: 
  764:         clipboard = gtk_clipboard_get(GDK_SELECTION_PRIMARY);
  765:         gtk_clipboard_set_text(clipboard, hostname, -1);
  766: 
  767:         g_free(hostname);
  768:     }
  769: 
  770:     return TRUE;
  771: }
  772: 
  773: static gchar *getSelectedHost(
  774:     GtkTreePath * path)
  775: {
  776:     GtkTreeIter iter;
  777:     gchar *name = NULL;
  778: 
  779:     if (gtk_tree_model_get_iter(GTK_TREE_MODEL(ReportStore), &iter, path)) {
  780:         gtk_tree_model_get(GTK_TREE_MODEL(ReportStore), &iter,
  781:                            COL_HOSTNAME, &name, -1);
  782:     }
  783:     gtk_tree_path_free(path);
  784:     return name;
  785: }
  786: 
  787: 
  788: static gboolean ReportTreeView_clicked(
  789:     GtkWidget * Tree ATTRIBUTE_UNUSED,
  790:     GdkEventButton * event,
  791:     gpointer data)
  792: {
  793:     GtkWidget *popup_menu;
  794:     GtkWidget *copy_item;
  795:     GtkWidget *newdestination_item;
  796:     GtkTreePath *path;
  797:     struct mtr_ctl *ctl = (struct mtr_ctl *) data;
  798: 
  799:     if (event->type != GDK_BUTTON_PRESS || event->button != 3)
  800:         return FALSE;
  801: 
  802:     if (!gtk_tree_view_get_path_at_pos(GTK_TREE_VIEW(ReportTreeView),
  803:                                        event->x, event->y, &path, NULL,
  804:                                        NULL, NULL))
  805:         return FALSE;
  806: 
  807:     gtk_tree_view_set_cursor(GTK_TREE_VIEW(ReportTreeView), path, NULL,
  808:                              FALSE);
  809: 
  810:     /* Single right click: prepare and show the popup menu */
  811:     popup_menu = gtk_menu_new();
  812: 
  813:     copy_item = gtk_menu_item_new_with_label("Copy to clipboard");
  814:     newdestination_item =
  815:         gtk_menu_item_new_with_label("Set as new destination");
  816: 
  817:     gtk_menu_shell_append(GTK_MENU_SHELL(popup_menu), copy_item);
  818:     gtk_menu_shell_append(GTK_MENU_SHELL(popup_menu), newdestination_item);
  819: 
  820:     g_signal_connect(G_OBJECT(copy_item), "activate",
  821:                      G_CALLBACK(Copy_activate), path);
  822: 
  823:     ctl->gtk_data = path;
  824:     g_signal_connect(G_OBJECT(newdestination_item), "activate",
  825:                      G_CALLBACK(NewDestination_activate), ctl);
  826: 
  827:     gtk_widget_show(copy_item);
  828:     gtk_widget_show(newdestination_item);
  829: 
  830: #ifdef HAVE_GTK3
  831:     gtk_menu_popup_at_pointer(GTK_MENU(popup_menu), NULL);
  832: #else
  833:     gtk_menu_popup(GTK_MENU(popup_menu), NULL, NULL, NULL, NULL,
  834:                    0, event->time);
  835: #endif
  836:     return TRUE;
  837: }

FreeBSD-CVSweb <freebsd-cvsweb@FreeBSD.org>