2 * Copyright (C) 2009 Sony Computer Entertainment Inc.
3 * Copyright 2009 Sony Corp.
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.
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.
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
26 #include "talloc/talloc.h"
27 #include "ui/common/ui-system.h"
31 * pmenu_exit_cb - Callback helper that runs run menu.on_exit().
34 int pmenu_exit_cb(struct pmenu_item *item)
36 assert(item->pmenu->on_exit);
37 item->pmenu->on_exit(item->pmenu);
42 * pmenu_find_selected - Find the selected pmenu_item.
45 struct pmenu_item *pmenu_find_selected(struct pmenu *menu)
47 return pmenu_item_from_arg(item_userptr(current_item(menu->ncm)));
50 static int pmenu_post(struct nc_scr *scr)
53 struct pmenu *menu = pmenu_from_scr(scr);
55 result = post_menu(menu->ncm);
57 nc_scr_frame_draw(scr);
58 redrawwin(menu->scr.main_ncw);
59 wrefresh(menu->scr.main_ncw);
64 static int pmenu_unpost(struct nc_scr *scr)
66 return unpost_menu(pmenu_from_scr(scr)->ncm);
69 static void pmenu_resize(struct nc_scr *scr)
71 /* FIXME: menus can't be resized, need to recreate here */
77 * pmenu_item_init - Allocate and initialize a new pmenu_item instance.
79 * Returns a pointer the the initialized struct pmenu_item instance or NULL
80 * on error. The caller is responsible for calling talloc_free() for the
84 struct pmenu_item *pmenu_item_alloc(struct pmenu *menu)
86 /* Items go with the menu, not the pointer array. */
88 struct pmenu_item *i = talloc_zero(menu, struct pmenu_item);
93 struct pmenu_item *pmenu_item_setup(struct pmenu *menu, struct pmenu_item *i,
94 unsigned int index, const char *name)
102 i->i_sig = pb_item_sig;
104 i->nci = new_item(name, NULL);
111 set_item_userptr(i->nci, i);
113 menu->items[index] = i->nci;
118 static int pmenu_item_get_index(const struct pmenu_item *item)
122 for (i = 0; i < item->pmenu->item_count; i++)
123 if (item->pmenu->items[i] == item->nci)
126 pb_log("%s: not found: %p %s\n", __func__, item,
127 (item ? item->nci->name.str : "(null)"));
132 * pmenu_item_replace - Replace the menu item with a new one.
134 * Use this routine to change a menu item's text.
137 int pmenu_item_replace(struct pmenu_item *i, const char *name)
147 index = pmenu_item_get_index(i);
150 assert(0 && "get_index failed");
154 nci = new_item(name, NULL);
157 assert(0 && "new_item failed");
161 set_item_userptr(nci, i);
163 menu->scr.unpost(&menu->scr);
164 set_menu_items(menu->ncm, NULL);
166 // FIXME: need to assure item name is a talloc string.
167 /* talloc_free((char *)item_name(i->nci)); */
170 menu->items[index] = nci;
173 set_menu_items(menu->ncm, menu->items);
174 menu->scr.post(&menu->scr);
180 * pmenu_move_cursor - Move the cursor.
181 * @req: An ncurses request or char to send to menu_driver().
184 static void pmenu_move_cursor(struct pmenu *menu, int req)
186 menu_driver(menu->ncm, req);
187 wrefresh(menu->scr.main_ncw);
191 * pmenu_process_key - Process a user keystroke.
194 static void pmenu_process_key(struct nc_scr *scr)
196 struct pmenu *menu = pmenu_from_scr(scr);
197 struct pmenu_item *item = pmenu_find_selected(menu);
199 nc_scr_status_free(&menu->scr);
207 DBGS("%d (%o)\n", c, c);
210 c = menu->hot_key(menu, item, c);
220 pmenu_move_cursor(menu, REQ_SCR_UPAGE);
223 pmenu_move_cursor(menu, REQ_SCR_DPAGE);
226 pmenu_move_cursor(menu, REQ_FIRST_ITEM);
229 pmenu_move_cursor(menu, REQ_LAST_ITEM);
232 pmenu_move_cursor(menu, REQ_UP_ITEM);
235 pmenu_move_cursor(menu, REQ_DOWN_ITEM);
242 DBGS("on_open: %p\n", menu->on_open);
248 if (item->on_execute)
249 item->on_execute(item);
252 menu_driver(menu->ncm, c);
259 * pmenu_grow - Grow the item array.
260 * @count: The count of new items.
262 * The item array must be disconnected prior to calling pmenu_grow().
263 * Returns the insert point index.
266 unsigned int pmenu_grow(struct pmenu *menu, unsigned int count)
270 assert(item_count(menu->ncm) == 0 && "not disconnected");
272 pb_log("%s: %u current + %u new = %u\n", __func__, menu->item_count,
273 count, menu->item_count + count);
275 /* Note that items array has a null terminator. */
277 menu->items = talloc_realloc(menu, menu->items, ITEM *,
278 menu->item_count + count + 1);
280 memmove(menu->items + menu->insert_pt + count,
281 menu->items + menu->insert_pt,
282 (menu->item_count - menu->insert_pt + 1) * sizeof(ITEM *));
284 memset(menu->items + menu->insert_pt, 0, count * sizeof(ITEM *));
286 tmp = menu->insert_pt;
287 menu->insert_pt += count;
288 menu->item_count += count;
294 * pmenu_remove - Remove an item from the item array.
296 * The item array must be disconnected prior to calling pmenu_remove()
299 int pmenu_remove(struct pmenu *menu, struct pmenu_item *item)
303 assert(item_count(menu->ncm) == 0 && "not disconnected");
305 assert(menu->item_count);
307 pb_log("%s: %u\n", __func__, menu->item_count);
309 index = pmenu_item_get_index(item);
314 free_item(item->nci);
317 /* Note that items array has a null terminator. */
322 memmove(&menu->items[index], &menu->items[index + 1],
323 (menu->item_count - index + 1) * sizeof(ITEM *));
324 menu->items = talloc_realloc(menu, menu->items, ITEM *,
325 menu->item_count + 1);
331 * pmenu_init - Allocate and initialize a new menu instance.
333 * Returns a pointer the the initialized struct pmenu instance or NULL on error.
334 * The caller is responsible for calling talloc_free() for the returned
338 struct pmenu *pmenu_init(void *ui_ctx, unsigned int item_count,
339 void (*on_exit)(struct pmenu *))
341 struct pmenu *menu = talloc_zero(ui_ctx, struct pmenu);
346 /* note items array has a null terminator */
348 menu->items = talloc_zero_array(menu, ITEM *, item_count + 1);
355 nc_scr_init(&menu->scr, pb_pmenu_sig, 0, ui_ctx, pmenu_process_key,
356 pmenu_post, pmenu_unpost, pmenu_resize);
358 menu->item_count = item_count;
359 menu->insert_pt = 0; /* insert from top */
360 menu->on_exit = on_exit;
366 * pmenu_setup - Create nc menu, setup nc windows.
370 int pmenu_setup(struct pmenu *menu)
374 menu->ncm = new_menu(menu->items);
377 pb_log("%s:%d: new_menu failed: %s\n", __func__, __LINE__,
382 set_menu_win(menu->ncm, menu->scr.main_ncw);
383 set_menu_sub(menu->ncm, menu->scr.sub_ncw);
385 /* Makes menu scrollable. */
386 set_menu_format(menu->ncm, LINES - nc_scr_frame_lines, 1);
392 * pmenu_delete - Delete a menu instance.
396 void pmenu_delete(struct pmenu *menu)
400 assert(menu->scr.sig == pb_pmenu_sig);
401 menu->scr.sig = pb_removed_sig;
403 for (i = item_count(menu->ncm); i; i--)
404 free_item(menu->items[i - 1]);
406 free_menu(menu->ncm);
407 delwin(menu->scr.sub_ncw);
408 delwin(menu->scr.main_ncw);