2 Widgets for the Midnight Commander
4 Copyright (C) 1994-2017
5 Free Software Foundation, Inc.
8 Radek Doulik, 1994, 1995
9 Miguel de Icaza, 1994, 1995
11 Andrej Borsenkow, 1996
13 Andrew Borodin <aborodin@vmail.ru>, 2009, 2010, 2013, 2016
15 This file is part of the Midnight Commander.
17 The Midnight Commander is free software: you can redistribute it
18 and/or modify it under the terms of the GNU General Public License as
19 published by the Free Software Foundation, either version 3 of the License,
20 or (at your option) any later version.
22 The Midnight Commander is distributed in the hope that it will be useful,
23 but WITHOUT ANY WARRANTY; without even the implied warranty of
24 MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
25 GNU General Public License for more details.
27 You should have received a copy of the GNU General Public License
28 along with this program. If not, see <http://www.gnu.org/licenses/>.
32 * \brief Source: WListbox widget
39 #include "lib/global.h"
41 #include "lib/tty/tty.h"
43 #include "lib/strutil.h"
44 #include "lib/util.h" /* Q_() */
45 #include "lib/keybind.h" /* global_keymap_t */
46 #include "lib/widget.h"
48 /*** global variables ****************************************************************************/
50 const global_keymap_t
*listbox_map
= NULL
;
52 /*** file scope macro definitions ****************************************************************/
54 /* Gives the position of the last item. */
55 #define LISTBOX_LAST(l) (listbox_is_empty (l) ? 0 : (int) g_queue_get_length ((l)->list) - 1)
57 /*** file scope type declarations ****************************************************************/
59 /*** file scope variables ************************************************************************/
61 /*** file scope functions ************************************************************************/
64 listbox_entry_cmp (const void *a
, const void *b
, void *user_data
)
66 const WLEntry
*ea
= (const WLEntry
*) a
;
67 const WLEntry
*eb
= (const WLEntry
*) b
;
71 return strcmp (ea
->text
, eb
->text
);
74 /* --------------------------------------------------------------------------------------------- */
77 listbox_entry_free (void *data
)
87 /* --------------------------------------------------------------------------------------------- */
90 listbox_drawscroll (WListbox
* l
)
92 Widget
*w
= WIDGET (l
);
93 int max_line
= w
->lines
- 1;
98 /* Are we at the top? */
99 widget_move (w
, 0, w
->cols
);
101 tty_print_one_vline (TRUE
);
103 tty_print_char ('^');
105 length
= g_queue_get_length (l
->list
);
107 /* Are we at the bottom? */
108 widget_move (w
, max_line
, w
->cols
);
109 if (l
->top
+ w
->lines
== length
|| w
->lines
>= length
)
110 tty_print_one_vline (TRUE
);
112 tty_print_char ('v');
114 /* Now draw the nice relative pointer */
115 if (!g_queue_is_empty (l
->list
))
116 line
= 1 + ((l
->pos
* (w
->lines
- 2)) / length
);
118 for (i
= 1; i
< max_line
; i
++)
120 widget_move (w
, i
, w
->cols
);
122 tty_print_one_vline (TRUE
);
124 tty_print_char ('*');
128 /* --------------------------------------------------------------------------------------------- */
131 listbox_draw (WListbox
* l
, gboolean focused
)
133 Widget
*w
= WIDGET (l
);
134 const WDialog
*h
= w
->owner
;
143 disabled
= widget_get_state (w
, WST_DISABLED
);
144 normalc
= disabled
? DISABLED_COLOR
: h
->color
[DLG_COLOR_NORMAL
];
149 ? h
->color
[DLG_COLOR_HOT_FOCUS
]
150 : h
->color
[DLG_COLOR_FOCUS
];
155 length
= g_queue_get_length (l
->list
);
156 le
= g_queue_peek_nth_link (l
->list
, (guint
) l
->top
);
159 /* pos = (le == NULL) ? 0 : g_list_position (l->list, le); */
160 pos
= (le
== NULL
) ? 0 : l
->top
;
162 for (i
= 0; i
< w
->lines
; i
++)
164 const char *text
= "";
166 /* Display the entry */
167 if (pos
== l
->pos
&& sel_line
== -1)
173 tty_setcolor (normalc
);
175 widget_move (l
, i
, 1);
177 if (l
->list
!= NULL
&& le
!= NULL
&& (i
== 0 || pos
< length
))
179 WLEntry
*e
= LENTRY (le
->data
);
182 le
= g_list_next (le
);
186 tty_print_string (str_fit_to_term (text
, w
->cols
- 2, J_LEFT_FIT
));
189 l
->cursor_y
= sel_line
;
191 if (l
->scrollbar
&& length
> w
->lines
)
193 tty_setcolor (normalc
);
194 listbox_drawscroll (l
);
198 /* --------------------------------------------------------------------------------------------- */
201 listbox_check_hotkey (WListbox
* l
, int key
)
203 if (!listbox_is_empty (l
))
208 for (i
= 0, le
= g_queue_peek_head_link (l
->list
); le
!= NULL
; i
++, le
= g_list_next (le
))
210 WLEntry
*e
= LENTRY (le
->data
);
212 if (e
->hotkey
== key
)
220 /* --------------------------------------------------------------------------------------------- */
222 /* Calculates the item displayed at screen row 'y' (y==0 being the widget's 1st row). */
224 listbox_y_pos (WListbox
* l
, int y
)
226 return MIN (l
->top
+ y
, LISTBOX_LAST (l
));
229 /* --------------------------------------------------------------------------------------------- */
232 listbox_fwd (WListbox
* l
, gboolean wrap
)
234 if (!listbox_is_empty (l
))
236 if ((guint
) l
->pos
+ 1 < g_queue_get_length (l
->list
))
237 listbox_select_entry (l
, l
->pos
+ 1);
239 listbox_select_first (l
);
243 /* --------------------------------------------------------------------------------------------- */
246 listbox_fwd_n (WListbox
* l
, int n
)
248 listbox_select_entry (l
, MIN (l
->pos
+ n
, LISTBOX_LAST (l
)));
251 /* --------------------------------------------------------------------------------------------- */
254 listbox_back (WListbox
* l
, gboolean wrap
)
256 if (!listbox_is_empty (l
))
259 listbox_select_entry (l
, l
->pos
- 1);
261 listbox_select_last (l
);
265 /* --------------------------------------------------------------------------------------------- */
268 listbox_back_n (WListbox
* l
, int n
)
270 listbox_select_entry (l
, MAX (l
->pos
- n
, 0));
273 /* --------------------------------------------------------------------------------------------- */
276 listbox_execute_cmd (WListbox
* l
, long command
)
278 cb_ret_t ret
= MSG_HANDLED
;
279 Widget
*w
= WIDGET (l
);
281 if (l
->list
== NULL
|| g_queue_is_empty (l
->list
))
282 return MSG_NOT_HANDLED
;
287 listbox_back (l
, TRUE
);
290 listbox_fwd (l
, TRUE
);
293 listbox_select_first (l
);
296 listbox_select_last (l
);
299 listbox_back_n (l
, w
->lines
- 1);
302 listbox_fwd_n (l
, w
->lines
- 1);
307 gboolean is_last
, is_more
;
310 length
= g_queue_get_length (l
->list
);
312 is_last
= (l
->pos
+ 1 >= length
);
313 is_more
= (l
->top
+ w
->lines
>= length
);
315 listbox_remove_current (l
);
316 if ((l
->top
> 0) && (is_last
|| is_more
))
321 if (l
->deletable
&& mc_global
.widget
.confirm_history_cleanup
322 /* TRANSLATORS: no need to translate 'DialogTitle', it's just a context prefix */
323 && (query_dialog (Q_ ("DialogTitle|History cleanup"),
324 _("Do you want clean this history?"),
325 D_ERROR
, 2, _("&Yes"), _("&No")) == 0))
326 listbox_remove_list (l
);
329 ret
= MSG_NOT_HANDLED
;
335 /* --------------------------------------------------------------------------------------------- */
337 /* Return MSG_HANDLED if we want a redraw */
339 listbox_key (WListbox
* l
, int key
)
344 return MSG_NOT_HANDLED
;
346 /* focus on listbox item N by '0'..'9' keys */
347 if (key
>= '0' && key
<= '9')
349 listbox_select_entry (l
, key
- '0');
353 command
= keybind_lookup_keymap_command (listbox_map
, key
);
354 if (command
== CK_IgnoreKey
)
355 return MSG_NOT_HANDLED
;
356 return listbox_execute_cmd (l
, command
);
359 /* --------------------------------------------------------------------------------------------- */
361 /* Listbox item adding function */
363 listbox_append_item (WListbox
* l
, WLEntry
* e
, listbox_append_t pos
)
367 l
->list
= g_queue_new ();
368 pos
= LISTBOX_APPEND_AT_END
;
373 case LISTBOX_APPEND_AT_END
:
374 g_queue_push_tail (l
->list
, e
);
377 case LISTBOX_APPEND_BEFORE
:
378 g_queue_insert_before (l
->list
, g_queue_peek_nth_link (l
->list
, (guint
) l
->pos
), e
);
381 case LISTBOX_APPEND_AFTER
:
382 g_queue_insert_after (l
->list
, g_queue_peek_nth_link (l
->list
, (guint
) l
->pos
), e
);
385 case LISTBOX_APPEND_SORTED
:
386 g_queue_insert_sorted (l
->list
, e
, (GCompareDataFunc
) listbox_entry_cmp
, NULL
);
394 /* --------------------------------------------------------------------------------------------- */
396 /* Call this whenever the user changes the selected item. */
398 listbox_on_change (WListbox
* l
)
400 listbox_draw (l
, TRUE
);
401 send_message (WIDGET (l
)->owner
, l
, MSG_NOTIFY
, l
->pos
, NULL
);
404 /* --------------------------------------------------------------------------------------------- */
407 listbox_do_action (WListbox
* l
)
411 if (listbox_is_empty (l
))
414 if (l
->callback
!= NULL
)
415 action
= l
->callback (l
);
417 action
= LISTBOX_DONE
;
419 if (action
== LISTBOX_DONE
)
421 WDialog
*h
= WIDGET (l
)->owner
;
423 h
->ret_value
= B_ENTER
;
428 /* --------------------------------------------------------------------------------------------- */
431 listbox_run_hotkey (WListbox
* l
, int pos
)
433 listbox_select_entry (l
, pos
);
434 listbox_on_change (l
);
435 listbox_do_action (l
);
438 /* --------------------------------------------------------------------------------------------- */
441 listbox_destroy (WListbox
* l
)
443 listbox_remove_list (l
);
446 /* --------------------------------------------------------------------------------------------- */
449 listbox_callback (Widget
* w
, Widget
* sender
, widget_msg_t msg
, int parm
, void *data
)
451 WListbox
*l
= LISTBOX (w
);
460 pos
= listbox_check_hotkey (l
, parm
);
462 return MSG_NOT_HANDLED
;
464 listbox_run_hotkey (l
, pos
);
470 ret_code
= listbox_key (l
, parm
);
471 if (ret_code
!= MSG_NOT_HANDLED
)
472 listbox_on_change (l
);
476 return listbox_execute_cmd (l
, parm
);
479 widget_move (l
, l
->cursor_y
, 0);
483 listbox_draw (l
, widget_get_state (w
, WST_FOCUSED
));
494 return widget_default_callback (w
, sender
, msg
, parm
, data
);
498 /* --------------------------------------------------------------------------------------------- */
501 listbox_mouse_callback (Widget
* w
, mouse_msg_t msg
, mouse_event_t
* event
)
503 WListbox
*l
= LISTBOX (w
);
512 listbox_select_entry (l
, listbox_y_pos (l
, event
->y
));
515 case MSG_MOUSE_SCROLL_UP
:
516 listbox_back (l
, FALSE
);
519 case MSG_MOUSE_SCROLL_DOWN
:
520 listbox_fwd (l
, FALSE
);
524 event
->result
.repeat
= TRUE
; /* It'd be functional even without this. */
525 listbox_select_entry (l
, listbox_y_pos (l
, event
->y
));
528 case MSG_MOUSE_CLICK
:
529 /* We don't call listbox_select_entry() here: MSG_MOUSE_DOWN/DRAG did this already. */
530 if (event
->count
== GPM_DOUBLE
) /* Double click */
531 listbox_do_action (l
);
538 /* If the selection has changed, we redraw the widget and notify the dialog. */
539 if (l
->pos
!= old_pos
)
540 listbox_on_change (l
);
543 /* --------------------------------------------------------------------------------------------- */
544 /*** public functions ****************************************************************************/
545 /* --------------------------------------------------------------------------------------------- */
548 listbox_new (int y
, int x
, int height
, int width
, gboolean deletable
, lcback_fn callback
)
556 l
= g_new (WListbox
, 1);
558 widget_init (w
, y
, x
, height
, width
, listbox_callback
, listbox_mouse_callback
);
559 w
->options
|= WOP_SELECTABLE
| WOP_WANT_HOTKEY
;
563 l
->deletable
= deletable
;
564 l
->callback
= callback
;
565 l
->allow_duplicates
= TRUE
;
566 l
->scrollbar
= !mc_global
.tty
.slow_terminal
;
571 /* --------------------------------------------------------------------------------------------- */
574 * Finds item by its label.
577 listbox_search_text (WListbox
* l
, const char *text
)
579 if (!listbox_is_empty (l
))
584 for (i
= 0, le
= g_queue_peek_head_link (l
->list
); le
!= NULL
; i
++, le
= g_list_next (le
))
586 WLEntry
*e
= LENTRY (le
->data
);
588 if (strcmp (e
->text
, text
) == 0)
596 /* --------------------------------------------------------------------------------------------- */
599 * Finds item by its 'data' slot.
602 listbox_search_data (WListbox
* l
, const void *data
)
604 if (!listbox_is_empty (l
))
609 for (i
= 0, le
= g_queue_peek_head_link (l
->list
); le
!= NULL
; i
++, le
= g_list_next (le
))
611 WLEntry
*e
= LENTRY (le
->data
);
621 /* --------------------------------------------------------------------------------------------- */
623 /* Selects the first entry and scrolls the list to the top */
625 listbox_select_first (WListbox
* l
)
630 /* --------------------------------------------------------------------------------------------- */
632 /* Selects the last entry and scrolls the list to the bottom */
634 listbox_select_last (WListbox
* l
)
636 int lines
= WIDGET (l
)->lines
;
639 if (!listbox_is_empty (l
))
640 length
= g_queue_get_length (l
->list
);
642 l
->pos
= length
> 0 ? length
- 1 : 0;
643 l
->top
= length
> lines
? length
- lines
: 0;
646 /* --------------------------------------------------------------------------------------------- */
649 listbox_select_entry (WListbox
* l
, int dest
)
653 gboolean top_seen
= FALSE
;
655 if (listbox_is_empty (l
) || dest
< 0)
659 for (pos
= 0, le
= g_queue_peek_head_link (l
->list
); le
!= NULL
; pos
++, le
= g_list_next (le
))
671 int lines
= WIDGET (l
)->lines
;
673 if (l
->pos
- l
->top
>= lines
)
674 l
->top
= l
->pos
- lines
+ 1;
680 /* If we are unable to find it, set decent values */
684 /* --------------------------------------------------------------------------------------------- */
686 /* Returns the current string text as well as the associated extra data */
688 listbox_get_current (WListbox
* l
, char **string
, void **extra
)
694 e
= listbox_get_nth_item (l
, l
->pos
);
699 *string
= ok
? e
->text
: NULL
;
702 *extra
= ok
? e
->data
: NULL
;
705 /* --------------------------------------------------------------------------------------------- */
708 listbox_get_nth_item (const WListbox
* l
, int pos
)
710 if (!listbox_is_empty (l
) && pos
>= 0)
714 item
= g_queue_peek_nth_link (l
->list
, (guint
) pos
);
716 return LENTRY (item
->data
);
722 /* --------------------------------------------------------------------------------------------- */
725 listbox_get_first_link (const WListbox
* l
)
727 return (l
== NULL
|| l
->list
== NULL
) ? NULL
: g_queue_peek_head_link (l
->list
);
730 /* --------------------------------------------------------------------------------------------- */
733 listbox_remove_current (WListbox
* l
)
735 if (!listbox_is_empty (l
))
740 current
= g_queue_peek_nth_link (l
->list
, (guint
) l
->pos
);
741 listbox_entry_free (LENTRY (current
->data
));
742 g_queue_delete_link (l
->list
, current
);
744 length
= g_queue_get_length (l
->list
);
748 else if (l
->pos
>= length
)
753 /* --------------------------------------------------------------------------------------------- */
756 listbox_is_empty (const WListbox
* l
)
758 return (l
== NULL
|| l
->list
== NULL
|| g_queue_is_empty (l
->list
));
761 /* --------------------------------------------------------------------------------------------- */
764 listbox_set_list (WListbox
* l
, GList
* list
)
766 listbox_remove_list (l
);
772 l
->list
= g_queue_new ();
774 for (ll
= list
; ll
!= NULL
; ll
= g_list_next (ll
))
775 g_queue_push_tail (l
->list
, ll
->data
);
781 /* --------------------------------------------------------------------------------------------- */
784 listbox_remove_list (WListbox
* l
)
790 g_queue_free_full (l
->list
, (GDestroyNotify
) listbox_entry_free
);
798 /* --------------------------------------------------------------------------------------------- */
801 listbox_add_item (WListbox
* l
, listbox_append_t pos
, int hotkey
, const char *text
, void *data
,
809 if (!l
->allow_duplicates
&& (listbox_search_text (l
, text
) >= 0))
812 entry
= g_new (WLEntry
, 1);
813 entry
->text
= g_strdup (text
);
815 entry
->free_data
= free_data
;
816 entry
->hotkey
= hotkey
;
818 listbox_append_item (l
, entry
, pos
);
823 /* --------------------------------------------------------------------------------------------- */