buttonbar.c 8.5 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292
  1. /*
  2. Widgets for the Midnight Commander
  3. Copyright (C) 1994, 1995, 1996, 1998, 1999, 2000, 2001, 2002, 2003,
  4. 2004, 2005, 2006, 2007, 2009, 2010, 2011, 2013
  5. The Free Software Foundation, Inc.
  6. Authors:
  7. Radek Doulik, 1994, 1995
  8. Miguel de Icaza, 1994, 1995
  9. Jakub Jelinek, 1995
  10. Andrej Borsenkow, 1996
  11. Norbert Warmuth, 1997
  12. Andrew Borodin <aborodin@vmail.ru>, 2009, 2010, 2013
  13. This file is part of the Midnight Commander.
  14. The Midnight Commander is free software: you can redistribute it
  15. and/or modify it under the terms of the GNU General Public License as
  16. published by the Free Software Foundation, either version 3 of the License,
  17. or (at your option) any later version.
  18. The Midnight Commander is distributed in the hope that it will be useful,
  19. but WITHOUT ANY WARRANTY; without even the implied warranty of
  20. MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
  21. GNU General Public License for more details.
  22. You should have received a copy of the GNU General Public License
  23. along with this program. If not, see <http://www.gnu.org/licenses/>.
  24. */
  25. /** \file buttonbar.c
  26. * \brief Source: WButtonBar widget
  27. */
  28. #include <config.h>
  29. #include <stdlib.h>
  30. #include <string.h>
  31. #include "lib/global.h"
  32. #include "lib/tty/tty.h"
  33. #include "lib/tty/mouse.h"
  34. #include "lib/tty/key.h" /* XCTRL and ALT macros */
  35. #include "lib/skin.h"
  36. #include "lib/strutil.h"
  37. #include "lib/util.h"
  38. #include "lib/keybind.h" /* global_keymap_t */
  39. #include "lib/widget.h"
  40. /*** global variables ****************************************************************************/
  41. /*** file scope macro definitions ****************************************************************/
  42. /*** file scope type declarations ****************************************************************/
  43. /*** file scope variables ************************************************************************/
  44. /*** file scope functions ************************************************************************/
  45. /* --------------------------------------------------------------------------------------------- */
  46. /* calculate positions of buttons; width is never less than 7 */
  47. static void
  48. buttonbar_init_button_positions (WButtonBar * bb)
  49. {
  50. int i;
  51. int pos = 0;
  52. if (COLS < BUTTONBAR_LABELS_NUM * 7)
  53. {
  54. for (i = 0; i < BUTTONBAR_LABELS_NUM; i++)
  55. {
  56. if (pos + 7 <= COLS)
  57. pos += 7;
  58. bb->labels[i].end_coord = pos;
  59. }
  60. }
  61. else
  62. {
  63. /* Distribute the extra width in a way that the middle vertical line
  64. (between F5 and F6) aligns with the two panels. The extra width
  65. is distributed in this order: F10, F5, F9, F4, ..., F6, F1. */
  66. int dv, md;
  67. dv = COLS / BUTTONBAR_LABELS_NUM;
  68. md = COLS % BUTTONBAR_LABELS_NUM;
  69. for (i = 0; i < BUTTONBAR_LABELS_NUM / 2; i++)
  70. {
  71. pos += dv;
  72. if (BUTTONBAR_LABELS_NUM / 2 - 1 - i < md / 2)
  73. pos++;
  74. bb->labels[i].end_coord = pos;
  75. }
  76. for (; i < BUTTONBAR_LABELS_NUM; i++)
  77. {
  78. pos += dv;
  79. if (BUTTONBAR_LABELS_NUM - 1 - i < (md + 1) / 2)
  80. pos++;
  81. bb->labels[i].end_coord = pos;
  82. }
  83. }
  84. }
  85. /* --------------------------------------------------------------------------------------------- */
  86. /* return width of one button */
  87. static int
  88. buttonbar_get_button_width (const WButtonBar * bb, int i)
  89. {
  90. if (i == 0)
  91. return bb->labels[0].end_coord;
  92. return bb->labels[i].end_coord - bb->labels[i - 1].end_coord;
  93. }
  94. /* --------------------------------------------------------------------------------------------- */
  95. static int
  96. buttonbar_get_button_by_x_coord (const WButtonBar * bb, int x)
  97. {
  98. int i;
  99. for (i = 0; i < BUTTONBAR_LABELS_NUM; i++)
  100. if (bb->labels[i].end_coord > x)
  101. return i;
  102. return (-1);
  103. }
  104. /* --------------------------------------------------------------------------------------------- */
  105. static void
  106. set_label_text (WButtonBar * bb, int idx, const char *text)
  107. {
  108. g_free (bb->labels[idx - 1].text);
  109. bb->labels[idx - 1].text = g_strdup (text);
  110. }
  111. /* --------------------------------------------------------------------------------------------- */
  112. /* returns TRUE if a function has been called, FALSE otherwise. */
  113. static gboolean
  114. buttonbar_call (WButtonBar * bb, int i)
  115. {
  116. cb_ret_t ret = MSG_NOT_HANDLED;
  117. Widget *w = WIDGET (bb);
  118. if ((bb != NULL) && (bb->labels[i].command != CK_IgnoreKey))
  119. ret = send_message (w->owner, w, MSG_ACTION, bb->labels[i].command, bb->labels[i].receiver);
  120. return ret;
  121. }
  122. /* --------------------------------------------------------------------------------------------- */
  123. static cb_ret_t
  124. buttonbar_callback (Widget * w, Widget * sender, widget_msg_t msg, int parm, void *data)
  125. {
  126. WButtonBar *bb = BUTTONBAR (w);
  127. int i;
  128. switch (msg)
  129. {
  130. case MSG_FOCUS:
  131. return MSG_NOT_HANDLED;
  132. case MSG_HOTKEY:
  133. for (i = 0; i < BUTTONBAR_LABELS_NUM; i++)
  134. if (parm == KEY_F (i + 1) && buttonbar_call (bb, i))
  135. return MSG_HANDLED;
  136. return MSG_NOT_HANDLED;
  137. case MSG_DRAW:
  138. if (bb->visible)
  139. {
  140. buttonbar_init_button_positions (bb);
  141. widget_move (w, 0, 0);
  142. tty_setcolor (DEFAULT_COLOR);
  143. tty_printf ("%-*s", w->cols, "");
  144. widget_move (w, 0, 0);
  145. for (i = 0; i < BUTTONBAR_LABELS_NUM; i++)
  146. {
  147. int width;
  148. const char *text;
  149. width = buttonbar_get_button_width (bb, i);
  150. if (width <= 0)
  151. break;
  152. tty_setcolor (BUTTONBAR_HOTKEY_COLOR);
  153. tty_printf ("%2d", i + 1);
  154. tty_setcolor (BUTTONBAR_BUTTON_COLOR);
  155. text = (bb->labels[i].text != NULL) ? bb->labels[i].text : "";
  156. tty_print_string (str_fit_to_term (text, width - 2, J_LEFT_FIT));
  157. }
  158. }
  159. return MSG_HANDLED;
  160. case MSG_DESTROY:
  161. for (i = 0; i < BUTTONBAR_LABELS_NUM; i++)
  162. g_free (bb->labels[i].text);
  163. return MSG_HANDLED;
  164. default:
  165. return widget_default_callback (w, sender, msg, parm, data);
  166. }
  167. }
  168. /* --------------------------------------------------------------------------------------------- */
  169. static int
  170. buttonbar_event (Gpm_Event * event, void *data)
  171. {
  172. Widget *w = WIDGET (data);
  173. if (!mouse_global_in_widget (event, w))
  174. return MOU_UNHANDLED;
  175. if ((event->type & GPM_UP) != 0)
  176. {
  177. WButtonBar *bb = BUTTONBAR (data);
  178. Gpm_Event local;
  179. int button;
  180. local = mouse_get_local (event, w);
  181. button = buttonbar_get_button_by_x_coord (bb, local.x - 1);
  182. if (button >= 0)
  183. buttonbar_call (bb, button);
  184. }
  185. return MOU_NORMAL;
  186. }
  187. /* --------------------------------------------------------------------------------------------- */
  188. /*** public functions ****************************************************************************/
  189. /* --------------------------------------------------------------------------------------------- */
  190. WButtonBar *
  191. buttonbar_new (gboolean visible)
  192. {
  193. WButtonBar *bb;
  194. Widget *w;
  195. bb = g_new0 (WButtonBar, 1);
  196. w = WIDGET (bb);
  197. widget_init (w, LINES - 1, 0, 1, COLS, buttonbar_callback, buttonbar_event);
  198. w->pos_flags = WPOS_KEEP_HORZ | WPOS_KEEP_BOTTOM;
  199. bb->visible = visible;
  200. widget_want_hotkey (w, TRUE);
  201. widget_want_cursor (w, FALSE);
  202. return bb;
  203. }
  204. /* --------------------------------------------------------------------------------------------- */
  205. void
  206. buttonbar_set_label (WButtonBar * bb, int idx, const char *text,
  207. const struct global_keymap_t *keymap, const Widget * receiver)
  208. {
  209. if ((bb != NULL) && (idx >= 1) && (idx <= BUTTONBAR_LABELS_NUM))
  210. {
  211. unsigned long command = CK_IgnoreKey;
  212. if (keymap != NULL)
  213. command = keybind_lookup_keymap_command (keymap, KEY_F (idx));
  214. if ((text == NULL) || (text[0] == '\0'))
  215. set_label_text (bb, idx, "");
  216. else
  217. set_label_text (bb, idx, text);
  218. bb->labels[idx - 1].command = command;
  219. bb->labels[idx - 1].receiver = WIDGET (receiver);
  220. }
  221. }
  222. /* --------------------------------------------------------------------------------------------- */
  223. /* Find ButtonBar widget in the dialog */
  224. WButtonBar *
  225. find_buttonbar (const WDialog * h)
  226. {
  227. return BUTTONBAR (find_widget_type (h, buttonbar_callback));
  228. }