2 Widgets for the Midnight Commander
4 Copyright (C) 1994, 1995, 1996, 1998, 1999, 2000, 2001, 2002, 2003,
5 2004, 2005, 2006, 2007, 2009, 2010, 2011
6 The Free Software Foundation, Inc.
9 Radek Doulik, 1994, 1995
10 Miguel de Icaza, 1994, 1995
12 Andrej Borsenkow, 1996
14 Andrew Borodin <aborodin@vmail.ru>, 2009, 2010
16 This file is part of the Midnight Commander.
18 The Midnight Commander is free software: you can redistribute it
19 and/or modify it under the terms of the GNU General Public License as
20 published by the Free Software Foundation, either version 3 of the License,
21 or (at your option) any later version.
23 The Midnight Commander is distributed in the hope that it will be useful,
24 but WITHOUT ANY WARRANTY; without even the implied warranty of
25 MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
26 GNU General Public License for more details.
28 You should have received a copy of the GNU General Public License
29 along with this program. If not, see <http://www.gnu.org/licenses/>.
33 * \brief Source: WButtonBar widget
41 #include "lib/global.h"
43 #include "lib/tty/tty.h"
44 #include "lib/tty/mouse.h"
45 #include "lib/tty/key.h" /* XCTRL and ALT macros */
47 #include "lib/strutil.h"
49 #include "lib/keybind.h" /* global_keymap_t */
50 #include "lib/widget.h"
52 /*** global variables ****************************************************************************/
54 /*** file scope macro definitions ****************************************************************/
56 /*** file scope type declarations ****************************************************************/
58 /*** file scope variables ************************************************************************/
60 /*** file scope functions ************************************************************************/
61 /* --------------------------------------------------------------------------------------------- */
63 /* calculate positions of buttons; width is never less than 7 */
65 buttonbar_init_button_positions (WButtonBar
* bb
)
70 if (COLS
< BUTTONBAR_LABELS_NUM
* 7)
72 for (i
= 0; i
< BUTTONBAR_LABELS_NUM
; i
++)
77 bb
->labels
[i
].end_coord
= pos
;
82 /* Distribute the extra width in a way that the middle vertical line
83 (between F5 and F6) aligns with the two panels. The extra width
84 is distributed in this order: F10, F5, F9, F4, ..., F6, F1. */
87 dv
= COLS
/ BUTTONBAR_LABELS_NUM
;
88 md
= COLS
% BUTTONBAR_LABELS_NUM
;
90 for (i
= 0; i
< BUTTONBAR_LABELS_NUM
/ 2; i
++)
93 if (BUTTONBAR_LABELS_NUM
/ 2 - 1 - i
< md
/ 2)
96 bb
->labels
[i
].end_coord
= pos
;
99 for (; i
< BUTTONBAR_LABELS_NUM
; i
++)
102 if (BUTTONBAR_LABELS_NUM
- 1 - i
< (md
+ 1) / 2)
105 bb
->labels
[i
].end_coord
= pos
;
110 /* --------------------------------------------------------------------------------------------- */
112 /* return width of one button */
114 buttonbar_get_button_width (const WButtonBar
* bb
, int i
)
117 return bb
->labels
[0].end_coord
;
118 return bb
->labels
[i
].end_coord
- bb
->labels
[i
- 1].end_coord
;
121 /* --------------------------------------------------------------------------------------------- */
124 buttonbar_get_button_by_x_coord (const WButtonBar
* bb
, int x
)
128 for (i
= 0; i
< BUTTONBAR_LABELS_NUM
; i
++)
129 if (bb
->labels
[i
].end_coord
> x
)
135 /* --------------------------------------------------------------------------------------------- */
138 set_label_text (WButtonBar
* bb
, int idx
, const char *text
)
140 g_free (bb
->labels
[idx
- 1].text
);
141 bb
->labels
[idx
- 1].text
= g_strdup (text
);
144 /* --------------------------------------------------------------------------------------------- */
146 /* returns TRUE if a function has been called, FALSE otherwise. */
148 buttonbar_call (WButtonBar
* bb
, int i
)
150 cb_ret_t ret
= MSG_NOT_HANDLED
;
152 if ((bb
!= NULL
) && (bb
->labels
[i
].command
!= CK_IgnoreKey
))
153 ret
= bb
->widget
.owner
->callback (bb
->widget
.owner
,
154 (Widget
*) bb
, DLG_ACTION
,
155 bb
->labels
[i
].command
, bb
->labels
[i
].receiver
);
159 /* --------------------------------------------------------------------------------------------- */
162 buttonbar_callback (Widget
* w
, widget_msg_t msg
, int parm
)
164 WButtonBar
*bb
= (WButtonBar
*) w
;
171 return MSG_NOT_HANDLED
;
174 for (i
= 0; i
< BUTTONBAR_LABELS_NUM
; i
++)
175 if (parm
== KEY_F (i
+ 1) && buttonbar_call (bb
, i
))
177 return MSG_NOT_HANDLED
;
182 buttonbar_init_button_positions (bb
);
183 widget_move (&bb
->widget
, 0, 0);
184 tty_setcolor (DEFAULT_COLOR
);
185 tty_printf ("%-*s", bb
->widget
.cols
, "");
186 widget_move (&bb
->widget
, 0, 0);
188 for (i
= 0; i
< BUTTONBAR_LABELS_NUM
; i
++)
192 width
= buttonbar_get_button_width (bb
, i
);
196 tty_setcolor (BUTTONBAR_HOTKEY_COLOR
);
197 tty_printf ("%2d", i
+ 1);
199 tty_setcolor (BUTTONBAR_BUTTON_COLOR
);
200 text
= (bb
->labels
[i
].text
!= NULL
) ? bb
->labels
[i
].text
: "";
201 tty_print_string (str_fit_to_term (text
, width
- 2, J_LEFT_FIT
));
207 for (i
= 0; i
< BUTTONBAR_LABELS_NUM
; i
++)
208 g_free (bb
->labels
[i
].text
);
212 return default_proc (msg
, parm
);
216 /* --------------------------------------------------------------------------------------------- */
219 buttonbar_event (Gpm_Event
* event
, void *data
)
221 Widget
*w
= (Widget
*) data
;
223 if (!mouse_global_in_widget (event
, w
))
224 return MOU_UNHANDLED
;
226 if ((event
->type
& GPM_UP
) != 0)
228 WButtonBar
*bb
= (WButtonBar
*) data
;
232 local
= mouse_get_local (event
, w
);
233 button
= buttonbar_get_button_by_x_coord (bb
, local
.x
- 1);
235 buttonbar_call (bb
, button
);
241 /* --------------------------------------------------------------------------------------------- */
242 /*** public functions ****************************************************************************/
243 /* --------------------------------------------------------------------------------------------- */
246 buttonbar_new (gboolean visible
)
250 bb
= g_new0 (WButtonBar
, 1);
252 init_widget (&bb
->widget
, LINES
- 1, 0, 1, COLS
, buttonbar_callback
, buttonbar_event
);
253 bb
->widget
.pos_flags
= WPOS_KEEP_HORZ
| WPOS_KEEP_BOTTOM
;
254 bb
->visible
= visible
;
255 widget_want_hotkey (bb
->widget
, 1);
256 widget_want_cursor (bb
->widget
, 0);
261 /* --------------------------------------------------------------------------------------------- */
264 buttonbar_set_label (WButtonBar
* bb
, int idx
, const char *text
,
265 const struct global_keymap_t
*keymap
, const Widget
* receiver
)
267 if ((bb
!= NULL
) && (idx
>= 1) && (idx
<= BUTTONBAR_LABELS_NUM
))
269 unsigned long command
= CK_IgnoreKey
;
272 command
= keybind_lookup_keymap_command (keymap
, KEY_F (idx
));
274 if ((text
== NULL
) || (text
[0] == '\0'))
275 set_label_text (bb
, idx
, "");
277 set_label_text (bb
, idx
, text
);
279 bb
->labels
[idx
- 1].command
= command
;
280 bb
->labels
[idx
- 1].receiver
= (Widget
*) receiver
;
284 /* --------------------------------------------------------------------------------------------- */
286 /* Find ButtonBar widget in the dialog */
288 find_buttonbar (const Dlg_head
* h
)
290 return (WButtonBar
*) find_widget_type (h
, buttonbar_callback
);