1 /* 2 Widget based utility functions. 3 4 Copyright (C) 1994-2025 5 Free Software Foundation, Inc. 6 7 Authors: 8 Miguel de Icaza, 1994, 1995, 1996 9 Radek Doulik, 1994, 1995 10 Jakub Jelinek, 1995 11 Andrej Borsenkow, 1995 12 Andrew Borodin <aborodin@vmail.ru>, 2009, 2010, 2013 13 14 This file is part of the Midnight Commander. 15 16 The Midnight Commander is free software: you can redistribute it 17 and/or modify it under the terms of the GNU General Public License as 18 published by the Free Software Foundation, either version 3 of the License, 19 or (at your option) any later version. 20 21 The Midnight Commander 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, see <http://www.gnu.org/licenses/>. 28 */ 29 30 /** \file listbox-window.c 31 * \brief Source: Listbox widget, a listbox within dialog window 32 */ 33 34 #include <config.h> 35 36 #include <stdlib.h> 37 38 #include "lib/global.h" 39 #include "lib/tty/tty.h" /* COLS */ 40 #include "lib/skin.h" 41 #include "lib/strutil.h" /* str_term_width1() */ 42 #include "lib/widget.h" 43 44 /*** global variables ****************************************************************************/ 45 46 /*** file scope macro definitions ****************************************************************/ 47 48 /*** file scope type declarations ****************************************************************/ 49 50 /*** file scope variables ************************************************************************/ 51 52 /*** file scope functions ************************************************************************/ 53 54 /* --------------------------------------------------------------------------------------------- */ 55 /*** public functions ****************************************************************************/ 56 /* --------------------------------------------------------------------------------------------- */ 57 58 Listbox * 59 listbox_window_centered_new (int center_y, int center_x, int lines, int cols, /* */ 60 const char *title, const char *help) 61 { 62 const int space = 4; 63 64 int xpos = 0, ypos = 0; 65 Listbox *listbox; 66 widget_pos_flags_t pos_flags = WPOS_TRYUP; 67 68 /* Adjust sizes */ 69 lines = MIN (lines, LINES - 6); 70 71 if (title != NULL) 72 { 73 int len; 74 75 len = str_term_width1 (title) + 4; 76 cols = MAX (cols, len); 77 } 78 79 cols = MIN (cols, COLS - 6); 80 81 /* adjust position */ 82 if ((center_y < 0) || (center_x < 0)) 83 pos_flags |= WPOS_CENTER; 84 else 85 { 86 /* Actually, this this is not used in MC. */ 87 88 ypos = center_y; 89 xpos = center_x; 90 91 ypos -= lines / 2; 92 xpos -= cols / 2; 93 94 if (ypos + lines >= LINES) 95 ypos = LINES - lines - space; 96 if (ypos < 0) 97 ypos = 0; 98 99 if (xpos + cols >= COLS) 100 xpos = COLS - cols - space; 101 if (xpos < 0) 102 xpos = 0; 103 } 104 105 listbox = g_new (Listbox, 1); 106 107 listbox->dlg = 108 dlg_create (TRUE, ypos, xpos, lines + space, cols + space, pos_flags, FALSE, listbox_colors, 109 NULL, NULL, help, title); 110 111 listbox->list = listbox_new (2, 2, lines, cols, FALSE, NULL); 112 group_add_widget (GROUP (listbox->dlg), listbox->list); 113 114 return listbox; 115 } 116 117 /* --------------------------------------------------------------------------------------------- */ 118 119 Listbox * 120 listbox_window_new (int lines, int cols, const char *title, const char *help) /* */ 121 { 122 return listbox_window_centered_new (-1, -1, lines, cols, title, help); 123 } 124 125 /* --------------------------------------------------------------------------------------------- */ 126 127 /** Returns the number of the item selected */ 128 int 129 listbox_run (Listbox *l) /* */ 130 { 131 int val = -1; 132 133 if (dlg_run (l->dlg) != B_CANCEL) 134 val = l->list->current; 135 widget_destroy (WIDGET (l->dlg)); 136 g_free (l); 137 return val; 138 } 139 140 /* --------------------------------------------------------------------------------------------- */ 141 142 /** 143 * A variant of listbox_run() which is more convenient to use when we 144 * need to select arbitrary 'data'. 145 * 146 * @param select the item to select initially, by its 'data'. Optional. 147 * @return the 'data' of the item selected, or NULL if none selected. 148 */ 149 void * 150 listbox_run_with_data (Listbox *l, const void *select) /* */ 151 { 152 void *val = NULL; 153 154 if (select != NULL) 155 listbox_set_current (l->list, listbox_search_data (l->list, select)); 156 157 if (dlg_run (l->dlg) != B_CANCEL) 158 { 159 WLEntry *e; 160 161 e = listbox_get_nth_entry (l->list, l->list->current); 162 if (e != NULL) 163 { 164 /* The assert guards against returning a soon-to-be deallocated 165 * pointer (as in listbox_add_item(..., TRUE)). */ 166 g_assert (!e->free_data); 167 val = e->data; 168 } 169 } 170 171 widget_destroy (WIDGET (l->dlg)); 172 g_free (l); 173 return val; 174 } 175 176 /* --------------------------------------------------------------------------------------------- */