2014-04-09 22:09:35 +00:00
|
|
|
#include "internal.h"
|
|
|
|
#include <stdlib.h>
|
|
|
|
#include <string.h>
|
|
|
|
#include <ctype.h>
|
|
|
|
#include <assert.h>
|
|
|
|
|
2014-04-10 08:07:37 +00:00
|
|
|
/**
|
|
|
|
* Filter function map.
|
|
|
|
*/
|
2014-04-10 17:04:06 +00:00
|
|
|
static bmItem** (*filterFunc[BM_FILTER_MODE_LAST])(bmMenu *menu, unsigned int *outNmemb, unsigned int *outHighlighted) = {
|
2014-04-10 08:07:37 +00:00
|
|
|
_bmFilterDmenu, /* BM_FILTER_DMENU */
|
|
|
|
_bmFilterDmenuCaseInsensitive /* BM_FILTER_DMENU_CASE_INSENSITIVE */
|
2014-04-09 22:09:35 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
static void _bmMenuFilter(bmMenu *menu)
|
|
|
|
{
|
2014-04-09 22:41:32 +00:00
|
|
|
assert(menu);
|
2014-04-09 22:09:35 +00:00
|
|
|
|
|
|
|
unsigned int count, selected;
|
|
|
|
bmItem **filtered = filterFunc[menu->filterMode](menu, &count, &selected);
|
|
|
|
|
2014-04-10 17:04:06 +00:00
|
|
|
_bmItemListSetItemsNoCopy(&menu->filtered, filtered, count);
|
2014-04-09 22:09:35 +00:00
|
|
|
menu->index = selected;
|
|
|
|
}
|
|
|
|
|
2014-04-10 17:04:06 +00:00
|
|
|
int _bmMenuItemIsSelected(const bmMenu *menu, const bmItem *item)
|
2014-04-09 22:09:35 +00:00
|
|
|
{
|
2014-04-10 17:04:06 +00:00
|
|
|
assert(menu);
|
|
|
|
assert(item);
|
2014-04-09 22:09:35 +00:00
|
|
|
|
2014-04-10 17:04:06 +00:00
|
|
|
unsigned int i, count;
|
|
|
|
bmItem **items = bmMenuGetSelectedItems(menu, &count);
|
|
|
|
for (i = 0; i < count && items[i] != item; ++i);
|
|
|
|
return (i < count);
|
2014-04-09 22:09:35 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Create new bmMenu instance.
|
|
|
|
*
|
|
|
|
* @param drawMode Render method to be used for this menu instance.
|
2014-04-10 14:26:42 +00:00
|
|
|
* @return bmMenu for new menu instance, **NULL** if creation failed.
|
2014-04-09 22:09:35 +00:00
|
|
|
*/
|
|
|
|
bmMenu* bmMenuNew(bmDrawMode drawMode)
|
|
|
|
{
|
|
|
|
bmMenu *menu = calloc(1, sizeof(bmMenu));
|
|
|
|
|
|
|
|
menu->drawMode = drawMode;
|
|
|
|
|
|
|
|
if (!menu)
|
|
|
|
return NULL;
|
|
|
|
|
|
|
|
int status = 1;
|
|
|
|
|
|
|
|
switch (menu->drawMode) {
|
|
|
|
case BM_DRAW_MODE_CURSES:
|
|
|
|
status = _bmDrawCursesInit(&menu->renderApi);
|
|
|
|
break;
|
|
|
|
|
|
|
|
default:break;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (status == 0) {
|
|
|
|
bmMenuFree(menu);
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
return menu;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Release bmMenu instance.
|
|
|
|
*
|
|
|
|
* @param menu bmMenu instance to be freed from memory.
|
|
|
|
*/
|
|
|
|
void bmMenuFree(bmMenu *menu)
|
|
|
|
{
|
2014-04-09 22:41:32 +00:00
|
|
|
assert(menu);
|
2014-04-09 22:09:35 +00:00
|
|
|
|
|
|
|
if (menu->renderApi.free)
|
|
|
|
menu->renderApi.free();
|
|
|
|
|
|
|
|
if (menu->title)
|
|
|
|
free(menu->title);
|
|
|
|
|
2014-04-10 17:04:06 +00:00
|
|
|
if (menu->filtered.list)
|
|
|
|
free(menu->filtered.list);
|
2014-04-09 22:25:37 +00:00
|
|
|
|
2014-04-09 22:09:35 +00:00
|
|
|
bmMenuFreeItems(menu);
|
|
|
|
free(menu);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Release items inside bmMenu instance.
|
|
|
|
*
|
|
|
|
* @param menu bmMenu instance which items will be freed from memory.
|
|
|
|
*/
|
|
|
|
void bmMenuFreeItems(bmMenu *menu)
|
|
|
|
{
|
2014-04-09 22:41:32 +00:00
|
|
|
assert(menu);
|
2014-04-10 17:04:06 +00:00
|
|
|
_bmItemListFreeList(&menu->selection);
|
|
|
|
_bmItemListFreeList(&menu->filtered);
|
|
|
|
_bmItemListFreeItems(&menu->items);
|
2014-04-09 22:09:35 +00:00
|
|
|
}
|
|
|
|
|
2014-04-10 17:09:34 +00:00
|
|
|
/**
|
|
|
|
* Set userdata pointer to bmMenu instance.
|
|
|
|
* Userdata will be carried unmodified by the instance.
|
|
|
|
*
|
|
|
|
* @param menu bmMenu instance where to set userdata pointer.
|
|
|
|
* @param userdata Pointer to userdata.
|
|
|
|
*/
|
|
|
|
void bmMenuSetUserdata(bmMenu *menu, void *userdata)
|
|
|
|
{
|
|
|
|
assert(menu);
|
|
|
|
menu->userdata = userdata;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Get userdata pointer from bmMenu instance.
|
|
|
|
*
|
|
|
|
* @param menu bmMenu instance which userdata pointer to get.
|
|
|
|
* @return Pointer for unmodified userdata.
|
|
|
|
*/
|
|
|
|
void* bmMenuGetUserdata(bmMenu *menu)
|
|
|
|
{
|
|
|
|
assert(menu);
|
|
|
|
return menu->userdata;
|
|
|
|
}
|
|
|
|
|
2014-04-09 22:09:35 +00:00
|
|
|
/**
|
|
|
|
* Set active filter mode to bmMenu instance.
|
|
|
|
*
|
|
|
|
* @param menu bmMenu instance where to set filter mode.
|
|
|
|
* @param mode bmFilterMode constant.
|
|
|
|
*/
|
|
|
|
void bmMenuSetFilterMode(bmMenu *menu, bmFilterMode mode)
|
|
|
|
{
|
2014-04-09 22:41:32 +00:00
|
|
|
assert(menu);
|
2014-04-09 22:09:35 +00:00
|
|
|
|
|
|
|
bmFilterMode oldMode = mode;
|
|
|
|
menu->filterMode = (mode >= BM_FILTER_MODE_LAST ? BM_FILTER_MODE_DMENU : mode);
|
|
|
|
|
|
|
|
if (oldMode != mode)
|
|
|
|
_bmMenuFilter(menu);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Get active filter mode from bmMenu instance.
|
|
|
|
*
|
|
|
|
* @param menu bmMenu instance where to get filter mode.
|
|
|
|
* @return bmFilterMode constant.
|
|
|
|
*/
|
|
|
|
bmFilterMode bmMenuGetFilterMode(const bmMenu *menu)
|
|
|
|
{
|
2014-04-09 22:41:32 +00:00
|
|
|
assert(menu);
|
2014-04-09 22:09:35 +00:00
|
|
|
return menu->filterMode;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Set title to bmMenu instance.
|
|
|
|
*
|
|
|
|
* @param menu bmMenu instance where to set title.
|
2014-04-10 14:26:42 +00:00
|
|
|
* @param title C "string" to set as title, can be **NULL** for empty title.
|
2014-04-09 22:09:35 +00:00
|
|
|
*/
|
|
|
|
int bmMenuSetTitle(bmMenu *menu, const char *title)
|
|
|
|
{
|
2014-04-09 22:41:32 +00:00
|
|
|
assert(menu);
|
2014-04-09 22:09:35 +00:00
|
|
|
|
|
|
|
char *copy = NULL;
|
|
|
|
if (title && !(copy = _bmStrdup(title)))
|
|
|
|
return 0;
|
|
|
|
|
|
|
|
if (menu->title)
|
|
|
|
free(menu->title);
|
|
|
|
|
|
|
|
menu->title = copy;
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Get title from bmMenu instance.
|
|
|
|
*
|
|
|
|
* @param menu bmMenu instance where to get title from.
|
2014-04-10 14:26:42 +00:00
|
|
|
* @return Pointer to null terminated C "string", can be **NULL** for empty title.
|
2014-04-09 22:09:35 +00:00
|
|
|
*/
|
|
|
|
const char* bmMenuGetTitle(const bmMenu *menu)
|
|
|
|
{
|
2014-04-09 22:41:32 +00:00
|
|
|
assert(menu);
|
2014-04-09 22:09:35 +00:00
|
|
|
return menu->title;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Add item to bmMenu instance at specific index.
|
|
|
|
*
|
|
|
|
* @param menu bmMenu instance where item will be added.
|
|
|
|
* @param item bmItem instance to add.
|
|
|
|
* @param index Index where item will be added.
|
|
|
|
* @return 1 on successful add, 0 on failure.
|
|
|
|
*/
|
|
|
|
int bmMenuAddItemAt(bmMenu *menu, bmItem *item, unsigned int index)
|
|
|
|
{
|
2014-04-09 22:41:32 +00:00
|
|
|
assert(menu);
|
2014-04-10 17:04:06 +00:00
|
|
|
return _bmItemListAddItemAt(&menu->items, item, index);;
|
2014-04-09 22:09:35 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Add item to bmMenu instance.
|
|
|
|
*
|
|
|
|
* @param menu bmMenu instance where item will be added.
|
|
|
|
* @param item bmItem instance to add.
|
|
|
|
* @return 1 on successful add, 0 on failure.
|
|
|
|
*/
|
|
|
|
int bmMenuAddItem(bmMenu *menu, bmItem *item)
|
|
|
|
{
|
2014-04-10 17:04:06 +00:00
|
|
|
return _bmItemListAddItem(&menu->items, item);
|
2014-04-09 22:09:35 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Remove item from bmMenu instance at specific index.
|
|
|
|
*
|
2014-04-10 14:26:42 +00:00
|
|
|
* @warning The item won't be freed, use bmItemFree to do that.
|
|
|
|
*
|
2014-04-09 22:09:35 +00:00
|
|
|
* @param menu bmMenu instance from where item will be removed.
|
|
|
|
* @param index Index of item to remove.
|
|
|
|
* @return 1 on successful add, 0 on failure.
|
|
|
|
*/
|
|
|
|
int bmMenuRemoveItemAt(bmMenu *menu, unsigned int index)
|
|
|
|
{
|
2014-04-09 22:41:32 +00:00
|
|
|
assert(menu);
|
2014-04-09 22:09:35 +00:00
|
|
|
|
2014-04-10 17:04:06 +00:00
|
|
|
if (!menu->items.list || menu->items.count <= index)
|
2014-04-09 22:09:35 +00:00
|
|
|
return 0;
|
|
|
|
|
2014-04-10 17:04:06 +00:00
|
|
|
bmItem *item = menu->items.list[index];
|
|
|
|
int ret = _bmItemListRemoveItemAt(&menu->items, index);
|
|
|
|
|
|
|
|
if (ret) {
|
|
|
|
_bmItemListRemoveItem(&menu->selection, item);
|
|
|
|
_bmItemListRemoveItem(&menu->filtered, item);
|
|
|
|
}
|
|
|
|
|
|
|
|
return ret;
|
2014-04-09 22:09:35 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Remove item from bmMenu instance.
|
|
|
|
*
|
2014-04-10 14:26:42 +00:00
|
|
|
* @warning The item won't be freed, use bmItemFree to do that.
|
|
|
|
*
|
2014-04-09 22:09:35 +00:00
|
|
|
* @param menu bmMenu instance from where item will be removed.
|
|
|
|
* @param item bmItem instance to remove.
|
|
|
|
* @return 1 on successful add, 0 on failure.
|
|
|
|
*/
|
|
|
|
int bmMenuRemoveItem(bmMenu *menu, bmItem *item)
|
|
|
|
{
|
2014-04-09 22:41:32 +00:00
|
|
|
assert(menu);
|
2014-04-09 22:09:35 +00:00
|
|
|
|
2014-04-10 17:04:06 +00:00
|
|
|
int ret = _bmItemListRemoveItem(&menu->items, item);
|
|
|
|
|
|
|
|
if (ret) {
|
|
|
|
_bmItemListRemoveItem(&menu->selection, item);
|
|
|
|
_bmItemListRemoveItem(&menu->filtered, item);
|
|
|
|
}
|
|
|
|
|
|
|
|
return ret;
|
2014-04-09 22:09:35 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
2014-04-10 17:04:06 +00:00
|
|
|
* Highlight item in menu by index.
|
2014-04-09 22:09:35 +00:00
|
|
|
*
|
2014-04-10 17:04:06 +00:00
|
|
|
* @param menu bmMenu instance from where to highlight item.
|
2014-04-10 17:12:42 +00:00
|
|
|
* @param index Index of item to highlight.
|
2014-04-10 17:04:06 +00:00
|
|
|
* @return 1 on successful highlight, 0 on failure.
|
|
|
|
*/
|
|
|
|
int bmMenuSetHighlightedIndex(bmMenu *menu, unsigned int index)
|
|
|
|
{
|
|
|
|
assert(menu);
|
|
|
|
unsigned int itemsCount = (menu->filtered.list ? menu->filtered.count : menu->items.count);
|
|
|
|
|
|
|
|
if (itemsCount <= index)
|
|
|
|
return 0;
|
|
|
|
|
|
|
|
return (menu->index = index);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Highlight item in menu.
|
2014-04-10 14:26:42 +00:00
|
|
|
*
|
2014-04-10 17:04:06 +00:00
|
|
|
* @param menu bmMenu instance from where to highlight item.
|
|
|
|
* @param item bmItem instance to highlight.
|
|
|
|
* @return 1 on successful highlight, 0 on failure.
|
|
|
|
*/
|
|
|
|
int bmMenuSetHighlighted(bmMenu *menu, bmItem *item)
|
|
|
|
{
|
|
|
|
assert(menu);
|
|
|
|
|
|
|
|
unsigned int i, itemsCount;
|
|
|
|
bmItem **items = bmMenuGetFilteredItems(menu, &itemsCount);
|
|
|
|
for (i = 0; i < itemsCount && items[i] != item; ++i);
|
|
|
|
|
|
|
|
if (itemsCount <= i)
|
|
|
|
return 0;
|
|
|
|
|
|
|
|
return (menu->index = i);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
2014-04-10 17:11:41 +00:00
|
|
|
* Get highlighted item from bmMenu instance.
|
2014-04-10 17:04:06 +00:00
|
|
|
*
|
2014-04-10 17:19:15 +00:00
|
|
|
* @warning The pointer returned by this function may be invalid after items change.
|
|
|
|
*
|
2014-04-10 17:11:41 +00:00
|
|
|
* @param menu bmMenu instance from where to get highlighted item.
|
|
|
|
* @return Selected bmItem instance, **NULL** if none highlighted.
|
2014-04-09 22:09:35 +00:00
|
|
|
*/
|
2014-04-10 17:11:41 +00:00
|
|
|
bmItem* bmMenuGetHighlightedItem(const bmMenu *menu)
|
2014-04-09 22:09:35 +00:00
|
|
|
{
|
2014-04-09 22:41:32 +00:00
|
|
|
assert(menu);
|
2014-04-10 17:11:41 +00:00
|
|
|
|
|
|
|
unsigned int count;
|
|
|
|
bmItem **items = bmMenuGetFilteredItems(menu, &count);
|
|
|
|
|
|
|
|
if (!items || count <= menu->index)
|
|
|
|
return NULL;
|
|
|
|
|
|
|
|
return items[menu->index];
|
2014-04-10 17:04:06 +00:00
|
|
|
}
|
2014-04-09 22:09:35 +00:00
|
|
|
|
2014-04-10 17:04:06 +00:00
|
|
|
/**
|
|
|
|
* Set selected items to bmMenu instance.
|
|
|
|
*
|
2014-04-10 17:19:15 +00:00
|
|
|
* @warning The list won't be copied, do not free it.
|
2014-04-10 17:04:06 +00:00
|
|
|
*
|
|
|
|
* @param menu bmMenu instance where items will be set.
|
|
|
|
* @param items Array of bmItem pointers to set.
|
|
|
|
* @param nmemb Total count of items in array.
|
|
|
|
* @return 1 on successful set, 0 on failure.
|
|
|
|
*/
|
|
|
|
int bmMenuSetSelectedItems(bmMenu *menu, bmItem **items, unsigned int nmemb)
|
|
|
|
{
|
|
|
|
assert(menu);
|
|
|
|
return _bmItemListSetItemsNoCopy(&menu->selection, items, nmemb);
|
2014-04-09 22:09:35 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
2014-04-10 17:11:41 +00:00
|
|
|
* Get selected items from bmMenu instance.
|
2014-04-09 22:09:35 +00:00
|
|
|
*
|
2014-04-10 17:19:15 +00:00
|
|
|
* @warning The pointer returned by this function may be invalid after selection or items change.
|
|
|
|
*
|
2014-04-10 17:11:41 +00:00
|
|
|
* @param menu bmMenu instance from where to get selected items.
|
2014-04-10 14:26:42 +00:00
|
|
|
* @param outNmemb Reference to unsigned int where total count of returned items will be stored.
|
2014-04-09 22:09:35 +00:00
|
|
|
* @return Pointer to array of bmItem pointers.
|
|
|
|
*/
|
2014-04-10 17:11:41 +00:00
|
|
|
bmItem** bmMenuGetSelectedItems(const bmMenu *menu, unsigned int *outNmemb)
|
2014-04-09 22:09:35 +00:00
|
|
|
{
|
2014-04-09 22:41:32 +00:00
|
|
|
assert(menu);
|
2014-04-10 17:11:41 +00:00
|
|
|
return _bmItemListGetItems(&menu->selection, outNmemb);
|
|
|
|
}
|
2014-04-09 22:09:35 +00:00
|
|
|
|
2014-04-10 17:11:41 +00:00
|
|
|
/**
|
|
|
|
* Set items to bmMenu instance.
|
|
|
|
* Will replace all the old items on bmMenu instance.
|
|
|
|
*
|
|
|
|
* If items is **NULL**, or nmemb is zero, all items will be freed from the menu.
|
|
|
|
*
|
|
|
|
* @param menu bmMenu instance where items will be set.
|
|
|
|
* @param items Array of bmItem pointers to set.
|
|
|
|
* @param nmemb Total count of items in array.
|
|
|
|
* @return 1 on successful set, 0 on failure.
|
|
|
|
*/
|
|
|
|
int bmMenuSetItems(bmMenu *menu, const bmItem **items, unsigned int nmemb)
|
|
|
|
{
|
|
|
|
assert(menu);
|
2014-04-09 22:09:35 +00:00
|
|
|
|
2014-04-10 17:11:41 +00:00
|
|
|
int ret = _bmItemListSetItems(&menu->items, items, nmemb);
|
|
|
|
|
|
|
|
if (ret) {
|
|
|
|
_bmItemListFreeList(&menu->selection);
|
|
|
|
_bmItemListFreeList(&menu->filtered);
|
|
|
|
}
|
|
|
|
|
|
|
|
return ret;
|
2014-04-10 17:04:06 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Get items from bmMenu instance.
|
|
|
|
*
|
|
|
|
* @warning The pointer returned by this function may be invalid after removing or adding new items.
|
|
|
|
*
|
|
|
|
* @param menu bmMenu instance from where to get items.
|
|
|
|
* @param outNmemb Reference to unsigned int where total count of returned items will be stored.
|
|
|
|
* @return Pointer to array of bmItem pointers.
|
|
|
|
*/
|
|
|
|
bmItem** bmMenuGetItems(const bmMenu *menu, unsigned int *outNmemb)
|
|
|
|
{
|
|
|
|
assert(menu);
|
|
|
|
return _bmItemListGetItems(&menu->items, outNmemb);
|
2014-04-09 22:09:35 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
2014-04-10 17:11:41 +00:00
|
|
|
* Get filtered (displayed) items from bmMenu instance.
|
2014-04-09 22:09:35 +00:00
|
|
|
*
|
2014-04-10 17:11:41 +00:00
|
|
|
* @warning The pointer returned by this function _will_ be invalid when menu internally filters its list again.
|
|
|
|
* Do not store this pointer.
|
2014-04-10 14:26:42 +00:00
|
|
|
*
|
2014-04-10 17:11:41 +00:00
|
|
|
* @param menu bmMenu instance from where to get filtered items.
|
|
|
|
* @param outNmemb Reference to unsigned int where total count of returned items will be stored.
|
|
|
|
* @return Pointer to array of bmItem pointers.
|
2014-04-09 22:09:35 +00:00
|
|
|
*/
|
2014-04-10 17:11:41 +00:00
|
|
|
bmItem** bmMenuGetFilteredItems(const bmMenu *menu, unsigned int *outNmemb)
|
2014-04-09 22:09:35 +00:00
|
|
|
{
|
2014-04-09 22:41:32 +00:00
|
|
|
assert(menu);
|
2014-04-09 22:09:35 +00:00
|
|
|
|
2014-04-10 17:11:41 +00:00
|
|
|
if (menu->filtered.list)
|
|
|
|
return _bmItemListGetItems(&menu->filtered, outNmemb);
|
2014-04-09 22:09:35 +00:00
|
|
|
|
2014-04-10 17:11:41 +00:00
|
|
|
return _bmItemListGetItems(&menu->items, outNmemb);
|
2014-04-09 22:09:35 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
2014-04-10 10:08:26 +00:00
|
|
|
* Render bmMenu instance using chosen draw method.
|
2014-04-09 22:09:35 +00:00
|
|
|
*
|
2014-04-10 10:08:26 +00:00
|
|
|
* @param menu bmMenu instance to be rendered.
|
2014-04-09 22:09:35 +00:00
|
|
|
*/
|
|
|
|
void bmMenuRender(const bmMenu *menu)
|
|
|
|
{
|
2014-04-09 22:41:32 +00:00
|
|
|
assert(menu);
|
2014-04-09 22:09:35 +00:00
|
|
|
|
|
|
|
if (menu->renderApi.render)
|
|
|
|
menu->renderApi.render(menu);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Poll key and unicode from underlying UI toolkit.
|
|
|
|
*
|
2014-04-10 14:26:42 +00:00
|
|
|
* This function will block on @link ::bmDrawMode BM_DRAW_MODE_CURSES @endlink draw mode.
|
2014-04-09 22:09:35 +00:00
|
|
|
*
|
|
|
|
* @param menu bmMenu instance from which to poll.
|
2014-04-10 14:26:42 +00:00
|
|
|
* @param outUnicode Reference to unsigned int.
|
2014-04-09 22:09:35 +00:00
|
|
|
* @return bmKey for polled key.
|
|
|
|
*/
|
2014-04-10 14:26:42 +00:00
|
|
|
bmKey bmMenuGetKey(bmMenu *menu, unsigned int *outUnicode)
|
2014-04-09 22:09:35 +00:00
|
|
|
{
|
2014-04-09 22:41:32 +00:00
|
|
|
assert(menu);
|
2014-04-10 14:26:42 +00:00
|
|
|
assert(outUnicode);
|
2014-04-09 22:09:35 +00:00
|
|
|
|
2014-04-10 14:26:42 +00:00
|
|
|
*outUnicode = 0;
|
2014-04-09 22:09:35 +00:00
|
|
|
bmKey key = BM_KEY_NONE;
|
|
|
|
|
|
|
|
if (menu->renderApi.getKey)
|
2014-04-10 14:26:42 +00:00
|
|
|
key = menu->renderApi.getKey(outUnicode);
|
2014-04-09 22:09:35 +00:00
|
|
|
|
|
|
|
return key;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Advances menu logic with key and unicode as input.
|
|
|
|
*
|
|
|
|
* @param menu bmMenu instance to be advanced.
|
2014-04-10 10:08:38 +00:00
|
|
|
* @param key Key input that will advance menu logic.
|
|
|
|
* @param unicode Unicode input that will advance menu logic.
|
2014-04-09 22:09:35 +00:00
|
|
|
* @return bmRunResult for menu state.
|
|
|
|
*/
|
|
|
|
bmRunResult bmMenuRunWithKey(bmMenu *menu, bmKey key, unsigned int unicode)
|
|
|
|
{
|
2014-04-09 22:41:32 +00:00
|
|
|
assert(menu);
|
2014-04-09 22:09:35 +00:00
|
|
|
char *oldFilter = _bmStrdup(menu->filter);
|
2014-04-10 17:04:06 +00:00
|
|
|
unsigned int itemsCount = (menu->filtered.list ? menu->filtered.count : menu->items.count);
|
2014-04-09 22:09:35 +00:00
|
|
|
|
|
|
|
switch (key) {
|
|
|
|
case BM_KEY_LEFT:
|
|
|
|
{
|
|
|
|
unsigned int oldCursor = menu->cursor;
|
|
|
|
menu->cursor -= _bmUtf8RunePrev(menu->filter, menu->cursor);
|
|
|
|
menu->cursesCursor -= _bmUtf8RuneWidth(menu->filter + menu->cursor, oldCursor - menu->cursor);
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
|
|
|
|
case BM_KEY_RIGHT:
|
|
|
|
{
|
|
|
|
unsigned int oldCursor = menu->cursor;
|
|
|
|
menu->cursor += _bmUtf8RuneNext(menu->filter, menu->cursor);
|
|
|
|
menu->cursesCursor += _bmUtf8RuneWidth(menu->filter + oldCursor, menu->cursor - oldCursor);
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
|
|
|
|
case BM_KEY_HOME:
|
|
|
|
menu->cursesCursor = menu->cursor = 0;
|
|
|
|
break;
|
|
|
|
|
|
|
|
case BM_KEY_END:
|
|
|
|
menu->cursor = strlen(menu->filter);
|
|
|
|
menu->cursesCursor = _bmUtf8StringScreenWidth(menu->filter);
|
|
|
|
break;
|
|
|
|
|
|
|
|
case BM_KEY_UP:
|
|
|
|
if (menu->index > 0)
|
|
|
|
menu->index--;
|
|
|
|
break;
|
|
|
|
|
|
|
|
case BM_KEY_DOWN:
|
|
|
|
if (menu->index < itemsCount - 1)
|
|
|
|
menu->index++;
|
|
|
|
break;
|
|
|
|
|
|
|
|
case BM_KEY_PAGE_UP:
|
|
|
|
menu->index = 0;
|
|
|
|
break;
|
|
|
|
|
|
|
|
case BM_KEY_PAGE_DOWN:
|
|
|
|
menu->index = itemsCount - 1;
|
|
|
|
break;
|
|
|
|
|
|
|
|
case BM_KEY_BACKSPACE:
|
|
|
|
{
|
|
|
|
size_t width;
|
|
|
|
menu->cursor -= _bmUtf8RuneRemove(menu->filter, menu->cursor, &width);
|
|
|
|
menu->cursesCursor -= width;
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
|
|
|
|
case BM_KEY_DELETE:
|
|
|
|
_bmUtf8RuneRemove(menu->filter, menu->cursor + 1, NULL);
|
|
|
|
break;
|
|
|
|
|
|
|
|
case BM_KEY_LINE_DELETE_LEFT:
|
|
|
|
{
|
|
|
|
while (menu->cursor > 0) {
|
|
|
|
size_t width;
|
|
|
|
menu->cursor -= _bmUtf8RuneRemove(menu->filter, menu->cursor, &width);
|
|
|
|
menu->cursesCursor -= width;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
|
|
|
|
case BM_KEY_LINE_DELETE_RIGHT:
|
|
|
|
menu->filter[menu->cursor] = 0;
|
|
|
|
break;
|
|
|
|
|
|
|
|
case BM_KEY_WORD_DELETE:
|
|
|
|
{
|
|
|
|
while (menu->cursor < strlen(menu->filter) && !isspace(menu->filter[menu->cursor])) {
|
|
|
|
unsigned int oldCursor = menu->cursor;
|
|
|
|
menu->cursor += _bmUtf8RuneNext(menu->filter, menu->cursor);
|
|
|
|
menu->cursesCursor += _bmUtf8RuneWidth(menu->filter + oldCursor, menu->cursor - oldCursor);
|
|
|
|
}
|
|
|
|
while (menu->cursor > 0 && isspace(menu->filter[menu->cursor - 1])) {
|
|
|
|
unsigned int oldCursor = menu->cursor;
|
|
|
|
menu->cursor -= _bmUtf8RunePrev(menu->filter, menu->cursor);
|
|
|
|
menu->cursesCursor -= _bmUtf8RuneWidth(menu->filter + menu->cursor, oldCursor - menu->cursor);
|
|
|
|
}
|
|
|
|
while (menu->cursor > 0 && !isspace(menu->filter[menu->cursor - 1])) {
|
|
|
|
size_t width;
|
|
|
|
menu->cursor -= _bmUtf8RuneRemove(menu->filter, menu->cursor, &width);
|
|
|
|
menu->cursesCursor -= width;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
|
|
|
|
case BM_KEY_UNICODE:
|
|
|
|
{
|
|
|
|
size_t width;
|
|
|
|
menu->cursor += _bmUnicodeInsert(menu->filter, sizeof(menu->filter) - 1, menu->cursor, unicode, &width);
|
|
|
|
menu->cursesCursor += width;
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
|
|
|
|
case BM_KEY_TAB:
|
|
|
|
{
|
2014-04-10 17:04:06 +00:00
|
|
|
bmItem *highlighted = bmMenuGetHighlightedItem(menu);
|
|
|
|
if (highlighted && bmItemGetText(highlighted)) {
|
|
|
|
const char *text = bmItemGetText(highlighted);
|
2014-04-09 22:09:35 +00:00
|
|
|
size_t len = strlen(text);
|
|
|
|
|
|
|
|
if (len > sizeof(menu->filter) - 1)
|
|
|
|
len = sizeof(menu->filter) - 1;
|
|
|
|
|
|
|
|
memset(menu->filter, 0, strlen(menu->filter));
|
|
|
|
memcpy(menu->filter, text, len);
|
|
|
|
menu->cursor = strlen(menu->filter);
|
|
|
|
menu->cursesCursor = _bmUtf8StringScreenWidth(menu->filter);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
|
2014-04-10 17:04:06 +00:00
|
|
|
case BM_KEY_SHIFT_RETURN:
|
|
|
|
case BM_KEY_RETURN:
|
|
|
|
{
|
|
|
|
bmItem *highlighted = bmMenuGetHighlightedItem(menu);
|
|
|
|
if (highlighted && !_bmMenuItemIsSelected(menu, highlighted))
|
|
|
|
_bmItemListAddItem(&menu->selection, highlighted);
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
|
|
|
|
case BM_KEY_ESCAPE:
|
|
|
|
_bmItemListFreeList(&menu->selection);
|
|
|
|
break;
|
|
|
|
|
2014-04-09 22:09:35 +00:00
|
|
|
default: break;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (oldFilter && strcmp(oldFilter, menu->filter))
|
|
|
|
_bmMenuFilter(menu);
|
|
|
|
|
|
|
|
if (oldFilter)
|
|
|
|
free(oldFilter);
|
2014-04-09 22:26:00 +00:00
|
|
|
|
|
|
|
switch (key) {
|
|
|
|
case BM_KEY_RETURN: return BM_RUN_RESULT_SELECTED;
|
|
|
|
case BM_KEY_ESCAPE: return BM_RUN_RESULT_CANCEL;
|
|
|
|
default: break;
|
|
|
|
}
|
|
|
|
|
2014-04-09 22:09:35 +00:00
|
|
|
return BM_RUN_RESULT_RUNNING;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* vim: set ts=8 sw=4 tw=0 :*/
|