]> git.ozlabs.org Git - petitboot/blob - ui/ncurses/nc-menu.c
ui/ncurses: Add nc_scr_{un,}post()
[petitboot] / ui / ncurses / nc-menu.c
1 /*
2  *  Copyright (C) 2009 Sony Computer Entertainment Inc.
3  *  Copyright 2009 Sony Corp.
4  *
5  *  This program is free software; you can redistribute it and/or modify
6  *  it under the terms of the GNU General Public License as published by
7  *  the Free Software Foundation; version 2 of the License.
8  *
9  *  This program is distributed in the hope that it will be useful,
10  *  but WITHOUT ANY WARRANTY; without even the implied warranty of
11  *  MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
12  *  GNU General Public License for more details.
13  *
14  *  You should have received a copy of the GNU General Public License
15  *  along with this program; if not, write to the Free Software
16  *  Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA  02111-1307  USA
17  */
18
19 #include "config.h"
20
21 #define _GNU_SOURCE
22
23 #include <assert.h>
24 #include <errno.h>
25 #include <string.h>
26
27 #include "log/log.h"
28 #include "talloc/talloc.h"
29 #include "ui/common/ui-system.h"
30 #include "nc-menu.h"
31
32 /**
33  * pmenu_exit_cb - Callback helper that runs run menu.on_exit().
34  */
35
36 int pmenu_exit_cb(struct pmenu_item *item)
37 {
38         assert(item->pmenu->on_exit);
39         item->pmenu->on_exit(item->pmenu);
40         return 0;
41 }
42
43 /**
44  * pmenu_find_selected - Find the selected pmenu_item.
45  */
46
47 struct pmenu_item *pmenu_find_selected(struct pmenu *menu)
48 {
49         return pmenu_item_from_arg(item_userptr(current_item(menu->ncm)));
50 }
51
52 static int pmenu_post(struct nc_scr *scr)
53 {
54         int result;
55         struct pmenu *menu = pmenu_from_scr(scr);
56
57         result = post_menu(menu->ncm);
58
59         nc_scr_frame_draw(scr);
60         redrawwin(menu->scr.main_ncw);
61         wrefresh(menu->scr.main_ncw);
62
63         return result;
64 }
65
66 static int pmenu_unpost(struct nc_scr *scr)
67 {
68         return unpost_menu(pmenu_from_scr(scr)->ncm);
69 }
70
71 static void pmenu_resize(struct nc_scr *scr)
72 {
73         /* FIXME: menus can't be resized, need to recreate here */
74         pmenu_unpost(scr);
75         pmenu_post(scr);
76 }
77
78 /**
79  * pmenu_item_init - Allocate and initialize a new pmenu_item instance.
80  *
81  * Returns a pointer the the initialized struct pmenu_item instance or NULL
82  * on error. The caller is responsible for calling talloc_free() for the
83  * returned instance.
84  */
85
86 struct pmenu_item *pmenu_item_alloc(struct pmenu *menu)
87 {
88         /* Items go with the menu, not the pointer array. */
89
90         struct pmenu_item *i = talloc_zero(menu, struct pmenu_item);
91
92         return i;
93 }
94
95 struct pmenu_item *pmenu_item_setup(struct pmenu *menu, struct pmenu_item *i,
96         unsigned int index, const char *name)
97 {
98         assert(i);
99         assert(name);
100
101         if (!i)
102                 return NULL;
103
104         i->i_sig = pb_item_sig;
105         i->pmenu = menu;
106         i->nci = new_item(name, NULL);
107
108         if (!i->nci) {
109                 talloc_free(i);
110                 return NULL;
111         }
112
113         set_item_userptr(i->nci, i);
114
115         menu->items[index] = i->nci;
116
117         return i;
118 }
119
120 static int pmenu_item_get_index(const struct pmenu_item *item)
121 {
122         unsigned int i;
123
124         for (i = 0; i < item->pmenu->item_count; i++)
125                 if (item->pmenu->items[i] == item->nci)
126                         return i;
127
128         pb_log("%s: not found: %p %s\n", __func__, item,
129                 (item ? item->nci->name.str : "(null)"));
130         return -1;
131 }
132
133 /**
134  * pmenu_item_replace - Replace the menu item with a new one.
135  *
136  * Use this routine to change a menu item's text.
137  */
138
139 int pmenu_item_replace(struct pmenu_item *i, const char *name)
140 {
141         struct pmenu *menu;
142         ITEM *nci;
143         int index;
144
145         assert(name);
146         assert(i->nci);
147
148         menu = i->pmenu;
149         index = pmenu_item_get_index(i);
150
151         if (index < 0) {
152                 assert(0 && "get_index failed");
153                 return -1;
154         }
155
156         nci = new_item(name, NULL);
157
158         if (!nci) {
159                 assert(0 && "new_item failed");
160                 return -1;
161         }
162
163         set_item_userptr(nci, i);
164
165         nc_scr_unpost(&menu->scr);
166         set_menu_items(menu->ncm, NULL);
167
168         // FIXME: need to assure item name is a talloc string.
169         /* talloc_free((char *)item_name(i->nci)); */
170
171         free_item(i->nci);
172         menu->items[index] = nci;
173         i->nci = nci;
174
175         set_menu_items(menu->ncm, menu->items);
176         nc_scr_post(&menu->scr);
177
178         return 0;
179 }
180
181 /**
182  * pmenu_move_cursor - Move the cursor.
183  * @req: An ncurses request or char to send to menu_driver().
184  */
185
186 static void pmenu_move_cursor(struct pmenu *menu, int req)
187 {
188         menu_driver(menu->ncm, req);
189         wrefresh(menu->scr.main_ncw);
190 }
191
192 /**
193  * pmenu_process_key - Process a user keystroke.
194  */
195
196 static void pmenu_process_key(struct nc_scr *scr, int key)
197 {
198         struct pmenu *menu = pmenu_from_scr(scr);
199         struct pmenu_item *item = pmenu_find_selected(menu);
200
201         nc_scr_status_free(&menu->scr);
202
203         if (menu->hot_key)
204                 key = menu->hot_key(menu, item, key);
205
206         switch (key) {
207         case 27: /* ESC */
208         case 'x':
209                 if (menu->on_exit)
210                         menu->on_exit(menu);
211                 nc_flush_keys();
212                 return;
213
214         case KEY_PPAGE:
215                 pmenu_move_cursor(menu, REQ_SCR_UPAGE);
216                 break;
217         case KEY_NPAGE:
218                 pmenu_move_cursor(menu, REQ_SCR_DPAGE);
219                 break;
220         case KEY_HOME:
221                 pmenu_move_cursor(menu, REQ_FIRST_ITEM);
222                 break;
223         case KEY_END:
224                 pmenu_move_cursor(menu, REQ_LAST_ITEM);
225                 break;
226         case KEY_UP:
227                 pmenu_move_cursor(menu, REQ_UP_ITEM);
228                 break;
229         case KEY_DOWN:
230                 pmenu_move_cursor(menu, REQ_DOWN_ITEM);
231                 break;
232         case 'e':
233                 if (item->on_edit)
234                         item->on_edit(item);
235                 break;
236         case 'n':
237                 if (menu->on_new)
238                         menu->on_new(menu);
239                 break;
240         case '\n':
241         case '\r':
242                 if (item->on_execute)
243                         item->on_execute(item);
244                 break;
245         default:
246                 menu_driver(menu->ncm, key);
247                 break;
248         }
249 }
250
251 /**
252  * pmenu_grow - Grow the item array.
253  * @count: The count of new items.
254  *
255  * The item array must be disconnected prior to calling pmenu_grow().
256  * Returns the insert point index.
257  */
258
259 unsigned int pmenu_grow(struct pmenu *menu, unsigned int count)
260 {
261         unsigned int tmp;
262
263         assert(item_count(menu->ncm) == 0 && "not disconnected");
264
265         pb_log("%s: %u current + %u new = %u\n", __func__, menu->item_count,
266                 count, menu->item_count + count);
267
268         /* Note that items array has a null terminator. */
269
270         menu->items = talloc_realloc(menu, menu->items, ITEM *,
271                 menu->item_count + count + 1);
272
273         memmove(menu->items + menu->insert_pt + count,
274                 menu->items + menu->insert_pt,
275                 (menu->item_count - menu->insert_pt + 1) * sizeof(ITEM *));
276
277         memset(menu->items + menu->insert_pt, 0, count * sizeof(ITEM *));
278
279         tmp = menu->insert_pt;
280         menu->insert_pt += count;
281         menu->item_count += count;
282
283         return tmp;
284 }
285
286 /**
287  * pmenu_remove - Remove an item from the item array.
288  *
289  * The item array must be disconnected prior to calling pmenu_remove()
290  */
291
292 int pmenu_remove(struct pmenu *menu, struct pmenu_item *item)
293 {
294         int index;
295
296         assert(item_count(menu->ncm) == 0 && "not disconnected");
297
298         assert(menu->item_count);
299
300         pb_log("%s: %u\n", __func__, menu->item_count);
301
302         index = pmenu_item_get_index(item);
303
304         if (index < 0)
305                 return -1;
306
307         free_item(item->nci);
308         talloc_free(item);
309
310         /* Note that items array has a null terminator. */
311
312         menu->insert_pt--;
313         menu->item_count--;
314
315         memmove(&menu->items[index], &menu->items[index + 1],
316                 (menu->item_count - index + 1) * sizeof(ITEM *));
317         menu->items = talloc_realloc(menu, menu->items, ITEM *,
318                 menu->item_count + 1);
319
320         return 0;
321 }
322
323 /**
324  * pmenu_init - Allocate and initialize a new menu instance.
325  *
326  * Returns a pointer the the initialized struct pmenu instance or NULL on error.
327  * The caller is responsible for calling talloc_free() for the returned
328  * instance.
329  */
330
331 struct pmenu *pmenu_init(void *ui_ctx, unsigned int item_count,
332         void (*on_exit)(struct pmenu *))
333 {
334         struct pmenu *menu = talloc_zero(ui_ctx, struct pmenu);
335
336         if (!menu)
337                 return NULL;
338
339         /* note items array has a null terminator */
340
341         menu->items = talloc_zero_array(menu, ITEM *, item_count + 1);
342
343         if (!menu->items) {
344                 talloc_free(menu);
345                 return NULL;
346         }
347
348         nc_scr_init(&menu->scr, pb_pmenu_sig, 0, ui_ctx, pmenu_process_key,
349                 pmenu_post, pmenu_unpost, pmenu_resize);
350
351         menu->item_count = item_count;
352         menu->insert_pt = 0; /* insert from top */
353         menu->on_exit = on_exit;
354
355         return menu;
356 }
357
358 /**
359  * pmenu_setup - Create nc menu, setup nc windows.
360  *
361  */
362
363 int pmenu_setup(struct pmenu *menu)
364 {
365         assert(!menu->ncm);
366
367         menu->ncm = new_menu(menu->items);
368
369         if (!menu->ncm) {
370                 pb_log("%s:%d: new_menu failed: %s\n", __func__, __LINE__,
371                         strerror(errno));
372                 return -1;
373         }
374
375         set_menu_win(menu->ncm, menu->scr.main_ncw);
376         set_menu_sub(menu->ncm, menu->scr.sub_ncw);
377
378         /* Makes menu scrollable. */
379         set_menu_format(menu->ncm, LINES - nc_scr_frame_lines, 1);
380
381         return 0;
382 }
383
384 /**
385  * pmenu_delete - Delete a menu instance.
386  *
387  */
388
389 void pmenu_delete(struct pmenu *menu)
390 {
391         unsigned int i;
392
393         assert(menu->scr.sig == pb_pmenu_sig);
394         menu->scr.sig = pb_removed_sig;
395
396         for (i = item_count(menu->ncm); i; i--)
397                 free_item(menu->items[i - 1]);
398
399         free_menu(menu->ncm);
400         delwin(menu->scr.sub_ncw);
401         delwin(menu->scr.main_ncw);
402         talloc_free(menu);
403 }