zori_menu.c 4.9 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166
  1. #include "monolog.h"
  2. #include "zori.h"
  3. #include "zori_widget.h"
  4. #include "zori_caption.h"
  5. #include "zori_menu.h"
  6. struct zori_menu * zori_widget_to_menu(struct zori_widget * widget) {
  7. if (!zori_widget_is_type(widget, ZORI_WIDGET_TYPE_MENU)) return NULL;
  8. return ZORI_CONTAINER_OF(widget, struct zori_menu, widget);
  9. }
  10. struct zori_widget * zori_menu_get_selected(struct zori_menu * menu) {
  11. if (!menu) return NULL;
  12. if (menu->selected_index < 0) return NULL;
  13. return miao_unsafe_get(&menu->widget.children, menu->selected_index);
  14. }
  15. int zori_menu_select_index(struct zori_menu * menu, int index) {
  16. struct zori_widget * selected = NULL;
  17. size_t size = miao_size(&menu->widget.children);
  18. if (size < 1) return ZORI_HANDLE_ERROR;
  19. if (index < 0) return ZORI_HANDLE_DONE;
  20. if ((size_t)index >= size) return ZORI_HANDLE_DONE;
  21. menu->selected_index = index;
  22. selected = zori_menu_get_selected(menu);
  23. zori_mark_widget(selected);
  24. return ZORI_HANDLE_DONE;
  25. }
  26. int zori_menu_select_previous(struct zori_menu * menu) {
  27. size_t size = miao_size(&menu->widget.children);
  28. int new_selection = menu->selected_index - 1;
  29. /* "roll over" */
  30. if (new_selection < 0) {
  31. new_selection = (int)size - 1;
  32. }
  33. return zori_menu_select_index(menu, new_selection);
  34. }
  35. int zori_menu_select_next(struct zori_menu * menu) {
  36. size_t size = miao_size(&menu->widget.children);
  37. int new_selection = menu->selected_index + 1;
  38. /* "roll over" */
  39. if ((size_t)new_selection >= size) {
  40. new_selection = 0;
  41. }
  42. return zori_menu_select_index(menu, new_selection);
  43. }
  44. int zori_menu_select_first(struct zori_menu * menu) {
  45. return zori_menu_select_index(menu, 0);
  46. }
  47. int zori_menu_activate_selected(struct zori_menu * menu) {
  48. struct zori_widget * selected = zori_menu_get_selected(menu);
  49. if (selected) {
  50. LOG_NOTE("Item selected: %p!\n", selected);
  51. return zori_widget_raise_internal_action_event(selected);
  52. }
  53. LOG_WARNING("No item selected!\n");
  54. return ZORI_HANDLE_IGNORE;
  55. }
  56. int zori_menu_close(struct zori_menu * menu) {
  57. struct zori_widget * parent;
  58. parent = menu->widget.parent;
  59. zori_widget_raise_close_event(parent, &menu->widget);
  60. zori_widget_active_(&menu->widget, false);
  61. zori_widget_visible_(&menu->widget, false);
  62. zori_widget_set_closed_result(&menu->widget, 1);
  63. return ZORI_HANDLE_DONE;
  64. }
  65. int zori_menu_on_child_close(union zori_event * event) {
  66. struct zori_widget * widget = event->any.widget;
  67. struct zori_menu * menu = zori_widget_to_menu(widget);
  68. zori_mark_widget(zori_menu_get_selected(menu));
  69. return ZORI_HANDLE_DONE;
  70. }
  71. int zori_menu_on_key_down(union zori_event * event) {
  72. struct zori_widget * widget = event->any.widget;
  73. struct zori_menu * menu = zori_widget_to_menu(widget);
  74. struct zori_widget * item = NULL;
  75. if (miao_out_of_bounds(&menu->widget.children, menu->selected_index)) {
  76. return ZORI_HANDLE_IGNORE;
  77. }
  78. item = miao_unsafe_get(&widget->children, menu->selected_index);
  79. switch (event->sys.ev->keyboard.keycode) {
  80. case ALLEGRO_KEY_UP:
  81. return zori_menu_select_previous(menu);
  82. case ALLEGRO_KEY_DOWN:
  83. return zori_menu_select_next(menu);
  84. case ALLEGRO_KEY_ENTER:
  85. case ALLEGRO_KEY_SPACE:
  86. case ALLEGRO_KEY_LCTRL:
  87. case ALLEGRO_KEY_RCTRL:
  88. return zori_menu_activate_selected(menu);
  89. case ALLEGRO_KEY_BACKSPACE:
  90. case ALLEGRO_KEY_LSHIFT:
  91. case ALLEGRO_KEY_RSHIFT:
  92. return zori_menu_close(menu);
  93. default:
  94. return ZORI_HANDLE_IGNORE;
  95. }
  96. return ZORI_HANDLE_IGNORE;
  97. }
  98. int zori_menu_on_draw(union zori_event * event) {
  99. struct zori_widget * widget = event->any.widget;
  100. if (zori_widget_visible(widget)) {
  101. zori_widget_draw_background(widget);
  102. }
  103. return zori_widget_must_draw_children(widget);
  104. }
  105. struct zori_handler zori_menu_handlers[] = {
  106. { ZORI_SYSTEM_EVENT_KEY_DOWN, zori_menu_on_key_down , NULL},
  107. { ZORI_EVENT_DRAW , zori_menu_on_draw , NULL},
  108. { ZORI_EVENT_CLOSE , zori_menu_on_child_close , NULL},
  109. };
  110. struct zori_menu * zori_menu_init(
  111. struct zori_menu * menu, zori_id id, zori_id parent_id,
  112. zori_rebox * box, struct zori_style * style
  113. ) {
  114. struct zori_widget * parent = zori_get_widget(parent_id);
  115. if (!menu) return NULL;
  116. menu->selected_index = 0;
  117. zori_widget_initall(&menu->widget, ZORI_WIDGET_TYPE_MENU, id, parent, box, style,
  118. ZORI_ARRAY_AND_AMOUNT( zori_menu_handlers));
  119. zori_widget_hover_(&menu->widget, false);
  120. return menu;
  121. }
  122. struct zori_menu * zori_menu_new(zori_id id, zori_id parent_id, zori_box * box) {
  123. struct zori_menu * menu = NULL;
  124. menu = calloc(1, sizeof(*menu));
  125. if (!menu) return NULL;
  126. if (!zori_menu_init(menu, id, parent_id, box, NULL)) {
  127. free(menu);
  128. return NULL;
  129. }
  130. return menu;
  131. }
  132. zori_id zori_new_menu(zori_id id, zori_id parent, zori_box * box) {
  133. struct zori_menu * menu = zori_menu_new(id, parent, box);
  134. if (!menu) return ZORI_ID_ENOMEM;
  135. return menu->widget.id;
  136. }