zori_menu.c 5.0 KB

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