194 lines
6.3 KiB
C
194 lines
6.3 KiB
C
#include "menu_api.h"
|
|
#include "../src/core/menu_core.h"
|
|
#include "../src/core/menu_event.h"
|
|
#include "../src/core/menu_stack.h"
|
|
#include "../src/core/menu_permission.h"
|
|
#include "../src/core/menu_persistence.h"
|
|
#include <stddef.h>
|
|
|
|
static MenuCoreCtx g_menu_ctx;
|
|
|
|
MenuErrCode menu_init(void) {
|
|
return menu_core_init(&g_menu_ctx);
|
|
}
|
|
|
|
void menu_deinit(void) {
|
|
// Cleanup if needed
|
|
}
|
|
|
|
void menu_main_loop(uint32_t tick) {
|
|
menu_core_loop(&g_menu_ctx, tick);
|
|
}
|
|
|
|
MenuNodeId menu_register_node(
|
|
MenuNodeId id,
|
|
MenuNodeId parent_id,
|
|
const char* name,
|
|
MenuCallback enter_cb,
|
|
MenuCallback exit_cb
|
|
) {
|
|
return menu_register_node_ex(id, parent_id, name, enter_cb, exit_cb, NULL, NULL);
|
|
}
|
|
|
|
MenuNodeId menu_register_node_ex(
|
|
MenuNodeId id,
|
|
MenuNodeId parent_id,
|
|
const char* name,
|
|
MenuCallback enter_cb,
|
|
MenuCallback exit_cb,
|
|
MenuCallback render_cb,
|
|
void* user_data
|
|
) {
|
|
MenuNode node = {0};
|
|
node.id = id;
|
|
node.parent_id = parent_id;
|
|
node.name = name;
|
|
node.enter_cb = enter_cb;
|
|
node.exit_cb = exit_cb;
|
|
node.render_cb = render_cb;
|
|
node.user_data = user_data;
|
|
node.flags.is_enabled = true;
|
|
node.flags.is_visible = true;
|
|
node.flags.is_registered = false; // Will be set by core
|
|
node.param_id = 0;
|
|
node.permission_level = 0;
|
|
|
|
MenuNodeId new_id = 0;
|
|
if (menu_core_register_node(&g_menu_ctx, &node, &new_id) == MENU_ERR_OK) {
|
|
return new_id;
|
|
}
|
|
return 0;
|
|
}
|
|
|
|
MenuErrCode menu_register_nodes(
|
|
const MenuNode* nodes,
|
|
size_t count,
|
|
MenuNodeId* out_ids
|
|
) {
|
|
if (!nodes || count == 0) return MENU_ERR_INVALID_PARAM;
|
|
|
|
for (size_t i = 0; i < count; i++) {
|
|
MenuNode node = nodes[i];
|
|
MenuNodeId new_id = 0;
|
|
|
|
node.flags.is_enabled = true;
|
|
node.flags.is_visible = true;
|
|
node.flags.is_registered = false;
|
|
node.param_id = node.param_id ? node.param_id : 0;
|
|
node.permission_level = node.permission_level ? node.permission_level : 0;
|
|
|
|
MenuErrCode err = menu_core_register_node(&g_menu_ctx, &node, &new_id);
|
|
if (err != MENU_ERR_OK) {
|
|
return err;
|
|
}
|
|
|
|
if (out_ids) {
|
|
out_ids[i] = new_id;
|
|
}
|
|
}
|
|
|
|
return MENU_ERR_OK;
|
|
}
|
|
|
|
MenuErrCode menu_post_event(MenuEventType type, uint32_t param, MenuEventPriority priority) {
|
|
return menu_event_post(&g_menu_ctx.event_queue, type, param, priority);
|
|
}
|
|
|
|
MenuErrCode menu_post_event_normal(MenuEventType type, uint32_t param) {
|
|
return menu_event_post(&g_menu_ctx.event_queue, type, param, MENU_EVENT_PRIORITY_NORMAL);
|
|
}
|
|
|
|
MenuErrCode menu_enter(void) {
|
|
// We try to enter the first available root node.
|
|
// We can trigger an event or just let the loop handle it.
|
|
// If we rely on handle_event to enter root on NULL current,
|
|
// we just need to ensure loop is called.
|
|
// But explicitly:
|
|
if (g_menu_ctx.current_node_id == 0) {
|
|
// Force finding a root
|
|
// Since we don't have public access to ctx nodes from here easily without exposing everything,
|
|
// we can just call handle_event with a dummy event or NONE event if handle_event checks NULL current.
|
|
// But handle_event checks event type.
|
|
// We can add MENU_EVENT_NONE handling in handle_event to check entry?
|
|
// Or just manually find root here.
|
|
// We have access to g_menu_ctx.
|
|
for(int i=0; i<MENU_CONFIG_MAX_NODES; i++) {
|
|
if(g_menu_ctx.nodes[i].flags.is_registered && g_menu_ctx.nodes[i].parent_id == 0) {
|
|
// Enter root
|
|
MenuErrCode ret = menu_core_enter_node(&g_menu_ctx, g_menu_ctx.nodes[i].id);
|
|
if (ret != MENU_ERR_OK) return ret;
|
|
|
|
// Optimization: Automatically enter first child if root has children
|
|
// This makes the menu start with the first item selected
|
|
if (g_menu_ctx.nodes[i].first_child_id != 0) {
|
|
// We need to simulate Enter key or just manually transition
|
|
// Manual transition:
|
|
// Push root to stack
|
|
// But menu_core_enter_node logic doesn't push to stack automatically unless via handle_event
|
|
// So we mimic handle_event logic
|
|
menu_stack_push(&g_menu_ctx.stack, g_menu_ctx.nodes[i].id);
|
|
// Exit root (deselect)
|
|
menu_core_exit_node(&g_menu_ctx, g_menu_ctx.nodes[i].id);
|
|
// Enter child
|
|
return menu_core_enter_node(&g_menu_ctx, g_menu_ctx.nodes[i].first_child_id);
|
|
}
|
|
return MENU_ERR_OK;
|
|
}
|
|
}
|
|
}
|
|
return MENU_ERR_OK;
|
|
}
|
|
|
|
MenuErrCode menu_back(void) {
|
|
return menu_post_event_normal(MENU_EVENT_KEY_ESC, 0);
|
|
}
|
|
|
|
MenuErrCode menu_node_bind_param(MenuNodeId node_id, uint16_t param_id) {
|
|
MenuNode* node = menu_core_get_node(&g_menu_ctx, node_id);
|
|
if (!node) return MENU_ERR_NOT_FOUND;
|
|
node->param_id = param_id;
|
|
return MENU_ERR_OK;
|
|
}
|
|
|
|
// Permission management functions
|
|
MenuErrCode menu_permission_register_role(uint8_t role_id, const char* name, MenuPermissionLevel level) {
|
|
return menu_permission_register_role_impl(&g_menu_ctx, role_id, name, level);
|
|
}
|
|
|
|
MenuErrCode menu_permission_set_current_role(uint8_t role_id) {
|
|
return menu_permission_set_current_role_impl(&g_menu_ctx, role_id);
|
|
}
|
|
|
|
MenuErrCode menu_permission_update_node_level(MenuNodeId node_id, MenuPermissionLevel level) {
|
|
return menu_permission_update_node_level_impl(&g_menu_ctx, node_id, level);
|
|
}
|
|
|
|
bool menu_permission_check_node_access(MenuNodeId node_id) {
|
|
return menu_permission_check_node_access_impl(&g_menu_ctx, node_id);
|
|
}
|
|
|
|
// Persistence functions
|
|
MenuErrCode menu_persistence_save(void) {
|
|
return menu_persistence_save_impl(&g_menu_ctx);
|
|
}
|
|
|
|
MenuErrCode menu_persistence_load(void) {
|
|
return menu_persistence_load_impl(&g_menu_ctx);
|
|
}
|
|
|
|
MenuErrCode menu_persistence_clear(void) {
|
|
return menu_persistence_clear_impl(&g_menu_ctx);
|
|
}
|
|
|
|
bool menu_persistence_is_dirty(void) {
|
|
return menu_persistence_is_dirty_impl(&g_menu_ctx);
|
|
}
|
|
|
|
MenuErrCode menu_refresh(void) {
|
|
MenuErrCode err = menu_post_event_normal(MENU_EVENT_RENDER, 0);
|
|
if (err != MENU_ERR_OK) {
|
|
return err;
|
|
}
|
|
return MENU_ERR_OK;
|
|
}
|