PageRenderTime 65ms CodeModel.GetById 25ms RepoModel.GetById 1ms app.codeStats 0ms

/ui/gtk/graph_analysis.c

https://github.com/labx-technologies-llc/wireshark
C | 2164 lines | 1657 code | 294 blank | 213 comment | 218 complexity | 523ba8559350c0763a72cdf697e01f31 MD5 | raw file
Possible License(s): GPL-2.0, BSD-3-Clause

Large files files are truncated, but you can click here to view the full file

  1. /* graph_analysis.c
  2. * Graphic Analysis addition for Wireshark
  3. *
  4. * $Id$
  5. *
  6. * Copyright 2004, Verso Technologies Inc.
  7. * By Alejandro Vaquero <alejandrovaquero@yahoo.com>
  8. *
  9. * based on rtp_analysis.c and io_stat
  10. *
  11. *
  12. * Wireshark - Network traffic analyzer
  13. * By Gerald Combs <gerald@wireshark.org>
  14. * Copyright 1998 Gerald Combs
  15. *
  16. * This program is free software; you can redistribute it and/or
  17. * modify it under the terms of the GNU General Public License
  18. * as published by the Free Software Foundation; either version 2
  19. * of the License, or (at your option) any later version.
  20. *
  21. * This program is distributed in the hope that it will be useful,
  22. * but WITHOUT ANY WARRANTY; without even the implied warranty of
  23. * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
  24. * GNU General Public License for more details.
  25. *
  26. * You should have received a copy of the GNU General Public License
  27. * along with this program; if not, write to the Free Software
  28. * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA.
  29. */
  30. #include "config.h"
  31. #include <stdio.h>
  32. #include <string.h>
  33. #include <math.h>
  34. #ifdef HAVE_UNISTD_H
  35. #include <unistd.h>
  36. #endif
  37. #include <gtk/gtk.h>
  38. #include <gdk/gdkkeysyms.h>
  39. #if GTK_CHECK_VERSION(3,0,0)
  40. # include <gdk/gdkkeysyms-compat.h>
  41. #endif
  42. #include <epan/epan_dissect.h>
  43. #include <epan/tap.h>
  44. #include <epan/dissectors/packet-rtp.h>
  45. #include <epan/addr_resolv.h>
  46. #include <epan/filesystem.h>
  47. #include "ui/util.h"
  48. #include "ui/alert_box.h"
  49. #include "ui/last_open_dir.h"
  50. #include "ui/recent.h"
  51. #include "ui/simple_dialog.h"
  52. #include <wsutil/file_util.h>
  53. #include "ui/gtk/gtkglobals.h"
  54. #include "ui/gtk/file_dlg.h"
  55. #include "ui/gtk/gui_utils.h"
  56. #include "ui/gtk/dlg_utils.h"
  57. #include "ui/gtk/main.h"
  58. #include "ui/gtk/graph_analysis.h"
  59. #include "ui/gtk/old-gtk-compat.h"
  60. #include "../../image/voip_bg.xpm"
  61. /****************************************************************************/
  62. #define MAX_LABEL 50
  63. #define MAX_COMMENT 100
  64. #define ITEM_HEIGHT 20
  65. #define NODE_WIDTH 100
  66. #define TOP_Y_BORDER 40
  67. #define BOTTOM_Y_BORDER 2
  68. #define COMMENT_WIDTH 400
  69. #define TIME_WIDTH 150
  70. #define NODE_CHARS_WIDTH 20
  71. #define CONV_TIME_HEADER "Conv.| Time "
  72. #define TIME_HEADER "|Time "
  73. #define CONV_TIME_EMPTY_HEADER " | "
  74. #define TIME_EMPTY_HEADER "| "
  75. #define CONV_TIME_HEADER_LENGTH 16
  76. #define TIME_HEADER_LENGTH 10
  77. /****************************************************************************/
  78. /* Reset the user_data structure */
  79. static void graph_analysis_reset(graph_analysis_data_t *user_data)
  80. {
  81. int i;
  82. user_data->num_nodes = 0;
  83. user_data->num_items = 0;
  84. for (i=0; i<MAX_NUM_NODES; i++) {
  85. user_data->nodes[i].type = AT_NONE;
  86. user_data->nodes[i].len = 0;
  87. g_free((void *)user_data->nodes[i].data);
  88. user_data->nodes[i].data = NULL;
  89. }
  90. user_data->dlg.first_node = 0;
  91. user_data->dlg.first_item = 0;
  92. user_data->dlg.left_x_border = 0;
  93. user_data->dlg.selected_item = 0xFFFFFFFF; /*not item selected */
  94. }
  95. /****************************************************************************/
  96. /* Init the user_data structure */
  97. static void graph_analysis_init_dlg(graph_analysis_data_t *user_data)
  98. {
  99. int i;
  100. user_data->num_nodes = 0;
  101. user_data->num_items = 0;
  102. user_data->on_destroy_user_data = NULL;
  103. user_data->data = NULL;
  104. for (i=0; i<MAX_NUM_NODES; i++) {
  105. user_data->nodes[i].type = AT_NONE;
  106. user_data->nodes[i].len = 0;
  107. user_data->nodes[i].data = NULL;
  108. }
  109. user_data->dlg.first_node = 0;
  110. user_data->dlg.first_item = 0;
  111. user_data->dlg.left_x_border = 0;
  112. user_data->dlg.selected_item = 0xFFFFFFFF; /*not item selected */
  113. /* init dialog_graph */
  114. user_data->dlg.needs_redraw = TRUE;
  115. user_data->dlg.draw_area_time = NULL;
  116. user_data->dlg.draw_area = NULL;
  117. user_data->dlg.draw_area_comments = NULL;
  118. #if GTK_CHECK_VERSION(2,22,0)
  119. user_data->dlg.surface_main = NULL;
  120. user_data->dlg.surface_time = NULL;
  121. user_data->dlg.surface_comments = NULL;
  122. #else
  123. user_data->dlg.pixmap_main = NULL;
  124. user_data->dlg.pixmap_time = NULL;
  125. user_data->dlg.pixmap_comments = NULL;
  126. #endif
  127. user_data->dlg.v_scrollbar = NULL;
  128. user_data->dlg.v_scrollbar_adjustment = NULL;
  129. user_data->dlg.hpane = NULL;
  130. user_data->dlg.surface_width = 350;
  131. user_data->dlg.surface_height = 400;
  132. user_data->dlg.first_node = 0;
  133. user_data->dlg.first_item = 0;
  134. user_data->dlg.left_x_border = 0;
  135. user_data->dlg.selected_item = 0xFFFFFFFF; /*not item selected */
  136. user_data->dlg.window = NULL;
  137. user_data->dlg.parent_w = NULL;
  138. user_data->dlg.inverse = FALSE;
  139. user_data->dlg.title = NULL;
  140. }
  141. /****************************************************************************/
  142. /* CALLBACKS */
  143. /****************************************************************************/
  144. /* close the dialog window */
  145. static void on_destroy(GtkWidget *win _U_, graph_analysis_data_t *user_data)
  146. {
  147. int i;
  148. for (i=0; i<MAX_NUM_NODES; i++) {
  149. user_data->nodes[i].type = AT_NONE;
  150. user_data->nodes[i].len = 0;
  151. g_free((void *)user_data->nodes[i].data);
  152. user_data->nodes[i].data = NULL;
  153. }
  154. user_data->dlg.window = NULL;
  155. g_free(user_data->dlg.title);
  156. user_data->dlg.title = NULL;
  157. if(user_data->on_destroy_user_data) {
  158. user_data->on_destroy_user_data(user_data->data);
  159. }
  160. }
  161. #define RIGHT_ARROW 1
  162. #define LEFT_ARROW 0
  163. #define WIDTH_ARROW 8
  164. #define HEIGHT_ARROW 6
  165. /****************************************************************************/
  166. #if GTK_CHECK_VERSION(2,22,0)
  167. static void draw_arrow(cairo_surface_t *surface, GdkRGBA *color, gint x, gint y, gboolean arrow_type)
  168. {
  169. cairo_t *cr;
  170. cr = cairo_create (surface);
  171. gdk_cairo_set_source_rgba (cr, color);
  172. if (arrow_type == LEFT_ARROW)
  173. {
  174. cairo_move_to (cr, x + WIDTH_ARROW, y);
  175. cairo_line_to (cr, x + WIDTH_ARROW, y + HEIGHT_ARROW);
  176. cairo_line_to (cr, x, y + HEIGHT_ARROW / 2.);
  177. }
  178. else if (arrow_type == RIGHT_ARROW)
  179. {
  180. cairo_move_to (cr, x, y);
  181. cairo_line_to (cr, x + WIDTH_ARROW, y + HEIGHT_ARROW / 2.);
  182. cairo_line_to (cr, x, y + HEIGHT_ARROW);
  183. }
  184. cairo_close_path (cr);
  185. cairo_fill (cr);
  186. cairo_destroy (cr);
  187. }
  188. #else
  189. static void draw_arrow(GdkDrawable *pixmap, GdkRGBA *color, gint x, gint y, gboolean arrow_type)
  190. {
  191. cairo_t *cr;
  192. if (GDK_IS_DRAWABLE(pixmap)) {
  193. cr = gdk_cairo_create (pixmap);
  194. gdk_cairo_set_source_rgba (cr, color);
  195. if (arrow_type == LEFT_ARROW)
  196. {
  197. cairo_move_to (cr, x + WIDTH_ARROW, y);
  198. cairo_line_to (cr, x + WIDTH_ARROW, y + HEIGHT_ARROW);
  199. cairo_line_to (cr, x, y + HEIGHT_ARROW / 2.);
  200. }
  201. else if (arrow_type == RIGHT_ARROW)
  202. {
  203. cairo_move_to (cr, x, y);
  204. cairo_line_to (cr, x + WIDTH_ARROW, y + HEIGHT_ARROW / 2.);
  205. cairo_line_to (cr, x, y + HEIGHT_ARROW);
  206. }
  207. cairo_close_path (cr);
  208. cairo_fill (cr);
  209. cairo_destroy (cr);
  210. }
  211. }
  212. #endif
  213. /****************************************************************************/
  214. /* Adds trailing characters to complete the requested length. */
  215. /****************************************************************************/
  216. static void enlarge_string(GString *gstr, guint32 length, char pad) {
  217. gsize i;
  218. for (i = gstr->len; i < length; i++) {
  219. g_string_append_c(gstr, pad);
  220. }
  221. }
  222. /****************************************************************************/
  223. /* overwrites the characters in a string, between positions p1 and p2, with */
  224. /* the characters of text_to_insert */
  225. /* NB: it does not check that p1 and p2 fit into string */
  226. /****************************************************************************/
  227. static void overwrite (GString *gstr, char *text_to_insert, guint32 p1, guint32 p2) {
  228. gsize len;
  229. gsize pos;
  230. if (p1 == p2)
  231. return;
  232. if (p1 > p2) {
  233. pos = p2;
  234. len = p1 - p2;
  235. }
  236. else{
  237. pos = p1;
  238. len = p2 - p1;
  239. }
  240. if (len > strlen(text_to_insert)) {
  241. len = strlen(text_to_insert);
  242. }
  243. if (pos > gstr->len)
  244. pos = gstr->len;
  245. /* ouch this is ugly but gtk1 needs it */
  246. if ((pos + len) > gstr->len)
  247. g_string_truncate(gstr, pos);
  248. else
  249. g_string_erase(gstr, pos, len);
  250. g_string_insert(gstr, pos, text_to_insert);
  251. }
  252. /****************************************************************************/
  253. static gboolean
  254. dialog_graph_dump_to_file(char *pathname, graph_analysis_data_t *user_data)
  255. {
  256. guint32 i, first_node, display_items, display_nodes;
  257. guint32 start_position, end_position, item_width, header_length;
  258. graph_analysis_item_t *gai;
  259. guint16 first_conv_num = 0;
  260. gboolean several_convs = FALSE;
  261. gboolean first_packet = TRUE;
  262. GString *label_string, *empty_line, *separator_line, *tmp_str, *tmp_str2;
  263. const char *empty_header;
  264. char src_port[8], dst_port[8];
  265. gchar *time_str;
  266. GList *list;
  267. FILE *of;
  268. of = ws_fopen(pathname, "w");
  269. if (of==NULL) {
  270. open_failure_alert_box(pathname, errno, TRUE);
  271. return FALSE;
  272. }
  273. time_str = (gchar *)g_malloc(COL_MAX_LEN);
  274. label_string = g_string_new("");
  275. empty_line = g_string_new("");
  276. separator_line = g_string_new("");
  277. tmp_str = g_string_new("");
  278. tmp_str2 = g_string_new("");
  279. display_items = 0;
  280. list = g_list_first(user_data->graph_info->list);
  281. while (list)
  282. {
  283. gai = (graph_analysis_item_t *)list->data;
  284. list = g_list_next(list);
  285. if (!gai->display)
  286. continue;
  287. display_items += 1;
  288. if (first_packet) {
  289. first_conv_num = gai->conv_num;
  290. first_packet = FALSE;
  291. }
  292. else if (gai->conv_num != first_conv_num) {
  293. several_convs = TRUE;
  294. }
  295. }
  296. /* if not items to display */
  297. if (display_items == 0)
  298. goto exit;
  299. display_nodes = user_data->num_nodes;
  300. first_node = user_data->dlg.first_node;
  301. /* Write the conv. and time headers */
  302. if (several_convs) {
  303. fprintf(of, CONV_TIME_HEADER);
  304. empty_header = CONV_TIME_EMPTY_HEADER;
  305. header_length = CONV_TIME_HEADER_LENGTH;
  306. }
  307. else{
  308. fprintf(of, TIME_HEADER);
  309. empty_header = TIME_EMPTY_HEADER;
  310. header_length = TIME_HEADER_LENGTH;
  311. }
  312. /* Write the node names on top */
  313. for (i=0; i<display_nodes; i+=2) {
  314. /* print the node identifiers */
  315. g_string_printf(label_string, "| %s",
  316. get_addr_name(&(user_data->nodes[i+first_node])));
  317. enlarge_string(label_string, NODE_CHARS_WIDTH*2, ' ');
  318. fprintf(of, "%s", label_string->str);
  319. g_string_printf(label_string, "| ");
  320. enlarge_string(label_string, NODE_CHARS_WIDTH, ' ');
  321. g_string_append(empty_line, label_string->str);
  322. }
  323. fprintf(of, "|\n%s", empty_header);
  324. g_string_printf(label_string, "| ");
  325. enlarge_string(label_string, NODE_CHARS_WIDTH, ' ');
  326. fprintf(of, "%s", label_string->str);
  327. /* Write the node names on top */
  328. for (i=1; i<display_nodes; i+=2) {
  329. /* print the node identifiers */
  330. g_string_printf(label_string, "| %s",
  331. get_addr_name(&(user_data->nodes[i+first_node])));
  332. if (label_string->len < NODE_CHARS_WIDTH)
  333. {
  334. enlarge_string(label_string, NODE_CHARS_WIDTH, ' ');
  335. g_string_append(label_string, "| ");
  336. }
  337. enlarge_string(label_string, NODE_CHARS_WIDTH*2, ' ');
  338. fprintf(of, "%s", label_string->str);
  339. g_string_printf(label_string, "| ");
  340. enlarge_string(label_string, NODE_CHARS_WIDTH, ' ');
  341. g_string_append(empty_line, label_string->str);
  342. }
  343. fprintf(of, "\n");
  344. g_string_append_c(empty_line, '|');
  345. enlarge_string(separator_line, (guint32) empty_line->len + header_length, '-');
  346. /*
  347. * Draw the items
  348. */
  349. list = g_list_first(user_data->graph_info->list);
  350. while (list)
  351. {
  352. gai = (graph_analysis_item_t *)list->data;
  353. list = g_list_next(list);
  354. if (!gai->display)
  355. continue;
  356. start_position = (gai->src_node-first_node)*NODE_CHARS_WIDTH+NODE_CHARS_WIDTH/2;
  357. end_position = (gai->dst_node-first_node)*NODE_CHARS_WIDTH+NODE_CHARS_WIDTH/2;
  358. if (start_position > end_position) {
  359. item_width = start_position-end_position;
  360. }
  361. else if (start_position < end_position) {
  362. item_width = end_position-start_position;
  363. }
  364. else{ /* same origin and destination address */
  365. end_position = start_position+NODE_CHARS_WIDTH;
  366. item_width = NODE_CHARS_WIDTH;
  367. }
  368. /* separator between conversations */
  369. if (gai->conv_num != first_conv_num) {
  370. fprintf(of, "%s\n", separator_line->str);
  371. first_conv_num = gai->conv_num;
  372. }
  373. /* write the conversation number */
  374. if (several_convs) {
  375. g_string_printf(label_string, "%i", gai->conv_num);
  376. enlarge_string(label_string, 5, ' ');
  377. fprintf(of, "%s", label_string->str);
  378. }
  379. #if 0
  380. /* write the time */
  381. g_string_printf(label_string, "|%.3f", nstime_to_sec(&gai->fd->rel_ts));
  382. #endif
  383. /* Write the time, using the same format as in the time col */
  384. set_fd_time(cfile.epan, gai->fd, time_str);
  385. g_string_printf(label_string, "|%s", time_str);
  386. enlarge_string(label_string, 10, ' ');
  387. fprintf(of, "%s", label_string->str);
  388. /* write the frame label */
  389. g_string_printf(tmp_str, "%s", empty_line->str);
  390. overwrite(tmp_str, gai->frame_label,
  391. start_position,
  392. end_position
  393. );
  394. fprintf(of, "%s", tmp_str->str);
  395. /* write the comments */
  396. fprintf(of, "%s\n", gai->comment);
  397. /* write the arrow and frame label*/
  398. fprintf(of, "%s", empty_header);
  399. g_string_printf(tmp_str, "%s", empty_line->str);
  400. g_string_truncate(tmp_str2, 0);
  401. if (start_position<end_position) {
  402. enlarge_string(tmp_str2, item_width-2, '-');
  403. g_string_append_c(tmp_str2, '>');
  404. }
  405. else{
  406. g_string_printf(tmp_str2, "<");
  407. enlarge_string(tmp_str2, item_width-1, '-');
  408. }
  409. overwrite(tmp_str, tmp_str2->str,
  410. start_position,
  411. end_position
  412. );
  413. g_snprintf(src_port, sizeof(src_port), "(%i)", gai->port_src);
  414. g_snprintf(dst_port, sizeof(dst_port), "(%i)", gai->port_dst);
  415. if (start_position<end_position) {
  416. overwrite(tmp_str, src_port, start_position-9, start_position-1);
  417. overwrite(tmp_str, dst_port, end_position+1, end_position+9);
  418. }
  419. else{
  420. overwrite(tmp_str, src_port, start_position+1, start_position+9);
  421. overwrite(tmp_str, dst_port, end_position-9, end_position+1);
  422. }
  423. fprintf(of, "%s\n", tmp_str->str);
  424. }
  425. exit:
  426. g_string_free(label_string, TRUE);
  427. g_string_free(empty_line, TRUE);
  428. g_string_free(separator_line, TRUE);
  429. g_string_free(tmp_str, TRUE);
  430. g_string_free(tmp_str2, TRUE);
  431. g_free(time_str);
  432. fclose (of);
  433. return TRUE;
  434. }
  435. /****************************************************************************/
  436. /* save in a file */
  437. static char *
  438. gtk_save_graph_as_plain_text_file(graph_analysis_data_t *user_data)
  439. {
  440. GtkWidget *save_to_file_w;
  441. char *pathname;
  442. save_to_file_w = file_selection_new("Wireshark: Save graph to plain text file",
  443. GTK_WINDOW(user_data->dlg.window),
  444. FILE_SELECTION_SAVE);
  445. gtk_dialog_set_default_response(GTK_DIALOG(save_to_file_w),
  446. GTK_RESPONSE_ACCEPT);
  447. pathname = file_selection_run(save_to_file_w);
  448. if (pathname == NULL) {
  449. /* User cancelled or closed the dialog. */
  450. return NULL;
  451. }
  452. /* We've crosed the Rubicon; get rid of the dialog box. */
  453. window_destroy(save_to_file_w);
  454. return pathname;
  455. }
  456. /****************************************************************************/
  457. static void
  458. on_save_bt_clicked (GtkWidget *button _U_,
  459. graph_analysis_data_t *user_data)
  460. {
  461. char *pathname;
  462. /*
  463. * Loop until the user either selects a file or gives up.
  464. */
  465. for (;;) {
  466. pathname = gtk_save_graph_as_plain_text_file(user_data);
  467. if (pathname == NULL) {
  468. /* User gave up. */
  469. break;
  470. }
  471. if (dialog_graph_dump_to_file(pathname, user_data)) {
  472. /* We succeeded. */
  473. g_free(pathname);
  474. break;
  475. }
  476. /* Dump failed; let the user select another file
  477. or give up. */
  478. g_free(pathname);
  479. }
  480. }
  481. /****************************************************************************/
  482. static void dialog_graph_draw(graph_analysis_data_t *user_data)
  483. {
  484. guint32 i, last_item, first_item, display_items;
  485. guint32 start_arrow, end_arrow, label_x, src_port_x, dst_port_x, arrow_width;
  486. guint32 current_item;
  487. guint32 left_x_border;
  488. guint32 right_x_border;
  489. guint32 top_y_border;
  490. guint32 bottom_y_border;
  491. graph_analysis_item_t *gai;
  492. PangoLayout *layout;
  493. PangoLayout *middle_layout;
  494. PangoLayout *small_layout;
  495. PangoFontDescription *middle_font_desc;
  496. gint middle_font_size;
  497. PangoFontDescription *small_font_desc;
  498. gint small_font_size;
  499. gint label_width, label_height;
  500. guint32 draw_width, draw_height;
  501. char label_string[MAX_COMMENT];
  502. GList *list;
  503. cairo_t *cr;
  504. gchar *time_str;
  505. #if 0
  506. GdkColor *color_p, *bg_color_p;
  507. GdkColor black_color = {0, 0, 0, 0};
  508. GdkColor white_color = {0, 0xffff, 0xffff, 0xffff};
  509. /* gray and soft gray colors */
  510. GdkColor grey_color0 = {0, 0x64ff, 0x64ff, 0x64ff};
  511. GdkColor grey_color1 = {0, 0x25ff, 0x25ff, 0x25ff};
  512. /* the first color is blue to highlight the selected item */
  513. static GdkColor background_color[MAX_NUM_COL_CONV+1] = {
  514. {0, 0x00FF, 0x00FF, 0xFFFF},
  515. {0, 0x90FF, 0xEEFF, 0x90FF},
  516. {0, 0xFFFF, 0xA0FF, 0x7AFF},
  517. {0, 0xFFFF, 0xB6FF, 0xC1FF},
  518. {0, 0xFAFF, 0xFAFF, 0xD2FF},
  519. {0, 0xFFFF, 0xFFFF, 0x33FF},
  520. {0, 0x66FF, 0xCDFF, 0xAAFF},
  521. {0, 0xE0FF, 0xFFFF, 0xFFFF},
  522. {0, 0xB0FF, 0xC4FF, 0xDEFF},
  523. {0, 0x87FF, 0xCEFF, 0xFAFF},
  524. {0, 0xD3FF, 0xD3FF, 0xD3FF}
  525. };
  526. #endif
  527. GdkRGBA *color_p, *bg_color_p;
  528. GdkRGBA black_color = {0.0, 0.0, 0.0, 1.0}; /* Black */
  529. GdkRGBA white_color = {1.0, 1.0, 1.0, 1.0 };
  530. /* gray and soft gray colors */
  531. GdkRGBA grey_color0 = {0.3945, 0.3945, 0.3945, 1.0};
  532. GdkRGBA grey_color1 = {0.1484, 0.1484, 0.1484, 1.0};
  533. static GdkRGBA background_color[MAX_NUM_COL_CONV+1] = {
  534. /* Red, Green, Blue Alpha */
  535. {0.0039, 0.0039, 1.0000, 1.0},
  536. {0.5664, 0.6289, 0.5664, 1.0},
  537. {1.0000, 0.6289, 0.4805, 1.0},
  538. {1.0000, 0.7148, 0.7578, 1.0},
  539. {0.9805, 0.9805, 0.8242, 1.0},
  540. {1.0000, 1.0000, 0.2031, 1.0},
  541. {0.4023, 0.8046, 0.6680, 1.0},
  542. {0.8789, 1.0000, 1.0000, 1.0},
  543. {0.6914, 0.7695, 0.8710, 1.0},
  544. {0.8281, 0.8281, 0.8281, 1.0},
  545. };
  546. /* XXX can't we just set the background color ? */
  547. GdkPixbuf *bg_pixbuf;
  548. /* Dashed line pattern */
  549. static const double dashed1[] = {5.0, 4.0};
  550. static int len1 = sizeof(dashed1) / sizeof(dashed1[0]);
  551. GtkAllocation draw_area_time_alloc, draw_area_alloc, draw_area_comments_alloc;
  552. if(!user_data->dlg.needs_redraw) {
  553. return;
  554. }
  555. bg_pixbuf = gdk_pixbuf_new_from_xpm_data(voip_bg_xpm);
  556. time_str = (gchar *)g_malloc(COL_MAX_LEN);
  557. user_data->dlg.needs_redraw = FALSE;
  558. gtk_widget_get_allocation(user_data->dlg.draw_area_time, &draw_area_time_alloc);
  559. gtk_widget_get_allocation(user_data->dlg.draw_area, &draw_area_alloc);
  560. gtk_widget_get_allocation(user_data->dlg.draw_area_comments, &draw_area_comments_alloc);
  561. /* Clear out old plot */
  562. #if GTK_CHECK_VERSION(2,22,0)
  563. cr = cairo_create (user_data->dlg.surface_time);
  564. cairo_set_source_rgb (cr, 1, 1, 1);
  565. cairo_rectangle (cr, 0, 0, draw_area_time_alloc.width, draw_area_time_alloc.height);
  566. cairo_fill (cr);
  567. cairo_destroy (cr);
  568. cr = cairo_create (user_data->dlg.surface_main);
  569. cairo_set_source_rgb (cr, 1, 1, 1);
  570. cairo_rectangle (cr, 0, 0, draw_area_alloc.width, draw_area_alloc.height);
  571. cairo_fill (cr);
  572. cairo_destroy (cr);
  573. cr = cairo_create (user_data->dlg.surface_comments);
  574. cairo_set_source_rgb (cr, 1, 1, 1);
  575. cairo_rectangle (cr, 0, 0, draw_area_comments_alloc.width, draw_area_comments_alloc.height);
  576. cairo_fill (cr);
  577. cairo_destroy (cr);
  578. #else
  579. if ( GDK_IS_DRAWABLE(user_data->dlg.pixmap_time) ) {
  580. cr = gdk_cairo_create (user_data->dlg.pixmap_time);
  581. cairo_set_source_rgb (cr, 1, 1, 1);
  582. cairo_rectangle (cr, 0, 0, draw_area_time_alloc.width, draw_area_time_alloc.height);
  583. cairo_fill (cr);
  584. cairo_destroy (cr);
  585. }
  586. if ( GDK_IS_DRAWABLE(user_data->dlg.pixmap_main) ) {
  587. cr = gdk_cairo_create (user_data->dlg.pixmap_main);
  588. cairo_set_source_rgb (cr, 1, 1, 1);
  589. cairo_rectangle (cr, 0, 0, draw_area_alloc.width, draw_area_alloc.height);
  590. cairo_fill (cr);
  591. cairo_destroy (cr);
  592. }
  593. if ( GDK_IS_DRAWABLE(user_data->dlg.pixmap_comments) ) {
  594. cr = gdk_cairo_create (user_data->dlg.pixmap_comments);
  595. cairo_set_source_rgb (cr, 1, 1, 1);
  596. cairo_rectangle (cr, 0, 0, draw_area_comments_alloc.width, draw_area_comments_alloc.height);
  597. cairo_fill (cr);
  598. cairo_destroy (cr);
  599. }
  600. #endif
  601. /* Calculate the y border */
  602. top_y_border = TOP_Y_BORDER; /* to display the node address */
  603. bottom_y_border = BOTTOM_Y_BORDER;
  604. draw_height = draw_area_alloc.height-top_y_border-bottom_y_border;
  605. first_item = user_data->dlg.first_item;
  606. display_items = draw_height/ITEM_HEIGHT;
  607. /* get the items to display and fill the matrix array */
  608. list = g_list_first(user_data->graph_info->list);
  609. current_item = 0;
  610. i = 0;
  611. while (list)
  612. {
  613. gai = (graph_analysis_item_t *)list->data;
  614. if (gai->display) {
  615. if (current_item>=display_items) break; /* the item is outside the display */
  616. if (i>=first_item) {
  617. user_data->dlg.items[current_item].fd = gai->fd;
  618. user_data->dlg.items[current_item].port_src = gai->port_src;
  619. user_data->dlg.items[current_item].port_dst = gai->port_dst;
  620. /* Add "..." if the length is 50 characters */
  621. if (strlen(gai->frame_label) > 48) {
  622. gai->frame_label[48] = '.';
  623. gai->frame_label[47] = '.';
  624. gai->frame_label[46] = '.';
  625. }
  626. user_data->dlg.items[current_item].frame_label = gai->frame_label;
  627. user_data->dlg.items[current_item].comment = gai->comment;
  628. user_data->dlg.items[current_item].conv_num = gai->conv_num;
  629. user_data->dlg.items[current_item].src_node = gai->src_node;
  630. user_data->dlg.items[current_item].dst_node = gai->dst_node;
  631. user_data->dlg.items[current_item].line_style = gai->line_style;
  632. current_item++;
  633. }
  634. i++;
  635. }
  636. list = g_list_next(list);
  637. }
  638. /* in case the window is resized we might have to move the top item */
  639. if ((first_item + display_items) > user_data->num_items) {
  640. if (display_items>user_data->num_items)
  641. first_item = 0;
  642. else
  643. first_item = user_data->num_items - display_items;
  644. }
  645. /* in case there are less items than possible displayed */
  646. display_items = current_item;
  647. last_item = first_item+display_items-1;
  648. user_data->dlg.last_item = last_item;
  649. /* if no items to display */
  650. if (display_items == 0) {
  651. g_free(time_str);
  652. return;
  653. }
  654. /* Calculate the x borders */
  655. /* We use time from the last display item to calcultate the x left border */
  656. #if 0
  657. g_snprintf(label_string, MAX_LABEL, "%.3f", nstime_to_sec(&user_data->dlg.items[display_items-1].fd->rel_ts));
  658. #endif
  659. /* Write the time, using the same format as in th etime col */
  660. set_fd_time(cfile.epan, user_data->dlg.items[display_items-1].fd, time_str);
  661. g_snprintf(label_string, MAX_LABEL, "%s", time_str);
  662. layout = gtk_widget_create_pango_layout(user_data->dlg.draw_area_time, label_string);
  663. middle_layout = gtk_widget_create_pango_layout(user_data->dlg.draw_area_time, label_string);
  664. small_layout = gtk_widget_create_pango_layout(user_data->dlg.draw_area_time, label_string);
  665. middle_font_desc = pango_font_description_copy(pango_context_get_font_description(pango_layout_get_context(middle_layout)));
  666. middle_font_size = pango_font_description_get_size(middle_font_desc);
  667. pango_font_description_set_size(middle_font_desc, (gint)(middle_font_size*0.8));
  668. pango_layout_set_font_description(middle_layout, middle_font_desc);
  669. small_font_desc = pango_font_description_copy(pango_context_get_font_description(pango_layout_get_context(small_layout)));
  670. small_font_size = pango_font_description_get_size(small_font_desc);
  671. pango_font_description_set_size(small_font_desc, (gint)(small_font_size*0.7));
  672. pango_layout_set_font_description(small_layout, small_font_desc);
  673. pango_layout_get_pixel_size(layout, &label_width, &label_height);
  674. /* resize the "time" draw area */
  675. left_x_border = 0;
  676. user_data->dlg.left_x_border = left_x_border;
  677. right_x_border = 0;
  678. draw_width = user_data->dlg.surface_width-right_x_border-left_x_border;
  679. #if GTK_CHECK_VERSION(2,22,0)
  680. /* Paint time title background */
  681. cr = cairo_create (user_data->dlg.surface_time);
  682. gdk_cairo_set_source_pixbuf (cr, bg_pixbuf, 0, 0);
  683. cairo_pattern_set_extend (cairo_get_source (cr), CAIRO_EXTEND_REPEAT);
  684. cairo_rectangle (cr, 0, 0, draw_area_time_alloc.width, top_y_border);
  685. cairo_fill (cr);
  686. cairo_destroy (cr);
  687. /* Paint main title background */
  688. cr = cairo_create (user_data->dlg.surface_main);
  689. gdk_cairo_set_source_pixbuf (cr, bg_pixbuf, 0, 0);
  690. cairo_pattern_set_extend (cairo_get_source (cr), CAIRO_EXTEND_REPEAT);
  691. cairo_rectangle (cr, 0, 0, draw_area_alloc.width, top_y_border);
  692. cairo_fill (cr);
  693. cairo_destroy (cr);
  694. /* Paint main comment background */
  695. cr = cairo_create (user_data->dlg.surface_comments);
  696. gdk_cairo_set_source_pixbuf (cr, bg_pixbuf, 0, 0);
  697. cairo_pattern_set_extend (cairo_get_source (cr), CAIRO_EXTEND_REPEAT);
  698. cairo_rectangle (cr, 0, 0, draw_area_comments_alloc.width, top_y_border);
  699. cairo_fill (cr);
  700. cairo_destroy (cr);
  701. #else
  702. /* Paint time title background */
  703. if ( GDK_IS_DRAWABLE(user_data->dlg.pixmap_time) ) {
  704. cr = gdk_cairo_create (user_data->dlg.pixmap_time);
  705. gdk_cairo_set_source_pixbuf (cr, bg_pixbuf, 0, 0);
  706. cairo_pattern_set_extend (cairo_get_source (cr), CAIRO_EXTEND_REPEAT);
  707. cairo_rectangle (cr, 0, 0, draw_area_time_alloc.width, top_y_border);
  708. cairo_fill (cr);
  709. cairo_destroy (cr);
  710. }
  711. /* Paint main title background */
  712. if ( GDK_IS_DRAWABLE(user_data->dlg.pixmap_main) ) {
  713. cr = gdk_cairo_create (user_data->dlg.pixmap_main);
  714. gdk_cairo_set_source_pixbuf (cr, bg_pixbuf, 0, 0);
  715. cairo_pattern_set_extend (cairo_get_source (cr), CAIRO_EXTEND_REPEAT);
  716. cairo_rectangle (cr, 0, 0, draw_area_alloc.width, top_y_border);
  717. cairo_fill (cr);
  718. cairo_destroy (cr);
  719. }
  720. /* Paint main comment background */
  721. if ( GDK_IS_DRAWABLE(user_data->dlg.pixmap_comments) ) {
  722. cr = gdk_cairo_create (user_data->dlg.pixmap_comments);
  723. gdk_cairo_set_source_pixbuf (cr, bg_pixbuf, 0, 0);
  724. cairo_pattern_set_extend (cairo_get_source (cr), CAIRO_EXTEND_REPEAT);
  725. cairo_rectangle (cr, 0, 0, draw_area_comments_alloc.width, top_y_border);
  726. cairo_fill (cr);
  727. cairo_destroy (cr);
  728. }
  729. #endif
  730. /* Draw the word "Time" on top of time column */
  731. g_snprintf(label_string, label_width, "%s", " Time");
  732. pango_layout_set_text(layout, label_string, -1);
  733. pango_layout_get_pixel_size(layout, &label_width, &label_height);
  734. #if GTK_CHECK_VERSION(2,22,0)
  735. cr = cairo_create (user_data->dlg.surface_time);
  736. cairo_move_to (cr, left_x_border, top_y_border/2-label_height/2);
  737. pango_cairo_show_layout (cr, layout);
  738. cairo_destroy (cr);
  739. cr = NULL;
  740. #else
  741. if (GDK_IS_DRAWABLE(user_data->dlg.pixmap_time)) {
  742. cr = gdk_cairo_create (user_data->dlg.pixmap_time);
  743. cairo_move_to (cr, left_x_border, top_y_border/2-label_height/2);
  744. pango_cairo_show_layout (cr, layout);
  745. cairo_destroy (cr);
  746. cr = NULL;
  747. }
  748. #endif
  749. /* Draw the word "Comment" on top of comment column */
  750. g_snprintf(label_string, label_width, "%s", "Comment");
  751. pango_layout_set_text(layout, label_string, -1);
  752. pango_layout_get_pixel_size(layout, &label_width, &label_height);
  753. #if GTK_CHECK_VERSION(2,22,0)
  754. cr = cairo_create (user_data->dlg.surface_comments);
  755. cairo_move_to (cr, MAX_COMMENT/2-label_width/2, top_y_border/2-label_height/2);
  756. pango_cairo_show_layout (cr, layout);
  757. cairo_destroy (cr);
  758. cr = NULL;
  759. #else
  760. if (GDK_IS_DRAWABLE(user_data->dlg.pixmap_comments)) {
  761. cr = gdk_cairo_create (user_data->dlg.pixmap_comments);
  762. cairo_move_to (cr, MAX_COMMENT/2-label_width/2, top_y_border/2-label_height/2);
  763. pango_cairo_show_layout (cr, layout);
  764. cairo_destroy (cr);
  765. cr = NULL;
  766. }
  767. #endif
  768. /* Paint the background items */
  769. for (current_item=0; current_item<display_items; current_item++) {
  770. /*select the color. if it is the selected item select blue color */
  771. if ( current_item+first_item == user_data->dlg.selected_item ) {
  772. bg_color_p = &background_color[0]; /* blue */
  773. } else {
  774. bg_color_p = &background_color[1+user_data->dlg.items[current_item].conv_num%MAX_NUM_COL_CONV];
  775. }
  776. #if GTK_CHECK_VERSION(2,22,0)
  777. /* Paint background */
  778. cr = cairo_create (user_data->dlg.surface_main);
  779. gdk_cairo_set_source_rgba (cr, bg_color_p);
  780. cairo_rectangle (cr, left_x_border, top_y_border+current_item*ITEM_HEIGHT, draw_width, ITEM_HEIGHT);
  781. cairo_fill (cr);
  782. cairo_destroy (cr);
  783. #else
  784. if (GDK_IS_DRAWABLE(user_data->dlg.pixmap_main)) {
  785. /* Paint background */
  786. cr = gdk_cairo_create (user_data->dlg.pixmap_main);
  787. gdk_cairo_set_source_rgba (cr, bg_color_p);
  788. cairo_rectangle (cr, left_x_border, top_y_border+current_item*ITEM_HEIGHT, draw_width, ITEM_HEIGHT);
  789. cairo_fill (cr);
  790. cairo_destroy (cr);
  791. }
  792. #endif
  793. }
  794. /* Draw the node names on top and the division lines */
  795. for (i=0; i<user_data->num_nodes; i++) {
  796. /* print the node identifiers */
  797. /* XXX we assign 5 pixels per character in the node identity */
  798. g_strlcpy(label_string, get_addr_name(&(user_data->nodes[i])), NODE_WIDTH/5);
  799. pango_layout_set_text(layout, label_string, -1);
  800. pango_layout_get_pixel_size(layout, &label_width, &label_height);
  801. #if GTK_CHECK_VERSION(2,22,0)
  802. cr = cairo_create (user_data->dlg.surface_main);
  803. cairo_move_to (cr, left_x_border+NODE_WIDTH/2-label_width/2+NODE_WIDTH*i, top_y_border/2-((i&1)?0:label_height));
  804. pango_cairo_show_layout (cr, layout);
  805. cairo_destroy (cr);
  806. cr = NULL;
  807. #else
  808. if (GDK_IS_DRAWABLE(user_data->dlg.pixmap_main)) {
  809. cr = gdk_cairo_create (user_data->dlg.pixmap_main);
  810. cairo_move_to (cr, left_x_border+NODE_WIDTH/2-label_width/2+NODE_WIDTH*i, top_y_border/2-((i&1)?0:label_height));
  811. pango_cairo_show_layout (cr, layout);
  812. cairo_destroy (cr);
  813. cr = NULL;
  814. }
  815. #endif
  816. #if GTK_CHECK_VERSION(2,22,0)
  817. /* draw the node division lines */
  818. cr = cairo_create (user_data->dlg.surface_main);
  819. gdk_cairo_set_source_rgba (cr, &grey_color0);
  820. cairo_set_line_width (cr, 1.0);
  821. cairo_set_line_cap(cr, CAIRO_LINE_CAP_BUTT);
  822. cairo_set_dash(cr, dashed1, len1, 0);
  823. cairo_move_to(cr, left_x_border+NODE_WIDTH/2+NODE_WIDTH*i, top_y_border);
  824. cairo_line_to(cr, (left_x_border+NODE_WIDTH/2+NODE_WIDTH*i), draw_area_alloc.height-bottom_y_border);
  825. cairo_stroke(cr);
  826. cairo_destroy(cr);
  827. #else
  828. /* draw the node division lines */
  829. if (GDK_IS_DRAWABLE(user_data->dlg.pixmap_main) ) {
  830. cr = gdk_cairo_create (user_data->dlg.pixmap_main);
  831. gdk_cairo_set_source_rgba (cr, &grey_color0);
  832. cairo_set_line_width (cr, 1.0);
  833. cairo_set_line_cap(cr, CAIRO_LINE_CAP_BUTT);
  834. cairo_set_dash(cr, dashed1, len1, 0);
  835. cairo_move_to(cr, left_x_border+NODE_WIDTH/2+NODE_WIDTH*i, top_y_border);
  836. cairo_line_to(cr, (left_x_border+NODE_WIDTH/2+NODE_WIDTH*i), draw_area_alloc.height-bottom_y_border);
  837. cairo_stroke(cr);
  838. cairo_destroy(cr);
  839. }
  840. #endif
  841. }
  842. /* Draw the items */
  843. for (current_item=0; current_item<display_items; current_item++) {
  844. #if 0
  845. /* Draw the time */
  846. g_snprintf(label_string, MAX_LABEL, "%.3f", nstime_to_sec(&user_data->dlg.items[current_item].fd->rel_ts));
  847. #endif
  848. /* Draw the time */
  849. set_fd_time(cfile.epan, user_data->dlg.items[current_item].fd, time_str);
  850. g_snprintf(label_string, MAX_LABEL, "%s", time_str);
  851. pango_layout_set_text(layout, label_string, -1);
  852. pango_layout_get_pixel_size(layout, &label_width, &label_height);
  853. #if GTK_CHECK_VERSION(2,22,0)
  854. cr = cairo_create (user_data->dlg.surface_time);
  855. cairo_move_to (cr, 3, top_y_border+current_item*ITEM_HEIGHT+ITEM_HEIGHT/2-label_height/2);
  856. pango_cairo_show_layout (cr, layout);
  857. cairo_destroy (cr);
  858. cr = NULL;
  859. #else
  860. if (GDK_IS_DRAWABLE(user_data->dlg.pixmap_time)) {
  861. cr = gdk_cairo_create (user_data->dlg.pixmap_time);
  862. cairo_move_to (cr, 3, top_y_border+current_item*ITEM_HEIGHT+ITEM_HEIGHT/2-label_height/2);
  863. pango_cairo_show_layout (cr, layout);
  864. cairo_destroy (cr);
  865. cr = NULL;
  866. }
  867. #endif
  868. /*draw the comments */
  869. g_snprintf(label_string, MAX_COMMENT, "%s", user_data->dlg.items[current_item].comment);
  870. pango_layout_set_text(middle_layout, label_string, -1);
  871. pango_layout_get_pixel_size(middle_layout, &label_width, &label_height);
  872. #if GTK_CHECK_VERSION(2,22,0)
  873. cr = cairo_create (user_data->dlg.surface_comments);
  874. cairo_move_to (cr, 2, top_y_border+current_item*ITEM_HEIGHT+ITEM_HEIGHT/2-label_height/2);
  875. pango_cairo_show_layout (cr, middle_layout);
  876. cairo_destroy (cr);
  877. cr = NULL;
  878. #else
  879. if (GDK_IS_DRAWABLE(user_data->dlg.pixmap_comments)) {
  880. cr = gdk_cairo_create (user_data->dlg.pixmap_comments);
  881. cairo_move_to (cr, 2, top_y_border+current_item*ITEM_HEIGHT+ITEM_HEIGHT/2-label_height/2);
  882. pango_cairo_show_layout (cr, middle_layout);
  883. cairo_destroy (cr);
  884. cr = NULL;
  885. }
  886. #endif
  887. /* draw the arrow line */
  888. start_arrow = left_x_border+(user_data->dlg.items[current_item].src_node)*NODE_WIDTH+NODE_WIDTH/2;
  889. end_arrow = left_x_border+(user_data->dlg.items[current_item].dst_node)*NODE_WIDTH+NODE_WIDTH/2;
  890. #if GTK_CHECK_VERSION(2,22,0)
  891. cr = cairo_create (user_data->dlg.surface_main);
  892. if (user_data->dlg.items[current_item].line_style == 2) {
  893. /* draw a line thick */
  894. cairo_set_line_width (cr, 2.0);
  895. }else{
  896. cairo_set_line_width (cr, 1.0);
  897. }
  898. if ( current_item+first_item == user_data->dlg.selected_item ) {
  899. /* draw white line */
  900. cairo_set_source_rgb (cr, 1, 1, 1);
  901. }else{
  902. /* draw black line */
  903. cairo_set_source_rgb (cr, 0, 0, 0);
  904. }
  905. cairo_move_to(cr, start_arrow, (top_y_border+current_item*ITEM_HEIGHT+ITEM_HEIGHT-7)+0.5);
  906. cairo_line_to(cr, end_arrow, (top_y_border+current_item*ITEM_HEIGHT+ITEM_HEIGHT-7)+0.5);
  907. cairo_stroke(cr);
  908. cairo_destroy(cr);
  909. #else
  910. if (GDK_IS_DRAWABLE(user_data->dlg.pixmap_main) ) {
  911. cr = gdk_cairo_create (user_data->dlg.pixmap_main);
  912. if (user_data->dlg.items[current_item].line_style == 2) {
  913. /* draw a line thick */
  914. cairo_set_line_width (cr, 2.0);
  915. }else{
  916. cairo_set_line_width (cr, 1.0);
  917. }
  918. if ( current_item+first_item == user_data->dlg.selected_item ) {
  919. /* draw white line */
  920. cairo_set_source_rgb (cr, 1, 1, 1);
  921. }else{
  922. /* draw black line */
  923. cairo_set_source_rgb (cr, 0, 0, 0);
  924. }
  925. cairo_move_to(cr, start_arrow, (top_y_border+current_item*ITEM_HEIGHT+ITEM_HEIGHT-7)+0.5);
  926. cairo_line_to(cr, end_arrow, (top_y_border+current_item*ITEM_HEIGHT+ITEM_HEIGHT-7)+0.5);
  927. cairo_stroke(cr);
  928. cairo_destroy(cr);
  929. }
  930. #endif
  931. /* select colors */
  932. if ( current_item+first_item == user_data->dlg.selected_item ) {
  933. color_p = &white_color;
  934. } else {
  935. color_p = &black_color;
  936. }
  937. /* draw the arrow */
  938. #if GTK_CHECK_VERSION(2,22,0)
  939. if (start_arrow<end_arrow)
  940. draw_arrow(user_data->dlg.surface_main, color_p, end_arrow-WIDTH_ARROW, (top_y_border+current_item*ITEM_HEIGHT+ITEM_HEIGHT-7)-(HEIGHT_ARROW/2), RIGHT_ARROW);
  941. else
  942. draw_arrow(user_data->dlg.surface_main, color_p, end_arrow, top_y_border+current_item*ITEM_HEIGHT+ITEM_HEIGHT-7-(HEIGHT_ARROW/2), LEFT_ARROW);
  943. #else
  944. if (start_arrow<end_arrow)
  945. draw_arrow(user_data->dlg.pixmap_main, color_p, end_arrow-WIDTH_ARROW, (top_y_border+current_item*ITEM_HEIGHT+ITEM_HEIGHT-7)-(HEIGHT_ARROW/2), RIGHT_ARROW);
  946. else
  947. draw_arrow(user_data->dlg.pixmap_main, color_p, end_arrow, top_y_border+current_item*ITEM_HEIGHT+ITEM_HEIGHT-7-(HEIGHT_ARROW/2), LEFT_ARROW);
  948. #endif
  949. /* draw the frame comment */
  950. g_snprintf(label_string, MAX_LABEL, "%s", user_data->dlg.items[current_item].frame_label);
  951. pango_layout_set_text(layout, label_string, -1);
  952. pango_layout_get_pixel_size(layout, &label_width, &label_height);
  953. if (start_arrow<end_arrow) {
  954. arrow_width = end_arrow-start_arrow;
  955. label_x = arrow_width/2+start_arrow;
  956. }
  957. else {
  958. arrow_width = start_arrow-end_arrow;
  959. label_x = arrow_width/2+end_arrow;
  960. }
  961. if ((int)left_x_border > ((int)label_x-(int)label_width/2))
  962. label_x = left_x_border + label_width/2;
  963. #if GTK_CHECK_VERSION(2,22,0)
  964. cr = cairo_create (user_data->dlg.surface_main);
  965. gdk_cairo_set_source_rgba (cr, color_p);
  966. cairo_move_to (cr, label_x - label_width/2, top_y_border+current_item*ITEM_HEIGHT+ITEM_HEIGHT/2-label_height/2-3);
  967. pango_cairo_show_layout (cr, layout);
  968. cairo_destroy (cr);
  969. cr = NULL;
  970. #else
  971. if (GDK_IS_DRAWABLE(user_data->dlg.pixmap_main)) {
  972. cr = gdk_cairo_create (user_data->dlg.pixmap_main);
  973. gdk_cairo_set_source_rgba (cr, color_p);
  974. cairo_move_to (cr, label_x - label_width/2, top_y_border+current_item*ITEM_HEIGHT+ITEM_HEIGHT/2-label_height/2-3);
  975. pango_cairo_show_layout (cr, layout);
  976. cairo_destroy (cr);
  977. cr = NULL;
  978. }
  979. #endif
  980. /* draw the source port number */
  981. g_snprintf(label_string, MAX_LABEL, "(%i)", user_data->dlg.items[current_item].port_src);
  982. pango_layout_set_text(small_layout, label_string, -1);
  983. pango_layout_get_pixel_size(small_layout, &label_width, &label_height);
  984. if (start_arrow<end_arrow) {
  985. src_port_x = start_arrow - label_width - 2;
  986. }
  987. else {
  988. src_port_x = start_arrow + 2;
  989. }
  990. #if GTK_CHECK_VERSION(2,22,0)
  991. cr = cairo_create (user_data->dlg.surface_main);
  992. /* select color */
  993. if ( current_item+first_item == user_data->dlg.selected_item ) {
  994. gdk_cairo_set_source_rgba (cr, &grey_color1);
  995. } else {
  996. gdk_cairo_set_source_rgba (cr, &grey_color0);
  997. }
  998. gdk_cairo_set_source_rgba (cr, &grey_color0);
  999. cairo_move_to (cr, src_port_x, top_y_border+current_item*ITEM_HEIGHT+ITEM_HEIGHT-2-label_height/2-2);
  1000. pango_cairo_show_layout (cr, small_layout);
  1001. cairo_destroy (cr);
  1002. #else
  1003. if (GDK_IS_DRAWABLE(user_data->dlg.pixmap_main)) {
  1004. cr = gdk_cairo_create (user_data->dlg.pixmap_main);
  1005. /* select color */
  1006. if ( current_item+first_item == user_data->dlg.selected_item ) {
  1007. gdk_cairo_set_source_rgba (cr, &grey_color1);
  1008. } else {
  1009. gdk_cairo_set_source_rgba (cr, &grey_color0);
  1010. }
  1011. gdk_cairo_set_source_rgba (cr, &grey_color0);
  1012. cairo_move_to (cr, src_port_x, top_y_border+current_item*ITEM_HEIGHT+ITEM_HEIGHT-2-label_height/2-2);
  1013. pango_cairo_show_layout (cr, small_layout);
  1014. cairo_destroy (cr);
  1015. }
  1016. #endif
  1017. /* draw the destination port number */
  1018. g_snprintf(label_string, MAX_LABEL, "(%i)", user_data->dlg.items[current_item].port_dst);
  1019. pango_layout_set_text(small_layout, label_string, -1);
  1020. pango_layout_get_pixel_size(small_layout, &label_width, &label_height);
  1021. if (start_arrow<end_arrow) {
  1022. dst_port_x = end_arrow + 2;
  1023. }
  1024. else {
  1025. dst_port_x = end_arrow - label_width - 2;
  1026. }
  1027. #if GTK_CHECK_VERSION(2,22,0)
  1028. cr = cairo_create (user_data->dlg.surface_main);
  1029. /* select color */
  1030. if ( current_item+first_item == user_data->dlg.selected_item ) {
  1031. gdk_cairo_set_source_rgba (cr, &grey_color1);
  1032. } else {
  1033. gdk_cairo_set_source_rgba (cr, &grey_color0);
  1034. }
  1035. cairo_move_to (cr, dst_port_x, top_y_border+current_item*ITEM_HEIGHT+ITEM_HEIGHT-2-label_height/2-2);
  1036. pango_cairo_show_layout (cr, small_layout);
  1037. cairo_destroy (cr);
  1038. #else
  1039. if (GDK_IS_DRAWABLE(user_data->dlg.pixmap_main)) {
  1040. cr = gdk_cairo_create (user_data->dlg.pixmap_main);
  1041. /* select color */
  1042. if ( current_item+first_item == user_data->dlg.selected_item ) {
  1043. gdk_cairo_set_source_rgba (cr, &grey_color1);
  1044. } else {
  1045. gdk_cairo_set_source_rgba (cr, &grey_color0);
  1046. }
  1047. cairo_move_to (cr, dst_port_x, top_y_border+current_item*ITEM_HEIGHT+ITEM_HEIGHT-2-label_height/2-2);
  1048. pango_cairo_show_layout (cr, small_layout);
  1049. cairo_destroy (cr);
  1050. }
  1051. #endif
  1052. /* draw the div line of the selected item with soft gray*/
  1053. if ( current_item+first_item == user_data->dlg.selected_item )
  1054. for (i=0; i<user_data->num_nodes; i++) {
  1055. #if GTK_CHECK_VERSION(2,22,0)
  1056. cr = cairo_create (user_data->dlg.surface_main);
  1057. gdk_cairo_set_source_rgba (cr, &grey_color1);
  1058. cairo_set_line_width (cr, 1.0);
  1059. cairo_set_line_cap(cr, CAIRO_LINE_CAP_BUTT);
  1060. cairo_set_dash(cr, dashed1, len1, 0);
  1061. cairo_move_to(cr, left_x_border+NODE_WIDTH/2+NODE_WIDTH*i, (user_data->dlg.selected_item-first_item)*ITEM_HEIGHT+TOP_Y_BORDER);
  1062. cairo_line_to(cr, (left_x_border+NODE_WIDTH/2+NODE_WIDTH*i), (user_data->dlg.selected_item-first_item)*ITEM_HEIGHT+TOP_Y_BORDER+ITEM_HEIGHT);
  1063. cairo_stroke(cr);
  1064. cairo_destroy(cr);
  1065. #else
  1066. if (GDK_IS_DRAWABLE(user_data->dlg.pixmap_main) ) {
  1067. cr = gdk_cairo_create (user_data->dlg.pixmap_main);
  1068. gdk_cairo_set_source_rgba (cr, &grey_color1);
  1069. cairo_set_line_width (cr, 1.0);
  1070. cairo_set_line_cap(cr, CAIRO_LINE_CAP_BUTT);
  1071. cairo_set_dash(cr, dashed1, len1, 0);
  1072. cairo_move_to(cr, left_x_border+NODE_WIDTH/2+NODE_WIDTH*i, (user_data->dlg.selected_item-first_item)*ITEM_HEIGHT+TOP_Y_BORDER);
  1073. cairo_line_to(cr, (left_x_border+NODE_WIDTH/2+NODE_WIDTH*i), (user_data->dlg.selected_item-first_item)*ITEM_HEIGHT+TOP_Y_BORDER+ITEM_HEIGHT);
  1074. cairo_stroke(cr);
  1075. cairo_destroy(cr);
  1076. }
  1077. #endif
  1078. }
  1079. }
  1080. g_object_unref(G_OBJECT(layout));
  1081. g_free(time_str);
  1082. /* refresh the draw areas */
  1083. if (gtk_widget_is_drawable(user_data->dlg.draw_area_time) ) {
  1084. cr = gdk_cairo_create (gtk_widget_get_window(user_data->dlg.draw_area_time));
  1085. #if GTK_CHECK_VERSION(2,22,0)
  1086. cairo_set_source_surface (cr, user_data->dlg.surface_time, 0, 0);
  1087. #else
  1088. gdk_cairo_set_source_pixmap (cr, user_data->dlg.pixmap_time, 0, 0);
  1089. #endif
  1090. cairo_rectangle (cr, 0, 0, draw_area_time_alloc.width, draw_area_time_alloc.height);
  1091. cairo_fill (cr);
  1092. cairo_destroy (cr);
  1093. cr = NULL;
  1094. }
  1095. if (gtk_widget_is_drawable(user_data->dlg.draw_area) ) {
  1096. cr = gdk_cairo_create (gtk_widget_get_window(user_data->dlg.draw_area));
  1097. #if GTK_CHECK_VERSION(2,22,0)
  1098. cairo_set_source_surface (cr, user_data->dlg.surface_main, 0, 0);
  1099. #else
  1100. gdk_cairo_set_source_pixmap (cr, user_data->dlg.pixmap_main, 0, 0);
  1101. #endif
  1102. cairo_rectangle (cr, 0, 0, draw_area_alloc.width, draw_area_alloc.height);
  1103. cairo_fill (cr);
  1104. cairo_destroy (cr);
  1105. cr = NULL;
  1106. }
  1107. if (gtk_widget_is_drawable(user_data->dlg.draw_area_comments) ) {
  1108. cr = gdk_cairo_create (gtk_widget_get_window(user_data->dlg.draw_area_comments));
  1109. #if GTK_CHECK_VERSION(2,22,0)
  1110. cairo_set_source_surface (cr, user_data->dlg.surface_comments, 0, 0);
  1111. #else
  1112. gdk_cairo_set_source_pixmap (cr, user_data->dlg.pixmap_comments, 0, 0);
  1113. #endif
  1114. cairo_rectangle (cr, 0, 0, draw_area_comments_alloc.width, draw_area_comments_alloc.height);
  1115. cairo_fill (cr);
  1116. cairo_destroy (cr);
  1117. cr = NULL;
  1118. }
  1119. /* update the v_scrollbar */
  1120. gtk_adjustment_set_upper(user_data->dlg.v_scrollbar_adjustment, (gdouble) user_data->num_items-1);
  1121. gtk_adjustment_set_step_increment(user_data->dlg.v_scrollbar_adjustment, 1);
  1122. gtk_adjustment_set_page_increment(user_data->dlg.v_scrollbar_adjustment, (gdouble) (last_item-first_item));
  1123. gtk_adjustment_set_page_size(user_data->dlg.v_scrollbar_adjustment, (gdouble) (last_item-first_item));
  1124. gtk_adjustment_set_value(user_data->dlg.v_scrollbar_adjustment, (gdouble) first_item);
  1125. gtk_adjustment_changed(user_data->dlg.v_scrollbar_adjustment);
  1126. gtk_adjustment_value_changed(user_data->dlg.v_scrollbar_adjustment);
  1127. }
  1128. /****************************************************************************/
  1129. static void dialog_graph_redraw(graph_analysis_data_t *user_data)
  1130. {
  1131. user_data->dlg.needs_redraw = TRUE;
  1132. dialog_graph_draw(user_data);
  1133. }
  1134. /****************************************************************************/
  1135. static gboolean button_press_event(GtkWidget *widget _U_, GdkEventButton *event, gpointer data)
  1136. {
  1137. graph_analysis_data_t *user_data = (graph_analysis_data_t *)data;
  1138. guint32 item;
  1139. if (event->type != GDK_BUTTON_PRESS) return TRUE;
  1140. if (event->y<TOP_Y_BORDER) return TRUE;
  1141. /* get the item clicked */
  1142. item = ((guint32)event->y - TOP_Y_BORDER) / ITEM_HEIGHT;
  1143. if (item > (user_data->dlg.last_item - user_data->dlg.first_item)) return TRUE;
  1144. user_data->dlg.selected_item = item + user_data->dlg.first_item;
  1145. user_data->dlg.needs_redraw = TRUE;
  1146. dialog_graph_draw(user_data);
  1147. cf_goto_frame(&cfile, user_data->dlg.items[item].fd->num);
  1148. return TRUE;
  1149. }
  1150. /****************************************************************************/
  1151. static gboolean key_press_event(GtkWidget *widget _U_, GdkEventKey *event, gpointer data)
  1152. {
  1153. graph_analysis_data_t *user_data = (graph_analysis_data_t *)data;
  1154. /* if there is nothing selected, just return */
  1155. if (user_data->dlg.selected_item == 0xFFFFFFFF) return TRUE;
  1156. /* Up arrow */
  1157. if (event->keyval == GDK_Up) {
  1158. if (user_data->dlg.selected_item == 0) return TRUE;
  1159. user_data->dlg.selected_item--;
  1160. if ( (user_data->dlg.selected_item<user_data->dlg.first_item) || (user_data->dlg.selected_item>user_data->dlg.first_item+gtk_adjustment_get_page_size(user_data->dlg.v_scrollbar_adjustment)) )
  1161. user_data->dlg.first_item = user_data->dlg.selected_item;
  1162. /* Down arrow */
  1163. } else if (event->keyval == GDK_Down) {
  1164. if (user_data->dlg.selected_item == user_data->num_items-1) return TRUE;
  1165. user_data->dlg.selected_item++;
  1166. if ( (user_data->dlg.selected_item<user_data->dlg.first_item) || (user_data->dlg.selected_item>user_data->dlg.first_item+gtk_adjustment_get_page_size(user_data->dlg.v_scrollbar_adjustment)) )
  1167. user_data->dlg.first_item = (guint32)user_data->dlg.selected_item-(guint32)gtk_adjustment_get_page_size(user_data->dlg.v_scrollbar_adjustment);
  1168. } else if (event->keyval == GDK_Left) {
  1169. if (user_data->dlg.first_node == 0) return TRUE;
  1170. user_data->dlg.first_node--;
  1171. } else return TRUE;
  1172. user_data->dlg.needs_redraw = TRUE;
  1173. dialog_graph_draw(user_data);
  1174. cf_goto_frame(&cfile, user_data->dlg.items[user_data->dlg.selected_item-user_data->dlg.first_item].fd->num);
  1175. return TRUE;
  1176. }
  1177. /****************************************************************************/
  1178. static gboolean draw_area_draw(GtkWidget *widget, cairo_t *cr, gpointer data)
  1179. {
  1180. graph_analysis_data_t *user_data = (graph_analysis_data_t *)data;
  1181. GtkAllocation allocation;
  1182. gtk_widget_get_allocation (widget, &allocation);
  1183. #if GTK_CHECK_VERSION(2,22,0)
  1184. cairo_set_source_surface (cr, user_data->dlg.surface_main, 0, 0);
  1185. #else
  1186. gdk_cairo_set_source_pixmap (cr, user_data->dlg.pixmap_main, 0, 0);
  1187. #endif
  1188. cairo_rectangle (cr, 0, 0, allocation.width, allocation.height);
  1189. cairo_fill (cr);
  1190. return FALSE;
  1191. }
  1192. #if !GTK_CHECK_VERSION(3,0,0)
  1193. /****************************************************************************/
  1194. static gboolean expose_event(GtkWidget *widget, GdkEventExpose *event _U_, gpointer data)
  1195. {
  1196. GtkAllocation allocation;
  1197. graph_analysis_data_t *user_data = (graph_analysis_data_t *)data;
  1198. cairo_t *cr = gdk_cairo_create (gtk_widget_get_window(widget));
  1199. if (gtk_widget_is_drawable(widget)) {
  1200. gtk_widget_get_allocation (widget, &allocation);
  1201. #if GTK_CHECK_VERSION(2,22,0)
  1202. cairo_set_source_surface (cr, user_data->dlg.surface_main, 0, 0);
  1203. #else
  1204. gdk_cairo_set_source_pixmap (cr, user_data->dlg.pixmap_main, 0, 0);
  1205. #endif
  1206. cairo_rectangle (cr, 0, 0, allocation.width, allocation.height);
  1207. cairo_fill (cr);
  1208. cairo_destroy (cr);
  1209. }
  1210. return FALSE;
  1211. }
  1212. #endif
  1213. static gboolean
  1214. draw_area_scrolled(GtkAdjustment *adjustment _U_, gpointer data)
  1215. {
  1216. graph_analysis_data_t *user_data = (graph_analysis_data_t *)data;
  1217. cairo_t *cr = gdk_cairo_create (gtk_widget_get_window(user_data->dlg.draw_area));
  1218. draw_area_draw(user_data->dlg.draw_area, cr, data);
  1219. return TRUE;
  1220. }
  1221. /****************************************************************************/
  1222. static gboolean
  1223. mouse_scrolled(GtkWidget *widget _U_, GdkEventScroll *event, gpointer data)
  1224. {
  1225. graph_analysis_data_t *user_data = (graph_analysis_data_t *)data;
  1226. GtkWidget *scroll_window;
  1227. GtkAdjustment *adjustment;
  1228. gdouble v_scroll_items;
  1229. GtkAllocation draw_area_alloc, scroll_window_alloc;
  1230. if (event->direction == GDK_SCROLL_UP) {
  1231. adjustment = user_data->dlg.v_scrollbar_adjustment;
  1232. v_scroll_items = pow(gtk_adjustment_get_page_size(adjustment), 2.0/3.0);
  1233. if ((gtk_adjustment_get_value(adjustment) - v_scroll_items) <= 0.0) {
  1234. gtk_adjustment_set_value(adjustment, 0.0);
  1235. }
  1236. else {
  1237. gtk_adjustment_set_value(adjustment, gtk_adjustment_get_value(adjustment) - v_scroll_items);
  1238. }
  1239. }
  1240. else if (event->direction == GDK_SCROLL_DOWN) {
  1241. adjustment = user_data->dlg.v_scrollbar_adjustment;
  1242. v_scroll_items = pow (gtk_adjustment_get_page_size(adjustment), 2.0/3.0);
  1243. if ((gtk_adjustment_get_value(adjustment) + v_scroll_items) >= (user_data->num_items - gtk_adjustment_get_page_size(adjustment) - 1)) {
  1244. gtk_adjustment_set_value(adjustment, (user_data->num_items - gtk_adjustment_get_page_size(adjustment) - 1));
  1245. }
  1246. else {
  1247. gtk_adjustment_set_value(adjustment, gtk_adjustment_get_value(adjustment) + v_scroll_items);
  1248. }
  1249. }
  1250. else if (event->direction == GDK_SCROLL_LEFT) {
  1251. if (widget == user_data->dlg.scroll_window) {
  1252. scroll_window = user_data->dlg.scroll_window;
  1253. }
  1254. else if (widget == user_data->dlg.scroll_window_comments) {
  1255. scroll_window = user_data->dlg.scroll_window_comments;
  1256. }
  1257. else return TRUE;
  1258. adjustment = gtk_scrolled_window_get_hadjustment(GTK_SCROLLED_WINDOW(scroll_window));
  1259. if ((gtk_adjustment_get_value(adjustment) - gtk_adjustment_get_step_increment(adjustment)) <= 0.0) {
  1260. gtk_adjustment_set_value(adjustment, 0.0);
  1261. }
  1262. else {
  1263. gtk_adjustment_set_value(adjustment, gtk_adjustment_get_value(adjustment) - gtk_adjustment_get_step_increment(adjustment));
  1264. }
  1265. gtk_scrolled_window_set_hadjustment(GTK_SCROLLED_WINDOW(scroll_window), adjustment);
  1266. }
  1267. else if (event->direction == GDK_SCROLL_RIGHT) {
  1268. if (widget == user_data->dlg.scroll_window) {
  1269. scroll_window = user_data->dlg.scroll_window;
  1270. gtk_widget_get_allocation(user_data->dlg.draw_area, &draw_area_alloc);
  1271. gtk_widget_get_allocation(user_data->dlg.scroll_window, &scroll_window_alloc);
  1272. }
  1273. else if (widget == user_data->dlg.scroll_window_comments) {
  1274. scroll_window = user_data->dlg.scroll_window_comments;
  1275. gtk_widget_get_allocation(user_data->dlg.draw_area_comments, &draw_area_alloc);
  1276. gtk_widget_get_allocation(user_data->dlg.scroll_window_comments, &scroll_window_alloc);
  1277. }
  1278. else return TRUE;
  1279. adjustment = gtk_scrolled_window_get_hadjustment(GTK_SCROLLED_WINDOW(scroll_window));
  1280. if ((gtk_adjustment_get_value(adjustment) + gtk_adjustment_get_step_increment(adjustment)) >= (draw_area_alloc.width - scroll_window_alloc.width)) {
  1281. gtk_adjustment_set_value(adjustment, (draw_area_alloc.width - scroll_window_alloc.width));
  1282. }
  1283. else {
  1284. gtk_adjustment_set_value(adjustment, gtk_adjustment_get_value(adjustment) + gtk_adjustment_get_step_increment(adjustment

Large files files are truncated, but you can click here to view the full file