/* * Copyright (c) 2003-2008 by FlashCode * See README for License detail, AUTHORS for developers list. * * This program is free software; you can redistribute it and/or modify * it under the terms of the GNU General Public License as published by * the Free Software Foundation; either version 3 of the License, or * (at your option) any later version. * * This program is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the * GNU General Public License for more details. * * You should have received a copy of the GNU General Public License * along with this program. If not, see . */ /* gui-gtk-window.c: window display functions for Gtk GUI */ #ifdef HAVE_CONFIG_H #include "config.h" #endif #include #include #include "../../core/weechat.h" #include "../../core/wee-config.h" #include "../../core/wee-log.h" #include "../gui-window.h" #include "../gui-bar.h" #include "../gui-buffer.h" #include "../gui-chat.h" #include "../gui-hotlist.h" #include "../gui-nicklist.h" #include "../gui-main.h" #include "../gui-status.h" #include "gui-gtk.h" /* * gui_window_get_width: get screen width */ int gui_window_get_width () { return 0; } /* * gui_window_get_height: get screen height */ int gui_window_get_height () { return 0; } /* * gui_window_objects_init: init Gtk widgets */ int gui_window_objects_init (struct t_gui_window *window) { struct t_gui_gtk_objects *new_objects; if ((new_objects = malloc (sizeof (*new_objects)))) { window->gui_objects = new_objects; GUI_GTK(window)->textview_chat = NULL; GUI_GTK(window)->textbuffer_chat = NULL; GUI_GTK(window)->texttag_chat = NULL; GUI_GTK(window)->textview_nicklist = NULL; GUI_GTK(window)->textbuffer_nicklist = NULL; GUI_GTK(window)->bar_windows = NULL; GUI_GTK(window)->last_bar_window = NULL; return 1; } return 0; } /* * gui_window_objects_free: free Gtk widgets for a window */ void gui_window_objects_free (struct t_gui_window *window, int free_separator, int free_bar_windows) { /* TODO: write this function for Gtk */ (void) window; (void) free_separator; (void) free_bar_windows; } /* * gui_window_set_weechat_color: set WeeChat color for window */ /* TODO: write this function for Gtk */ /*void gui_window_set_weechat_color (WINDOW *window, int num_color) { if ((num_color >= 0) && (num_color <= GUI_NUM_COLORS - 1)) { wattroff (window, A_BOLD | A_UNDERLINE | A_REVERSE); wattron (window, COLOR_PAIR(gui_color_get_pair (num_color)) | gui_color[num_color]->attributes); } }*/ /* * gui_window_calculate_pos_size: calculate position and size for a window & sub-win */ int gui_window_calculate_pos_size (struct t_gui_window *window, int force_calculate) { /* TODO: write this function for Gtk */ (void) window; (void) force_calculate; return 0; } /* * gui_window_draw_separator: draw window separation */ void gui_window_draw_separator (struct t_gui_window *window) { /* TODO: write this function for Gtk */ /*if (window->win_separator) delwin (window->win_separator); if (window->win_x > 0) { window->win_separator = newwin (window->win_height, 1, window->win_y, window->win_x - 1); gui_window_set_weechat_color (window->win_separator, COLOR_WIN_SEPARATOR); wborder (window->win_separator, ' ', ' ', ' ', ' ', ' ', ' ', ' ', ' '); wnoutrefresh (window->win_separator); refresh (); }*/ (void) window; } /* * gui_window_redraw_buffer: redraw a buffer */ void gui_window_redraw_buffer (struct t_gui_buffer *buffer) { /* TODO: write this function for Gtk */ (void) buffer; } /* * gui_window_redraw_all_buffers: redraw all buffers */ void gui_window_redraw_all_buffers () { /* TODO: write this function for Gtk */ } /* * gui_window_switch: switch to another window */ void gui_window_switch (struct t_gui_window *window) { if (gui_current_window == window) return; /* remove unused bars from current window */ /* ... */ gui_current_window = window; gui_window_switch_to_buffer (gui_current_window, gui_current_window->buffer); gui_window_redraw_buffer (gui_current_window->buffer); } /* * gui_window_switch_to_buffer: switch to another buffer */ void gui_window_switch_to_buffer (struct t_gui_window *window, struct t_gui_buffer *buffer) { GtkTextIter start, end; if (window->buffer->num_displayed > 0) window->buffer->num_displayed--; if (window->buffer != buffer) { window->buffer->last_read_line = window->buffer->last_line; if (buffer->last_read_line == buffer->last_line) buffer->last_read_line = NULL; } window->buffer = buffer; window->win_nick_start = 0; gui_window_calculate_pos_size (window, 1); if (!GUI_GTK(window)->textview_chat) { GUI_GTK(window)->textview_chat = gtk_text_view_new (); gtk_widget_show (GUI_GTK(window)->textview_chat); gtk_container_add (GTK_CONTAINER (gui_gtk_scrolledwindow_chat), GUI_GTK(window)->textview_chat); gtk_widget_set_size_request (GUI_GTK(window)->textview_chat, 300, -1); gtk_text_view_set_editable (GTK_TEXT_VIEW (GUI_GTK(window)->textview_chat), FALSE); gtk_text_view_set_cursor_visible (GTK_TEXT_VIEW (GUI_GTK(window)->textview_chat), FALSE); GUI_GTK(window)->textbuffer_chat = gtk_text_buffer_new (NULL); gtk_text_view_set_buffer (GTK_TEXT_VIEW (GUI_GTK(window)->textview_chat), GUI_GTK(window)->textbuffer_chat); /*GUI_GTK(window)->texttag_chat = gtk_text_buffer_create_tag(GUI_GTK(window)->textbuffer_chat, "courier", "font_family", "lucida");*/ gtk_text_buffer_get_bounds (GUI_GTK(window)->textbuffer_chat, &start, &end); gtk_text_buffer_apply_tag (GUI_GTK(window)->textbuffer_chat, GUI_GTK(window)->texttag_chat, &start, &end); } if (buffer->nicklist && !GUI_GTK(window)->textbuffer_nicklist) { GUI_GTK(window)->textview_nicklist = gtk_text_view_new (); gtk_widget_show (GUI_GTK(window)->textview_nicklist); gtk_container_add (GTK_CONTAINER (gui_gtk_scrolledwindow_nick), GUI_GTK(window)->textview_nicklist); gtk_text_view_set_editable (GTK_TEXT_VIEW (GUI_GTK(window)->textview_nicklist), FALSE); gtk_text_view_set_cursor_visible (GTK_TEXT_VIEW (GUI_GTK(window)->textview_nicklist), FALSE); GUI_GTK(window)->textbuffer_nicklist = gtk_text_buffer_new (NULL); gtk_text_view_set_buffer (GTK_TEXT_VIEW (GUI_GTK(window)->textview_nicklist), GUI_GTK(window)->textbuffer_nicklist); } window->start_line = NULL; window->start_line_pos = 0; buffer->num_displayed++; gui_hotlist_remove_buffer (buffer); } /* * gui_window_page_up: display previous page on buffer */ void gui_window_page_up (struct t_gui_window *window) { if (!gui_ok) return; if (!window->first_line_displayed) { gui_chat_calculate_line_diff (window, &window->start_line, &window->start_line_pos, (window->start_line) ? (-1) * (window->win_chat_height - 1) : (-1) * ((window->win_chat_height - 1) * 2)); gui_chat_draw (window->buffer, 0); gui_status_refresh_needed = 1; } } /* * gui_window_page_down: display next page on buffer */ void gui_window_page_down (struct t_gui_window *window) { struct t_gui_line *ptr_line; int line_pos; if (!gui_ok) return; if (window->start_line) { gui_chat_calculate_line_diff (window, &window->start_line, &window->start_line_pos, window->win_chat_height - 1); /* check if we can display all */ ptr_line = window->start_line; line_pos = window->start_line_pos; gui_chat_calculate_line_diff (window, &ptr_line, &line_pos, window->win_chat_height - 1); if (!ptr_line) { window->start_line = NULL; window->start_line_pos = 0; } gui_chat_draw (window->buffer, 0); gui_status_refresh_needed = 1; } } /* * gui_window_scroll_up: display previous few lines in buffer */ void gui_window_scroll_up (struct t_gui_window *window) { if (!gui_ok) return; if (!window->first_line_displayed) { gui_chat_calculate_line_diff (window, &window->start_line, &window->start_line_pos, (window->start_line) ? (-1) * CONFIG_INTEGER(config_look_scroll_amount) : (-1) * ( (window->win_chat_height - 1) + CONFIG_INTEGER(config_look_scroll_amount))); gui_chat_draw (window->buffer, 0); gui_status_refresh_needed = 1; } } /* * gui_window_scroll_down: display next few lines in buffer */ void gui_window_scroll_down (struct t_gui_window *window) { struct t_gui_line *ptr_line; int line_pos; if (!gui_ok) return; if (window->start_line) { gui_chat_calculate_line_diff (window, &window->start_line, &window->start_line_pos, CONFIG_INTEGER(config_look_scroll_amount)); /* check if we can display all */ ptr_line = window->start_line; line_pos = window->start_line_pos; gui_chat_calculate_line_diff (window, &ptr_line, &line_pos, window->win_chat_height - 1); if (!ptr_line) { window->start_line = NULL; window->start_line_pos = 0; } gui_chat_draw (window->buffer, 0); gui_status_refresh_needed = 1; } } /* * gui_window_scroll_top: scroll to top of buffer */ void gui_window_scroll_top (struct t_gui_window *window) { if (!gui_ok) return; if (!window->first_line_displayed) { window->start_line = window->buffer->lines; window->start_line_pos = 0; gui_chat_draw (window->buffer, 0); gui_status_refresh_needed = 1; } } /* * gui_window_scroll_bottom: scroll to bottom of buffer */ void gui_window_scroll_bottom (struct t_gui_window *window) { if (!gui_ok) return; if (window->start_line) { window->start_line = NULL; window->start_line_pos = 0; gui_chat_draw (window->buffer, 0); gui_status_refresh_needed = 1; } } /* * gui_window_scroll_topic_left: scroll left topic */ void gui_window_scroll_topic_left (struct t_gui_window *window) { if (!gui_ok) return; if (window->win_title_start > 0) window->win_title_start -= (window->win_width * 3) / 4; if (window->win_title_start < 0) window->win_title_start = 0; gui_chat_draw_title (window->buffer, 1); } /* * gui_window_scroll_topic_right: scroll right topic */ void gui_window_scroll_topic_right (struct t_gui_window *window) { if (!gui_ok) return; window->win_title_start += (window->win_width * 3) / 4; gui_chat_draw_title (window->buffer, 1); } /* * gui_window_nicklist_page_up: scroll one page up in nicklist */ void gui_window_nicklist_page_up (struct t_gui_window *window) { if (!gui_ok) return; if (window->buffer->nicklist) { if (window->win_nick_start > 0) { window->win_nick_start -= (window->win_nick_height - 1); if (window->win_nick_start <= 1) window->win_nick_start = 0; gui_nicklist_draw (window->buffer, 1); } } } /* * gui_window_nicklist_page_down: scroll one page down in nicklist */ void gui_window_nicklist_page_down (struct t_gui_window *window) { if (!gui_ok) return; if (window->buffer->nicklist) { if ((window->buffer->nicklist_visible_count > window->win_nick_height) && (window->win_nick_start + window->win_nick_height - 1 < window->buffer->nicklist_visible_count)) { if (window->win_nick_start == 0) window->win_nick_start += (window->win_nick_height - 1); else window->win_nick_start += (window->win_nick_height - 2); gui_nicklist_draw (window->buffer, 1); } } } /* * gui_window_nicklist_beginning: go to beginning of nicklist */ void gui_window_nicklist_beginning (struct t_gui_window *window) { if (!gui_ok) return; if (window->buffer->nicklist) { if (window->win_nick_start > 0) { window->win_nick_start = 0; gui_nicklist_draw (window->buffer, 1); } } } /* * gui_window_nicklist_end: go to the end of nicklist */ void gui_window_nicklist_end (struct t_gui_window *window) { int new_start; if (!gui_ok) return; if (window->buffer->nicklist) { new_start = window->buffer->nicklist_visible_count - window->win_nick_height; if (new_start < 0) new_start = 0; else if (new_start >= 1) new_start++; if (new_start != window->win_nick_start) { window->win_nick_start = new_start; gui_nicklist_draw (window->buffer, 1); } } } /* * gui_window_auto_resize: auto-resize all windows, according to % of global size * This function is called after a terminal resize. * Returns 0 if ok, -1 if all window should be merged * (not enough space according to windows %) */ int gui_window_auto_resize (struct t_gui_window_tree *tree, int x, int y, int width, int height, int simulate) { int size1, size2; if (tree) { if (tree->window) { if (!simulate) { tree->window->win_x = x; tree->window->win_y = y; tree->window->win_width = width; tree->window->win_height = height; } } else { if (tree->split_horiz) { size1 = (height * tree->split_pct) / 100; size2 = height - size1; if (gui_window_auto_resize (tree->child1, x, y + size1, width, size2, simulate) < 0) return -1; if (gui_window_auto_resize (tree->child2, x, y, width, size1, simulate) < 0) return -1; } else { size1 = (width * tree->split_pct) / 100; size2 = width - size1 - 1; if (gui_window_auto_resize (tree->child1, x, y, size1, height, simulate) < 0) return -1; if (gui_window_auto_resize (tree->child2, x + size1 + 1, y, size2, height, simulate) < 0) return -1; } } } return 0; } /* * gui_window_refresh_windows: auto resize and refresh all windows */ void gui_window_refresh_windows () { /*struct t_gui_window *ptr_win, *old_current_window;*/ if (gui_ok) { /* TODO: write this function for Gtk */ } } /* * gui_window_split_horiz: split a window horizontally */ void gui_window_split_horiz (struct t_gui_window *window, int pourcentage) { struct t_gui_window *new_window; int height1, height2; if (!gui_ok) return; height1 = (window->win_height * pourcentage) / 100; height2 = window->win_height - height1; if ((pourcentage > 0) && (pourcentage <= 100)) { if ((new_window = gui_window_new (window, window->win_x, window->win_y, window->win_width, height1, 100, pourcentage))) { /* reduce old window height (bottom window) */ window->win_y = new_window->win_y + new_window->win_height; window->win_height = height2; window->win_height_pct = 100 - pourcentage; /* assign same buffer for new window (top window) */ new_window->buffer = window->buffer; new_window->buffer->num_displayed++; gui_window_switch_to_buffer (window, window->buffer); gui_current_window = new_window; gui_window_switch_to_buffer (gui_current_window, gui_current_window->buffer); gui_window_redraw_buffer (gui_current_window->buffer); } } } /* * gui_window_split_vertic: split a window vertically */ void gui_window_split_vertic (struct t_gui_window *window, int pourcentage) { struct t_gui_window *new_window; int width1, width2; if (!gui_ok) return; width1 = (window->win_width * pourcentage) / 100; width2 = window->win_width - width1 - 1; if ((pourcentage > 0) && (pourcentage <= 100)) { if ((new_window = gui_window_new (window, window->win_x + width1 + 1, window->win_y, width2, window->win_height, pourcentage, 100))) { /* reduce old window height (left window) */ window->win_width = width1; window->win_width_pct = 100 - pourcentage; /* assign same buffer for new window (right window) */ new_window->buffer = window->buffer; new_window->buffer->num_displayed++; gui_window_switch_to_buffer (window, window->buffer); gui_current_window = new_window; gui_window_switch_to_buffer (gui_current_window, gui_current_window->buffer); gui_window_redraw_buffer (gui_current_window->buffer); /* create & draw separator */ gui_window_draw_separator (gui_current_window); } } } /* * gui_window_resize: resize window */ void gui_window_resize (struct t_gui_window *window, int pourcentage) { /* TODO: write this function for Gtk */ (void) window; (void) pourcentage; } /* * gui_window_merge: merge window with its sister */ int gui_window_merge (struct t_gui_window *window) { struct t_gui_window_tree *parent, *sister; parent = window->ptr_tree->parent_node; if (parent) { sister = (parent->child1->window == window) ? parent->child2 : parent->child1; if (!(sister->window)) return 0; if (window->win_y == sister->window->win_y) { /* horizontal merge */ window->win_width += sister->window->win_width + 1; window->win_width_pct += sister->window->win_width_pct; } else { /* vertical merge */ window->win_height += sister->window->win_height; window->win_height_pct += sister->window->win_height_pct; } if (sister->window->win_x < window->win_x) window->win_x = sister->window->win_x; if (sister->window->win_y < window->win_y) window->win_y = sister->window->win_y; gui_window_free (sister->window); gui_window_tree_node_to_leaf (parent, window); gui_window_switch_to_buffer (window, window->buffer); gui_window_redraw_buffer (window->buffer); return 1; } return 0; } /* * gui_window_merge_all: merge all windows into only one */ void gui_window_merge_all (struct t_gui_window *window) { /* TODO: write this function for Gtk */ (void) window; } /* * gui_window_side_by_side: return a code about position of 2 windows: * 0 = they're not side by side * 1 = side by side (win2 is over the win1) * 2 = side by side (win2 on the right) * 3 = side by side (win2 below win1) * 4 = side by side (win2 on the left) */ int gui_window_side_by_side (struct t_gui_window *win1, struct t_gui_window *win2) { /* win2 over win1 ? */ if (win2->win_y + win2->win_height == win1->win_y) { if (win2->win_x >= win1->win_x + win1->win_width) return 0; if (win2->win_x + win2->win_width <= win1->win_x) return 0; return 1; } /* win2 on the right ? */ if (win2->win_x == win1->win_x + win1->win_width + 1) { if (win2->win_y >= win1->win_y + win1->win_height) return 0; if (win2->win_y + win2->win_height <= win1->win_y) return 0; return 2; } /* win2 below win1 ? */ if (win2->win_y == win1->win_y + win1->win_height) { if (win2->win_x >= win1->win_x + win1->win_width) return 0; if (win2->win_x + win2->win_width <= win1->win_x) return 0; return 3; } /* win2 on the left ? */ if (win2->win_x + win2->win_width + 1 == win1->win_x) { if (win2->win_y >= win1->win_y + win1->win_height) return 0; if (win2->win_y + win2->win_height <= win1->win_y) return 0; return 4; } return 0; } /* * gui_window_switch_up: search and switch to a window over current window */ void gui_window_switch_up (struct t_gui_window *window) { struct t_gui_window *ptr_win; for (ptr_win = gui_windows; ptr_win; ptr_win = ptr_win->next_window) { if ((ptr_win != window) && (gui_window_side_by_side (window, ptr_win) == 1)) { gui_current_window = ptr_win; gui_window_switch_to_buffer (gui_current_window, gui_current_window->buffer); gui_window_redraw_buffer (gui_current_window->buffer); return; } } } /* * gui_window_switch_down: search and switch to a window below current window */ void gui_window_switch_down (struct t_gui_window *window) { struct t_gui_window *ptr_win; for (ptr_win = gui_windows; ptr_win; ptr_win = ptr_win->next_window) { if ((ptr_win != window) && (gui_window_side_by_side (window, ptr_win) == 3)) { gui_current_window = ptr_win; gui_window_switch_to_buffer (gui_current_window, gui_current_window->buffer); gui_window_redraw_buffer (gui_current_window->buffer); return; } } } /* * gui_window_switch_left: search and switch to a window on the left of current window */ void gui_window_switch_left (struct t_gui_window *window) { struct t_gui_window *ptr_win; for (ptr_win = gui_windows; ptr_win; ptr_win = ptr_win->next_window) { if ((ptr_win != window) && (gui_window_side_by_side (window, ptr_win) == 4)) { gui_current_window = ptr_win; gui_window_switch_to_buffer (gui_current_window, gui_current_window->buffer); gui_window_redraw_buffer (gui_current_window->buffer); return; } } } /* * gui_window_switch_right: search and switch to a window on the right of current window */ void gui_window_switch_right (struct t_gui_window *window) { struct t_gui_window *ptr_win; for (ptr_win = gui_windows; ptr_win; ptr_win = ptr_win->next_window) { if ((ptr_win != window) && (gui_window_side_by_side (window, ptr_win) == 2)) { gui_current_window = ptr_win; gui_window_switch_to_buffer (gui_current_window, gui_current_window->buffer); gui_window_redraw_buffer (gui_current_window->buffer); return; } } } /* * gui_window_refresh_screen: called when term size is modified */ void gui_window_refresh_screen () { /* TODO: write this function for Gtk */ } /* * gui_window_title_set: set terminal title */ void gui_window_title_set () { /* TODO: write this function for Gtk */ } /* * gui_window_title_reset: reset terminal title */ void gui_window_title_reset () { /* This function does nothing in Gtk GUI */ } /* * gui_window_objects_print_log: print Gtk objects infos in log * (usually for crash dump) */ void gui_window_objects_print_log (struct t_gui_window *window) { struct t_gui_bar_window *ptr_bar_win; log_printf (" textview_chat . . . : 0x%x", GUI_GTK(window)->textview_chat); log_printf (" textbuffer_chat . . : 0x%x", GUI_GTK(window)->textbuffer_chat); log_printf (" texttag_chat. . . . : 0x%x", GUI_GTK(window)->texttag_chat); log_printf (" textview_nicklist . : 0x%x", GUI_GTK(window)->textview_nicklist); log_printf (" textbuffer_nicklist : 0x%x", GUI_GTK(window)->textbuffer_nicklist); log_printf (" bar_windows . . . . : 0x%x", GUI_GTK(window)->bar_windows); log_printf (" last_bar_windows. . : 0x%x", GUI_GTK(window)->last_bar_window); log_printf (" current_style_fg. . : %d", GUI_GTK(window)->current_style_fg); log_printf (" current_style_bg. . : %d", GUI_GTK(window)->current_style_bg); log_printf (" current_style_attr. : %d", GUI_GTK(window)->current_style_attr); log_printf (" current_color_attr. : %d", GUI_GTK(window)->current_color_attr); for (ptr_bar_win = GUI_GTK(window)->bar_windows; ptr_bar_win; ptr_bar_win = ptr_bar_win->next_bar_window) { log_printf (""); log_printf (" [window bar (addr:0x%x)]", ptr_bar_win); log_printf (" bar . . . . . . . : 0x%x", ptr_bar_win->bar); log_printf (" x . . . . . . . . : %d", ptr_bar_win->x); log_printf (" y . . . . . . . . : %d", ptr_bar_win->y); log_printf (" width . . . . . . : %d", ptr_bar_win->width); log_printf (" height. . . . . . : %d", ptr_bar_win->height); } }