From 113486564fc1a5fd60013385a1c4f1ab1c6904a8 Mon Sep 17 00:00:00 2001 From: Timo Sirainen Date: Sat, 20 Oct 2001 10:01:29 +0000 Subject: Fixed restoring split windows layout. Some other split window resizing fixes git-svn-id: http://svn.irssi.org/repos/irssi/trunk@1854 dbcabf3a-b0e7-0310-adc4-f8d773084564 --- src/fe-text/mainwindows-layout.c | 71 ++++++++++++++++++--- src/fe-text/mainwindows.c | 132 ++++++++++++++++++++++++++------------- src/fe-text/mainwindows.h | 7 ++- 3 files changed, 159 insertions(+), 51 deletions(-) diff --git a/src/fe-text/mainwindows-layout.c b/src/fe-text/mainwindows-layout.c index 78eebef3..0e6af783 100644 --- a/src/fe-text/mainwindows-layout.c +++ b/src/fe-text/mainwindows-layout.c @@ -113,16 +113,70 @@ static void sig_layout_restore(void) WINDOW_REC *window; CONFIG_NODE *node; GSList *tmp, *sorted_config; - int lower_size; + int avail_height, height, *heights; + int i, lower_size, windows_count, diff; node = iconfig_node_traverse("mainwindows", FALSE); if (node == NULL) return; + sorted_config = get_sorted_windows_config(node); + windows_count = g_slist_length(sorted_config); + + /* calculate the saved terminal height */ + avail_height = screen_height - + screen_reserved_top - screen_reserved_bottom; + height = 0; + heights = g_new0(int, windows_count); + for (i = 0, tmp = sorted_config; tmp != NULL; tmp = tmp->next, i++) { + CONFIG_NODE *node = tmp->data; + + heights[i] = config_node_get_int(node, "lines", 0); + height += heights[i]; + } + + if (avail_height <= (WINDOW_MIN_SIZE*2)+1) { + /* we can fit only one window to screen - + give it all the height we can */ + windows_count = 1; + heights[0] = avail_height; + } else if (height != avail_height) { + /* Terminal's height is different from the saved one. + Resize the windows so they fit to screen. */ + while (height > avail_height && + windows_count*(WINDOW_MIN_SIZE+1) > avail_height) { + /* all windows can't fit into screen, + remove the lowest ones */ + windows_count--; + } + + /* try to keep the windows' size about the same in percents */ + for (i = 0; i < windows_count; i++) { + int size = avail_height*heights[i]/height; + if (size < WINDOW_MIN_SIZE+1) + size = WINDOW_MIN_SIZE+1; + heights[i] = size; + } + + /* give/remove the last bits */ + height = 0; + for (i = 0; i < windows_count; i++) + height += heights[i]; + + diff = height < avail_height ? 1 : -1; + for (i = 0; height != avail_height; i++) { + if (i == windows_count) + i = 0; + + if (heights[i] > WINDOW_MIN_SIZE+1) { + height += diff; + heights[i] += diff; + } + } + } + /* create all the visible windows with correct size */ lower_window = NULL; lower_size = 0; - - sorted_config = get_sorted_windows_config(node); - for (tmp = sorted_config; tmp != NULL; tmp = tmp->next) { + for (i = 0, tmp = sorted_config; i < windows_count; tmp = tmp->next, i++) { CONFIG_NODE *node = tmp->data; /* create a new window + mainwindow */ @@ -133,18 +187,21 @@ static void sig_layout_restore(void) window_set_refnum(window, atoi(node->key)); if (lower_size > 0) - mainwindow_set_size(lower_window, lower_size); + mainwindow_set_size(lower_window, lower_size, FALSE); window_set_active(window); active_mainwin = WINDOW_MAIN(window); lower_window = WINDOW_MAIN(window); - lower_size = config_node_get_int(node, "lines", 0); + lower_size = heights[i]; + if (lower_size < WINDOW_MIN_SIZE+1) + lower_size = WINDOW_MIN_SIZE+1; } g_slist_free(sorted_config); + g_free(heights); if (lower_size > 0) - mainwindow_set_size(lower_window, lower_size); + mainwindow_set_size(lower_window, lower_size, FALSE); } static void sig_layout_reset(void) diff --git a/src/fe-text/mainwindows.c b/src/fe-text/mainwindows.c index e1d3f097..603f6226 100644 --- a/src/fe-text/mainwindows.c +++ b/src/fe-text/mainwindows.c @@ -35,7 +35,7 @@ GSList *mainwindows; MAIN_WINDOW_REC *active_mainwin; -static int reserved_top, reserved_bottom; +int screen_reserved_top, screen_reserved_bottom; static int old_screen_width, old_screen_height; #define mainwindow_create_screen(window) \ @@ -61,7 +61,7 @@ static MAIN_WINDOW_REC *find_window_with_room(void) for (tmp = mainwindows; tmp != NULL; tmp = tmp->next) { MAIN_WINDOW_REC *rec = tmp->data; - space = rec->height; + space = MAIN_WINDOW_TEXT_HEIGHT(rec); if (space >= WINDOW_MIN_SIZE+NEW_WINDOW_SIZE && space > biggest) { biggest = space; biggest_rec = rec; @@ -178,12 +178,13 @@ MAIN_WINDOW_REC *mainwindow_create(void) if (mainwindows == NULL) { active_mainwin = rec; - rec->first_line = reserved_top; - rec->last_line = screen_height-1 - reserved_bottom; + rec->first_line = screen_reserved_top; + rec->last_line = screen_height-1 - screen_reserved_bottom; rec->height = rec->last_line-rec->first_line+1; } else { parent = WINDOW_MAIN(active_win); - if (parent->height < WINDOW_MIN_SIZE+NEW_WINDOW_SIZE) + if (MAIN_WINDOW_TEXT_HEIGHT(parent) < + WINDOW_MIN_SIZE+NEW_WINDOW_SIZE) parent = find_window_with_room(); if (parent == NULL) return NULL; /* not enough space */ @@ -376,7 +377,7 @@ static void mainwindows_resize_smaller(int xdiff, int ydiff) for (tmp = mainwindows; tmp != NULL; tmp = tmp->next) { MAIN_WINDOW_REC *rec = tmp->data; - space += rec->height-WINDOW_MIN_SIZE; + space += MAIN_WINDOW_TEXT_HEIGHT(rec)-WINDOW_MIN_SIZE; } if (space < -ydiff) { @@ -390,7 +391,7 @@ static void mainwindows_resize_smaller(int xdiff, int ydiff) for (tmp = sorted; tmp != NULL && ydiff < 0; tmp = tmp->next) { MAIN_WINDOW_REC *rec = tmp->data; - space = rec->height-WINDOW_MIN_SIZE; + space = MAIN_WINDOW_TEXT_HEIGHT(rec)-WINDOW_MIN_SIZE; if (space <= 0) { rec->first_line += ydiff; rec->last_line += ydiff; @@ -419,7 +420,7 @@ static void mainwindows_resize_bigger(int xdiff, int ydiff) for (tmp = sorted; tmp != NULL; tmp = tmp->next) { MAIN_WINDOW_REC *rec = tmp->data; - space = rec->height-WINDOW_MIN_SIZE; + space = MAIN_WINDOW_TEXT_HEIGHT(rec)-WINDOW_MIN_SIZE; if (ydiff == 0 || (space >= 0 && tmp->next != NULL)) { if (moved > 0) { rec->first_line += moved; @@ -488,10 +489,10 @@ int mainwindows_reserve_lines(int top, int bottom) ret = -1; if (top != 0) { - g_return_val_if_fail(top > 0 || reserved_top > top, -1); + g_return_val_if_fail(top > 0 || screen_reserved_top > top, -1); - ret = reserved_top; - reserved_top += top; + ret = screen_reserved_top; + screen_reserved_top += top; window = mainwindows_find_lower(-1); if (window != NULL) { @@ -501,10 +502,10 @@ int mainwindows_reserve_lines(int top, int bottom) } if (bottom != 0) { - g_return_val_if_fail(bottom > 0 || reserved_bottom > bottom, -1); + g_return_val_if_fail(bottom > 0 || screen_reserved_bottom > bottom, -1); - ret = reserved_bottom; - reserved_bottom += bottom; + ret = screen_reserved_bottom; + screen_reserved_bottom += bottom; window = mainwindows_find_upper(screen_height); if (window != NULL) { @@ -551,60 +552,104 @@ static void mainwindows_resize_two(MAIN_WINDOW_REC *grow_win, gui_window_redraw(shrink_win->active); } -static int mainwindow_grow(MAIN_WINDOW_REC *window, int count) +static int try_shrink_lower(MAIN_WINDOW_REC *window, int count) { MAIN_WINDOW_REC *shrink_win; - /* shrink lower window */ shrink_win = mainwindows_find_lower(window->last_line); - if (shrink_win != NULL && shrink_win->height-count >= WINDOW_MIN_SIZE) { + if (shrink_win != NULL && + MAIN_WINDOW_TEXT_HEIGHT(shrink_win)-count >= WINDOW_MIN_SIZE) { window->last_line += count; shrink_win->first_line += count; - } else { - /* shrink upper window */ - shrink_win = mainwindows_find_upper(window->first_line); - if (shrink_win == NULL || - shrink_win->height-count < WINDOW_MIN_SIZE) - return FALSE; + mainwindows_resize_two(window, shrink_win, count); + return TRUE; + } + + return FALSE; +} +static int try_shrink_upper(MAIN_WINDOW_REC *window, int count) +{ + MAIN_WINDOW_REC *shrink_win; + + shrink_win = mainwindows_find_upper(window->first_line); + if (shrink_win != NULL && + MAIN_WINDOW_TEXT_HEIGHT(shrink_win)-count >= WINDOW_MIN_SIZE) { window->first_line -= count; shrink_win->last_line -= count; + mainwindows_resize_two(window, shrink_win, count); + return TRUE; + } + + return FALSE; +} + +static int mainwindow_grow(MAIN_WINDOW_REC *window, int count, + int resize_lower) +{ + if (!resize_lower || !try_shrink_lower(window, count)) { + if (!try_shrink_upper(window, count)) { + if (resize_lower || !try_shrink_lower(window, count)) + return FALSE; + } } - mainwindows_resize_two(window, shrink_win, count); return TRUE; } -static int mainwindow_shrink(MAIN_WINDOW_REC *window, int count) +static int try_grow_lower(MAIN_WINDOW_REC *window, int count) { MAIN_WINDOW_REC *grow_win; - if (window->height-count < WINDOW_MIN_SIZE) - return FALSE; - grow_win = mainwindows_find_lower(window->last_line); if (grow_win != NULL) { window->last_line -= count; grow_win->first_line -= count; - } else { - grow_win = mainwindows_find_upper(window->first_line); - if (grow_win == NULL) return FALSE; + mainwindows_resize_two(grow_win, window, count); + } + + return grow_win != NULL; +} + +static int try_grow_upper(MAIN_WINDOW_REC *window, int count) +{ + MAIN_WINDOW_REC *grow_win; + grow_win = mainwindows_find_upper(window->first_line); + if (grow_win != NULL) { window->first_line += count; grow_win->last_line += count; + mainwindows_resize_two(grow_win, window, count); + } + + return grow_win != NULL; +} + +static int mainwindow_shrink(MAIN_WINDOW_REC *window, int count, int resize_lower) +{ + if (MAIN_WINDOW_TEXT_HEIGHT(window)-count < WINDOW_MIN_SIZE) + return FALSE; + + if (!resize_lower || !try_grow_lower(window, count)) { + if (!try_grow_upper(window, count)) { + if (resize_lower || !try_grow_lower(window, count)) + return FALSE; + } } - mainwindows_resize_two(grow_win, window, count); return TRUE; } -void mainwindow_set_size(MAIN_WINDOW_REC *window, int size) +/* Change the window height - the height includes the lines needed for + statusbars. If resize_lower is TRUE, the lower window is first tried + to be resized instead of upper window. */ +void mainwindow_set_size(MAIN_WINDOW_REC *window, int height, int resize_lower) { - size -= window->height; - if (size < 0) - mainwindow_shrink(window, -size); + height -= window->height; + if (height < 0) + mainwindow_shrink(window, -height, resize_lower); else - mainwindow_grow(window, size); + mainwindow_grow(window, height, resize_lower); } /* SYNTAX: WINDOW GROW [] */ @@ -616,7 +661,7 @@ static void cmd_window_grow(const char *data) count = *data == '\0' ? 1 : atoi(data); window = WINDOW_MAIN(active_win); - if (!mainwindow_grow(window, count)) { + if (!mainwindow_grow(window, count, FALSE)) { printformat_window(active_win, MSGLEVEL_CLIENTNOTICE, TXT_WINDOW_TOO_SMALL); } @@ -628,7 +673,7 @@ static void cmd_window_shrink(const char *data) int count; count = *data == '\0' ? 1 : atoi(data); - if (!mainwindow_shrink(WINDOW_MAIN(active_win), count)) { + if (!mainwindow_shrink(WINDOW_MAIN(active_win), count, FALSE)) { printformat_window(active_win, MSGLEVEL_CLIENTNOTICE, TXT_WINDOW_TOO_SMALL); } @@ -643,7 +688,8 @@ static void cmd_window_size(const char *data) if (!is_numeric(data, 0)) return; size = atoi(data); - size -= WINDOW_MAIN(active_win)->height; + size -= WINDOW_MAIN(active_win)->height - + WINDOW_MAIN(active_win)->statusbar_lines; if (size == 0) return; ltoa(sizestr, size < 0 ? -size : size); @@ -663,12 +709,12 @@ static void cmd_window_balance(void) windows = g_slist_length(mainwindows); if (windows == 1) return; - avail_size = screen_height - reserved_top-reserved_bottom; + avail_size = screen_height - screen_reserved_top-screen_reserved_bottom; unit_size = avail_size/windows; bigger_units = avail_size%windows; sorted = mainwindows_get_sorted(FALSE); - last_line = reserved_top; + last_line = screen_reserved_top; for (tmp = sorted; tmp != NULL; tmp = tmp->next) { MAIN_WINDOW_REC *rec = tmp->data; @@ -995,7 +1041,7 @@ void mainwindows_init(void) mainwindows = NULL; active_mainwin = NULL; - reserved_top = reserved_bottom = 0; + screen_reserved_top = screen_reserved_bottom = 0; command_bind("window grow", NULL, (SIGNAL_FUNC) cmd_window_grow); command_bind("window shrink", NULL, (SIGNAL_FUNC) cmd_window_shrink); diff --git a/src/fe-text/mainwindows.h b/src/fe-text/mainwindows.h index 514677b5..6850fe09 100644 --- a/src/fe-text/mainwindows.h +++ b/src/fe-text/mainwindows.h @@ -26,6 +26,7 @@ typedef struct { extern GSList *mainwindows; extern MAIN_WINDOW_REC *active_mainwin; +extern int screen_reserved_top, screen_reserved_bottom; void mainwindows_init(void); void mainwindows_deinit(void); @@ -36,7 +37,11 @@ void mainwindow_destroy(MAIN_WINDOW_REC *window); void mainwindows_redraw(void); void mainwindows_recreate(void); -void mainwindow_set_size(MAIN_WINDOW_REC *window, int size); +/* Change the window height - the height includes the lines needed for + statusbars. If resize_lower is TRUE, the lower window is first tried + to be resized instead of upper window. */ +void mainwindow_set_size(MAIN_WINDOW_REC *window, int height, + int resize_lower); void mainwindows_resize(int width, int height); void mainwindow_change_active(MAIN_WINDOW_REC *mainwin, -- cgit v1.2.3