menu.c 6.2 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224
  1. #include <rtgui/dc.h>
  2. #include <rtgui/widgets/menu.h>
  3. #include <rtgui/rtgui_theme.h>
  4. #define RTGUI_MENU_IMAGE_MAGIN 18
  5. #define RTGUI_MENU_SUBMENU_MAGIN 16
  6. static void rtgui_menu_item_unselect(struct rtgui_menu_item* item);
  7. static rt_bool_t rtgui_menu_on_deactivate(rtgui_widget_t* widget, rtgui_event_t* event);
  8. const static rt_uint8_t right_arrow[] = {0x80, 0xc0, 0xe0, 0xf0, 0xe0, 0xc0, 0x80};
  9. static void _rtgui_menu_constructor(rtgui_menu_t *menu)
  10. {
  11. /* set window style */
  12. RTGUI_WIN(menu)->style = RTGUI_WIN_STYLE_NO_TITLE;
  13. /* set deactivate handler */
  14. rtgui_win_set_ondeactivate(RTGUI_WIN(menu), rtgui_menu_on_deactivate);
  15. /* set proper of control */
  16. menu->parent_menu = RT_NULL;
  17. menu->sub_menu = RT_NULL;
  18. menu->items = RT_NULL;
  19. menu->items_count = 0;
  20. menu->items_list = RT_NULL;
  21. menu->on_menupop = RT_NULL;
  22. menu->on_menuhide = RT_NULL;
  23. }
  24. static void _rtgui_menu_destructor(rtgui_menu_t* menu)
  25. {
  26. }
  27. static void _rtgui_menu_onitem(struct rtgui_widget* widget, struct rtgui_event* event)
  28. {
  29. struct rtgui_menu* menu;
  30. /* get menu */
  31. menu = RTGUI_MENU(rtgui_widget_get_toplevel(widget));
  32. if (menu->items[menu->items_list->current_item].type == RTGUI_ITEM_SUBMENU)
  33. {
  34. const rtgui_menu_item_t* items;
  35. rt_uint16_t count;
  36. items = menu->items[menu->items_list->current_item].submenu;
  37. count = menu->items[menu->items_list->current_item].submenu_count;
  38. menu->sub_menu = rtgui_menu_create("submenu", menu, items, count);
  39. rtgui_menu_pop(menu, 10, 10);
  40. }
  41. else /* other menu item */
  42. {
  43. rtgui_win_hiden(RTGUI_WIN(menu));
  44. }
  45. }
  46. static void _rtgui_menu_item_ondraw(struct rtgui_listctrl *list, struct rtgui_dc* dc, rtgui_rect_t* rect, rt_uint16_t index)
  47. {
  48. rtgui_rect_t item_rect;
  49. struct rtgui_menu_item* item;
  50. item_rect = *rect;
  51. item_rect.x1 += 5;
  52. /* re-fill item */
  53. if (list->current_item == index)
  54. {
  55. rtgui_color_t bc;
  56. bc = RTGUI_WIDGET_BACKGROUND(RTGUI_WIDGET(list));
  57. RTGUI_WIDGET_BACKGROUND(RTGUI_WIDGET(list)) = blue;
  58. rtgui_dc_fill_rect(dc, rect);
  59. RTGUI_WIDGET_BACKGROUND(RTGUI_WIDGET(list)) = bc;
  60. }
  61. /* get menu item */
  62. item = (rtgui_menu_item_t*)list->items;
  63. item = &item[index];
  64. if (item->type == RTGUI_ITEM_SUBMENU)
  65. {
  66. rtgui_rect_t r = {0, 0, 8, 8};
  67. rtgui_dc_draw_text(dc, item->label, &item_rect);
  68. item_rect.x1 = item_rect.x2 - 16; item_rect.x2 -= 8;
  69. rtgui_rect_moveto_align(&item_rect, &r, RTGUI_ALIGN_CENTER_HORIZONTAL | RTGUI_ALIGN_CENTER_VERTICAL);
  70. rtgui_dc_draw_byte(dc, r.x1, r.y1, 8, right_arrow);
  71. }
  72. else if (item->type == RTGUI_ITEM_SEPARATOR)
  73. {
  74. rtgui_dc_draw_horizontal_line(dc, item_rect.x1, item_rect.x2, (item_rect.y2 + item_rect.y1)/2);
  75. }
  76. else if (item->type == RTGUI_ITEM_CHECK)
  77. {
  78. /* not support right now */
  79. }
  80. else
  81. {
  82. /* normal menu item */
  83. rtgui_dc_draw_text(dc, item->label, &item_rect);
  84. if (item->image != RT_NULL)
  85. rtgui_image_blit(item->image, dc, &item_rect);
  86. }
  87. }
  88. static void _rtgui_menu_item_onmouse()
  89. {
  90. }
  91. rtgui_type_t *rtgui_menu_type_get(void)
  92. {
  93. static rtgui_type_t *menu_type = RT_NULL;
  94. if (!menu_type)
  95. {
  96. menu_type = rtgui_type_create("menu", RTGUI_WIN_TYPE,
  97. sizeof(rtgui_menu_t),
  98. RTGUI_CONSTRUCTOR(_rtgui_menu_constructor),
  99. RTGUI_DESTRUCTOR (_rtgui_menu_destructor));
  100. }
  101. return menu_type;
  102. }
  103. static rt_bool_t rtgui_menu_on_deactivate(rtgui_widget_t* widget, rtgui_event_t* event)
  104. {
  105. rtgui_menu_t* menu = (rtgui_menu_t*) widget;
  106. /* submenu is activate */
  107. if (menu->items[menu->items_list->current_item].type == RTGUI_ITEM_SUBMENU)
  108. {
  109. /* if sub menu activated, not hide menu */
  110. }
  111. rtgui_win_hiden(RTGUI_WIN(menu));
  112. if (menu->on_menuhide != RT_NULL)
  113. {
  114. menu->on_menuhide(RTGUI_WIDGET(menu), RT_NULL);
  115. }
  116. /* if it's a submenu, try to hide parent menu */
  117. if (menu->parent_menu != RT_NULL &&
  118. !rtgui_win_is_activated(RTGUI_WIN(menu->parent_menu)))
  119. {
  120. rtgui_menu_on_deactivate(RTGUI_WIDGET(menu->parent_menu), event);
  121. }
  122. return RT_TRUE;
  123. }
  124. struct rtgui_menu* rtgui_menu_create(const char* title, struct rtgui_menu* parent_menu,
  125. const struct rtgui_menu_item* items, rt_uint16_t count)
  126. {
  127. rtgui_rect_t rect = {0, 0, 100, 100};
  128. struct rtgui_menu* menu;
  129. menu = (struct rtgui_menu*) rtgui_widget_create ( RTGUI_MENU_TYPE );
  130. if (menu != RT_NULL)
  131. {
  132. rtgui_win_set_title(RTGUI_WIN(menu), title);
  133. menu->parent_menu = parent_menu;
  134. menu->items = items;
  135. menu->items_count = count;
  136. rtgui_widget_set_rect(RTGUI_WIDGET(menu), &rect);
  137. rtgui_rect_inflate(&rect, -1);
  138. /* create menu item list */
  139. menu->items_list = rtgui_listctrl_create((rt_uint32_t)items, count, &rect, _rtgui_menu_item_ondraw);
  140. RTGUI_WIDGET_BACKGROUND(RTGUI_WIDGET(menu->items_list)) = rtgui_theme_default_bc();
  141. rtgui_container_add_child(RTGUI_CONTAINER(menu), RTGUI_WIDGET(menu->items_list));
  142. rtgui_listctrl_set_onitem(menu->items_list, _rtgui_menu_onitem);
  143. }
  144. return menu;
  145. }
  146. void rtgui_menu_destroy(struct rtgui_menu* menu)
  147. {
  148. rtgui_widget_destroy (RTGUI_WIDGET(menu));
  149. }
  150. void rtgui_menu_set_onmenupop(struct rtgui_menu* menu, rtgui_event_handler_ptr handler)
  151. {
  152. if (menu == RT_NULL) return;
  153. menu->on_menupop = handler;
  154. }
  155. void rtgui_menu_set_onmenuhide(struct rtgui_menu* menu, rtgui_event_handler_ptr handler)
  156. {
  157. if (menu == RT_NULL) return;
  158. menu->on_menuhide = handler;
  159. }
  160. void rtgui_menu_pop(struct rtgui_menu* menu, int x, int y)
  161. {
  162. rtgui_rect_t rect;
  163. struct rtgui_event_resize eresize;
  164. if (menu == RT_NULL) return;
  165. /* set window extent */
  166. rect.x1 = 0; rect.y1 = 0;
  167. rect.x2 = 100; rect.y2 = menu->items_count * (rtgui_theme_get_selected_height() + 2) + 5;
  168. rtgui_rect_moveto(&rect, x, y);
  169. rtgui_win_set_rect(RTGUI_WIN(menu), &rect);
  170. rtgui_rect_inflate(&rect, -1);
  171. rtgui_widget_set_rect(RTGUI_WIDGET(menu->items_list), &rect);
  172. eresize.parent.type = RTGUI_EVENT_RESIZE;
  173. eresize.x = rect.x1; eresize.y = rect.y1;
  174. eresize.h = rect.y2 - rect.y1; eresize.w = rect.x2 - rect.x1;
  175. rtgui_listctrl_event_handler(RTGUI_WIDGET(menu->items_list), &(eresize.parent));
  176. /* on menu pop handler */
  177. if (menu->on_menupop != RT_NULL)
  178. {
  179. menu->on_menupop(RTGUI_WIDGET(menu), RT_NULL);
  180. }
  181. /* show menu window */
  182. rtgui_win_show(RTGUI_WIN(menu), RT_FALSE);
  183. }