menu.c 7.8 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287
  1. #include <rtgui/dc.h>
  2. #include <rtgui/widgets/menu.h>
  3. #include <rtgui/rtgui_theme.h>
  4. static rt_bool_t rtgui_menu_on_deactivate(struct rtgui_object* object, rtgui_event_t* event);
  5. const static rt_uint8_t right_arrow[] = {0x80, 0xc0, 0xe0, 0xf0, 0xe0, 0xc0, 0x80};
  6. static void _rtgui_menu_constructor(rtgui_menu_t *menu)
  7. {
  8. /* set window style */
  9. RTGUI_WIN(menu)->style = RTGUI_WIN_STYLE_NO_TITLE;
  10. /* set deactivate handler */
  11. rtgui_win_set_ondeactivate(RTGUI_WIN(menu), rtgui_menu_on_deactivate);
  12. /* set proper of control */
  13. menu->parent_menu = RT_NULL;
  14. menu->sub_menu = RT_NULL;
  15. menu->items = RT_NULL;
  16. menu->items_count = 0;
  17. menu->items_list = RT_NULL;
  18. menu->on_menupop = RT_NULL;
  19. menu->on_menuhide = RT_NULL;
  20. }
  21. static void _rtgui_menu_destructor(rtgui_menu_t* menu)
  22. {
  23. if (menu->sub_menu != RT_NULL)
  24. {
  25. rtgui_menu_destroy(menu->sub_menu);
  26. menu->sub_menu = RT_NULL;
  27. }
  28. rtgui_listctrl_destroy(menu->items_list);
  29. menu->items_list = RT_NULL;
  30. }
  31. static rt_bool_t _rtgui_menu_onitem(struct rtgui_object* object, struct rtgui_event* event)
  32. {
  33. struct rtgui_menu* menu;
  34. RTGUI_WIDGET_EVENT_HANDLER_PREPARE
  35. /* get menu */
  36. menu = RTGUI_MENU(rtgui_widget_get_toplevel(widget));
  37. if (menu->items[menu->items_list->current_item].type == RTGUI_ITEM_SUBMENU)
  38. {
  39. const rtgui_menu_item_t* items;
  40. rt_uint16_t count;
  41. rtgui_rect_t item_rect;
  42. items = (rtgui_menu_item_t*)menu->items[menu->items_list->current_item].submenu;
  43. count = menu->items[menu->items_list->current_item].submenu_count;
  44. if (menu->sub_menu != RT_NULL)
  45. {
  46. if (menu->sub_menu->items == items)
  47. {
  48. if (!RTGUI_WIDGET_IS_HIDE(RTGUI_WIDGET(menu->sub_menu)))
  49. {
  50. /* hide this sub menu */
  51. rtgui_win_hiden(RTGUI_WIN(menu->sub_menu));
  52. return RT_FALSE;
  53. }
  54. /* show this sub menu */
  55. rtgui_listctrl_get_item_rect(menu->items_list, menu->items_list->current_item, &item_rect);
  56. rtgui_menu_pop(menu->sub_menu, item_rect.x2, item_rect.y1);
  57. return RT_FALSE;
  58. }
  59. /* delete sub menu */
  60. rtgui_menu_destroy(menu->sub_menu);
  61. menu->sub_menu = RT_NULL;
  62. }
  63. /* create sub menu */
  64. menu->sub_menu = rtgui_menu_create("submenu", menu, items, count);
  65. rtgui_listctrl_get_item_rect(menu->items_list, menu->items_list->current_item, &item_rect);
  66. rtgui_menu_pop(menu->sub_menu, item_rect.x2 + 5, item_rect.y1);
  67. }
  68. else /* other menu item */
  69. {
  70. /* invoke action */
  71. if (menu->items[menu->items_list->current_item].on_menuaction != RT_NULL)
  72. menu->items[menu->items_list->current_item].on_menuaction(RTGUI_WIDGET(menu), RT_NULL);
  73. /* hide sub-menu */
  74. if (menu->sub_menu != RT_NULL)
  75. {
  76. rtgui_menu_hiden(menu->sub_menu);
  77. }
  78. rtgui_menu_hiden(menu);
  79. }
  80. return RT_FALSE;
  81. }
  82. static void _rtgui_menu_item_ondraw(struct rtgui_listctrl *list,
  83. struct rtgui_dc* dc,
  84. rtgui_rect_t* rect,
  85. rt_uint16_t index)
  86. {
  87. rtgui_rect_t item_rect;
  88. struct rtgui_menu_item* item;
  89. item_rect = *rect;
  90. item_rect.x1 += 5;
  91. /* re-fill item */
  92. if (list->current_item == index)
  93. {
  94. rtgui_color_t bc;
  95. bc = RTGUI_WIDGET_BACKGROUND(RTGUI_WIDGET(list));
  96. RTGUI_WIDGET_BACKGROUND(RTGUI_WIDGET(list)) = blue;
  97. rtgui_dc_fill_rect(dc, rect);
  98. RTGUI_WIDGET_BACKGROUND(RTGUI_WIDGET(list)) = bc;
  99. }
  100. /* get menu item */
  101. item = (rtgui_menu_item_t*)list->items;
  102. item = &item[index];
  103. if (item->type == RTGUI_ITEM_SUBMENU)
  104. {
  105. rtgui_rect_t r = {0, 0, 8, 8};
  106. rtgui_dc_draw_text(dc, item->label, &item_rect);
  107. item_rect.x1 = item_rect.x2 - 16; item_rect.x2 -= 8;
  108. rtgui_rect_moveto_align(&item_rect, &r, RTGUI_ALIGN_CENTER_HORIZONTAL | RTGUI_ALIGN_CENTER_VERTICAL);
  109. rtgui_dc_draw_byte(dc, r.x1, r.y1, 8, right_arrow);
  110. }
  111. else if (item->type == RTGUI_ITEM_SEPARATOR)
  112. {
  113. rtgui_dc_draw_horizontal_line(dc, item_rect.x1, item_rect.x2, (item_rect.y2 + item_rect.y1)/2);
  114. }
  115. else if (item->type == RTGUI_ITEM_CHECK)
  116. {
  117. /* not support right now */
  118. }
  119. else
  120. {
  121. /* normal menu item */
  122. rtgui_dc_draw_text(dc, item->label, &item_rect);
  123. if (item->image != RT_NULL)
  124. rtgui_image_blit(item->image, dc, &item_rect);
  125. }
  126. }
  127. DEFINE_CLASS_TYPE(menu, "menu",
  128. RTGUI_WIN_TYPE,
  129. _rtgui_menu_constructor,
  130. _rtgui_menu_destructor,
  131. sizeof(struct rtgui_menu));
  132. static rt_bool_t rtgui_menu_on_deactivate(struct rtgui_object *object, rtgui_event_t* event)
  133. {
  134. rtgui_menu_t* menu;
  135. RTGUI_WIDGET_EVENT_HANDLER_PREPARE
  136. menu = RTGUI_MENU(object);
  137. if (menu->parent_menu != RT_NULL)
  138. {
  139. /* whether click on parent menu */
  140. if (rtgui_win_is_activated(RTGUI_WIN(menu->parent_menu)) == RT_TRUE &&
  141. menu->parent_menu->items[menu->parent_menu->items_list->current_item].submenu
  142. == (struct rtgui_menu_item_t *)menu->items)
  143. return RT_TRUE;
  144. }
  145. /* submenu is activate */
  146. if (menu->items[menu->items_list->current_item].type == RTGUI_ITEM_SUBMENU)
  147. {
  148. /* if sub menu activated, not hide menu. But we cannot use the
  149. * activated flag as criteria since the old window is deactivated
  150. * before the new window got activated. But the window will be shown in
  151. * this context, so use 'is not hide'. */
  152. if (menu->sub_menu != RT_NULL &&
  153. !RTGUI_WIDGET_IS_HIDE(RTGUI_WIDGET(menu->sub_menu)))
  154. return RT_TRUE;
  155. }
  156. rtgui_win_hiden(RTGUI_WIN(menu));
  157. if (menu->on_menuhide != RT_NULL)
  158. {
  159. menu->on_menuhide(RTGUI_OBJECT(menu), RT_NULL);
  160. }
  161. /* un-select item */
  162. menu->items_list->current_item = -1;
  163. /* if it's a submenu, try to hide parent menu */
  164. if (menu->parent_menu != RT_NULL &&
  165. rtgui_win_is_activated(RTGUI_WIN(menu->parent_menu)) == RT_FALSE)
  166. {
  167. rtgui_menu_on_deactivate(RTGUI_OBJECT(menu->parent_menu), event);
  168. }
  169. return RT_TRUE;
  170. }
  171. struct rtgui_menu* rtgui_menu_create(const char* title, struct rtgui_menu* parent_menu,
  172. const struct rtgui_menu_item* items, rt_uint16_t count)
  173. {
  174. rtgui_rect_t rect = {0, 0, 100, 100};
  175. struct rtgui_menu* menu;
  176. menu = (struct rtgui_menu*) rtgui_widget_create ( RTGUI_MENU_TYPE );
  177. if (menu != RT_NULL)
  178. {
  179. rtgui_win_set_title(RTGUI_WIN(menu), title);
  180. menu->parent_menu = parent_menu;
  181. menu->items = items;
  182. menu->items_count = count;
  183. rtgui_widget_set_rect(RTGUI_WIDGET(menu), &rect);
  184. rtgui_rect_inflate(&rect, -1);
  185. /* create menu item list */
  186. menu->items_list = rtgui_listctrl_create((rt_uint32_t)items, count, &rect, _rtgui_menu_item_ondraw);
  187. RTGUI_WIDGET_BACKGROUND(RTGUI_WIDGET(menu->items_list)) = rtgui_theme_default_bc();
  188. rtgui_container_add_child(RTGUI_CONTAINER(menu), RTGUI_WIDGET(menu->items_list));
  189. rtgui_listctrl_set_onitem(menu->items_list, _rtgui_menu_onitem);
  190. }
  191. return menu;
  192. }
  193. void rtgui_menu_destroy(struct rtgui_menu* menu)
  194. {
  195. rtgui_widget_destroy (RTGUI_WIDGET(menu));
  196. }
  197. void rtgui_menu_set_onmenupop(struct rtgui_menu* menu, rtgui_event_handler_ptr handler)
  198. {
  199. if (menu == RT_NULL) return;
  200. menu->on_menupop = handler;
  201. }
  202. void rtgui_menu_set_onmenuhide(struct rtgui_menu* menu, rtgui_event_handler_ptr handler)
  203. {
  204. if (menu == RT_NULL) return;
  205. menu->on_menuhide = handler;
  206. }
  207. void rtgui_menu_pop(struct rtgui_menu* menu, int x, int y)
  208. {
  209. rtgui_rect_t rect;
  210. struct rtgui_event_resize eresize;
  211. if (menu == RT_NULL)
  212. return;
  213. /* set window extent */
  214. rect.x1 = 0; rect.y1 = 0;
  215. rect.x2 = 100; rect.y2 = menu->items_count * (rtgui_theme_get_selected_height() + 2) + 5;
  216. rtgui_rect_moveto(&rect, x, y);
  217. rtgui_win_set_rect(RTGUI_WIN(menu), &rect);
  218. rtgui_rect_inflate(&rect, -1);
  219. rtgui_widget_set_rect(RTGUI_WIDGET(menu->items_list), &rect);
  220. eresize.parent.type = RTGUI_EVENT_RESIZE;
  221. eresize.x = rect.x1; eresize.y = rect.y1;
  222. eresize.h = rect.y2 - rect.y1; eresize.w = rect.x2 - rect.x1;
  223. rtgui_listctrl_event_handler(RTGUI_OBJECT(menu->items_list), &(eresize.parent));
  224. /* on menu pop handler */
  225. if (menu->on_menupop != RT_NULL)
  226. {
  227. menu->on_menupop(RTGUI_OBJECT(menu), RT_NULL);
  228. }
  229. /* show menu window */
  230. rtgui_win_show(RTGUI_WIN(menu), RT_FALSE);
  231. }
  232. void rtgui_menu_hiden(struct rtgui_menu* menu)
  233. {
  234. rtgui_win_hiden(RTGUI_WIN(menu));
  235. /* un-select item */
  236. menu->items_list->current_item = -1;
  237. if (menu->parent_menu != RT_NULL)
  238. rtgui_menu_hiden(menu->parent_menu);
  239. }