1
0

list_view.c 7.9 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290
  1. /*
  2. * File : list_view.c
  3. * This file is part of RTGUI in RT-Thread RTOS
  4. * COPYRIGHT (C) 2010, RT-Thread Development Team
  5. *
  6. * The license and distribution terms for this file may be
  7. * found in the file LICENSE in this distribution or at
  8. * http://www.rt-thread.org/license/LICENSE
  9. *
  10. * Change Logs:
  11. * Date Author Notes
  12. * 2010-01-06 Bernard first version
  13. */
  14. #include <rtgui/rtgui_theme.h>
  15. #include <rtgui/widgets/list_view.h>
  16. #define LIST_MARGIN 5
  17. static void _rtgui_list_view_constructor(struct rtgui_list_view *view)
  18. {
  19. /* default rect */
  20. struct rtgui_rect rect = {0, 0, 200, 200};
  21. /* set default widget rect and set event handler */
  22. rtgui_widget_set_event_handler(RTGUI_WIDGET(view),rtgui_list_view_event_handler);
  23. rtgui_widget_set_rect(RTGUI_WIDGET(view), &rect);
  24. RTGUI_WIDGET(view)->flag |= RTGUI_WIDGET_FLAG_FOCUSABLE;
  25. view->current_item = 0;
  26. view->items_count = 0;
  27. view->page_items = 0;
  28. RTGUI_WIDGET_BACKGROUND(RTGUI_WIDGET(view)) = white;
  29. RTGUI_WIDGET_TEXTALIGN(RTGUI_WIDGET(view)) = RTGUI_ALIGN_CENTER_VERTICAL;
  30. }
  31. rtgui_type_t *rtgui_list_view_type_get(void)
  32. {
  33. static rtgui_type_t *list_view_type = RT_NULL;
  34. if (!list_view_type)
  35. {
  36. list_view_type = rtgui_type_create("listview", RTGUI_VIEW_TYPE,
  37. sizeof(rtgui_list_view_t), RTGUI_CONSTRUCTOR(_rtgui_list_view_constructor), RT_NULL);
  38. }
  39. return list_view_type;
  40. }
  41. void rtgui_list_view_ondraw(struct rtgui_list_view* view)
  42. {
  43. struct rtgui_rect rect, item_rect;
  44. struct rtgui_dc* dc;
  45. rt_uint16_t page_index, index;
  46. const struct rtgui_list_item* item;
  47. dc = rtgui_dc_begin_drawing(RTGUI_WIDGET(view));
  48. if (dc == RT_NULL) return;
  49. rtgui_widget_get_rect(RTGUI_WIDGET(view), &rect);
  50. rtgui_dc_fill_rect(dc, &rect);
  51. /* get item base rect */
  52. item_rect = rect;
  53. item_rect.y1 += 2;
  54. item_rect.y2 = item_rect.y1 + (2 + rtgui_theme_get_selected_height());
  55. /* get current page */
  56. page_index = (view->current_item / view->page_items) * view->page_items;
  57. for (index = 0; index < view->page_items; index ++)
  58. {
  59. if (page_index + index >= view->items_count) break;
  60. item = &(view->items[page_index + index]);
  61. if (page_index + index == view->current_item)
  62. {
  63. rtgui_theme_draw_selected(dc, &item_rect);
  64. }
  65. item_rect.x1 += LIST_MARGIN;
  66. if (item->image != RT_NULL)
  67. {
  68. rtgui_image_blit(item->image, dc, &item_rect);
  69. item_rect.x1 += item->image->w + 2;
  70. }
  71. /* draw text */
  72. rtgui_dc_draw_text(dc, item->name, &item_rect);
  73. if (item->image != RT_NULL)
  74. item_rect.x1 -= (item->image->w + 2);
  75. item_rect.x1 -= LIST_MARGIN;
  76. /* move to next item position */
  77. item_rect.y1 += (rtgui_theme_get_selected_height() + 2);
  78. item_rect.y2 += (rtgui_theme_get_selected_height() + 2);
  79. }
  80. rtgui_dc_end_drawing(dc);
  81. }
  82. void rtgui_list_view_update_current(struct rtgui_list_view* view, rt_uint16_t old_item)
  83. {
  84. struct rtgui_dc* dc;
  85. const struct rtgui_list_item* item;
  86. rtgui_rect_t rect, item_rect;
  87. if (old_item/view->page_items != view->current_item/view->page_items)
  88. {
  89. /* it's not a same page, update all */
  90. rtgui_widget_update(RTGUI_WIDGET(view));
  91. return;
  92. }
  93. dc = rtgui_dc_begin_drawing(RTGUI_WIDGET(view));
  94. if (dc == RT_NULL) return;
  95. rtgui_widget_get_rect(RTGUI_WIDGET(view), &rect);
  96. item_rect = rect;
  97. /* get old item's rect */
  98. item_rect.y1 += 2;
  99. item_rect.y1 += (old_item % view->page_items) * (2 + rtgui_theme_get_selected_height());
  100. item_rect.y2 = item_rect.y1 + (2 + rtgui_theme_get_selected_height());
  101. /* draw old item */
  102. rtgui_dc_fill_rect(dc, &item_rect);
  103. item_rect.x1 += LIST_MARGIN;
  104. item = &(view->items[old_item]);
  105. if (item->image != RT_NULL)
  106. {
  107. rtgui_image_blit(item->image, dc, &item_rect);
  108. item_rect.x1 += item->image->w + 2;
  109. }
  110. rtgui_dc_draw_text(dc, item->name, &item_rect);
  111. /* draw current item */
  112. item_rect = rect;
  113. /* get current item's rect */
  114. item_rect.y1 += 2;
  115. item_rect.y1 += (view->current_item % view->page_items) * (2 + rtgui_theme_get_selected_height());
  116. item_rect.y2 = item_rect.y1 + (2 + rtgui_theme_get_selected_height());
  117. /* draw current item */
  118. rtgui_theme_draw_selected(dc, &item_rect);
  119. item_rect.x1 += LIST_MARGIN;
  120. item = &(view->items[view->current_item]);
  121. if (item->image != RT_NULL)
  122. {
  123. rtgui_image_blit(item->image, dc, &item_rect);
  124. item_rect.x1 += (item->image->w + 2);
  125. }
  126. rtgui_dc_draw_text(dc, item->name, &item_rect);
  127. rtgui_dc_end_drawing(dc);
  128. }
  129. rt_bool_t rtgui_list_view_event_handler(struct rtgui_widget* widget, struct rtgui_event* event)
  130. {
  131. struct rtgui_list_view* view = RT_NULL;
  132. view = RTGUI_LIST_VIEW(widget);
  133. switch (event->type)
  134. {
  135. case RTGUI_EVENT_PAINT:
  136. rtgui_list_view_ondraw(view);
  137. return RT_FALSE;
  138. case RTGUI_EVENT_RESIZE:
  139. {
  140. struct rtgui_event_resize* resize;
  141. resize = (struct rtgui_event_resize*)event;
  142. /* recalculate page items */
  143. view->page_items = resize->h / (2 + rtgui_theme_get_selected_height());
  144. }
  145. break;
  146. case RTGUI_EVENT_MOUSE_BUTTON:
  147. {
  148. rtgui_rect_t rect;
  149. struct rtgui_event_mouse* emouse;
  150. emouse = (struct rtgui_event_mouse*)event;
  151. /* calculate selected item */
  152. /* get physical extent information */
  153. rtgui_widget_get_rect(widget, &rect);
  154. rtgui_widget_rect_to_device(widget, &rect);
  155. if (rtgui_rect_contains_point(&rect, emouse->x, emouse->y) == RT_EOK)
  156. {
  157. rt_uint16_t index;
  158. index = (emouse->y - rect.y1) / (2 + rtgui_theme_get_selected_height());
  159. if ((index < view->items_count) && (index < view->page_items))
  160. {
  161. rt_uint16_t old_item;
  162. old_item = view->current_item;
  163. /* set selected item */
  164. view->current_item = view->current_item/view->page_items + index;
  165. rtgui_list_view_update_current(view, old_item);
  166. }
  167. }
  168. }
  169. break;
  170. case RTGUI_EVENT_KBD:
  171. {
  172. struct rtgui_event_kbd* ekbd = (struct rtgui_event_kbd*)event;
  173. if (ekbd->type == RTGUI_KEYDOWN)
  174. {
  175. rt_uint16_t old_item;
  176. old_item = view->current_item;
  177. switch (ekbd->key)
  178. {
  179. case RTGUIK_LEFT:
  180. if (view->current_item - view->page_items >= 0)
  181. view->current_item -= view->page_items;
  182. rtgui_list_view_update_current(view, old_item);
  183. return RT_FALSE;
  184. case RTGUIK_UP:
  185. if (view->current_item > 0)
  186. view->current_item --;
  187. rtgui_list_view_update_current(view, old_item);
  188. return RT_FALSE;
  189. case RTGUIK_RIGHT:
  190. if (view->current_item + view->page_items < view->items_count - 1)
  191. view->current_item += view->page_items;
  192. rtgui_list_view_update_current(view, old_item);
  193. return RT_FALSE;
  194. case RTGUIK_DOWN:
  195. if (view->current_item < view->items_count - 1)
  196. view->current_item ++;
  197. rtgui_list_view_update_current(view, old_item);
  198. return RT_FALSE;
  199. case RTGUIK_RETURN:
  200. if (view->items[view->current_item].action != RT_NULL)
  201. {
  202. view->items[view->current_item].action(view->items[view->current_item].parameter);
  203. }
  204. return RT_FALSE;
  205. default:
  206. break;
  207. }
  208. }
  209. }
  210. return RT_FALSE;
  211. }
  212. /* use view event handler */
  213. return rtgui_view_event_handler(widget, event);
  214. }
  215. rtgui_list_view_t* rtgui_list_view_create(const struct rtgui_list_item* items, rt_uint16_t count, rtgui_rect_t *rect)
  216. {
  217. struct rtgui_list_view* view = RT_NULL;
  218. view = (struct rtgui_list_view*) rtgui_widget_create(RTGUI_LIST_VIEW_TYPE);
  219. if (view != RT_NULL)
  220. {
  221. view->items = items;
  222. view->items_count = count;
  223. view->page_items = rtgui_rect_height(*rect) / (2 + rtgui_theme_get_selected_height());
  224. }
  225. return view;
  226. }
  227. void rtgui_list_view_destroy(rtgui_list_view_t* view)
  228. {
  229. /* destroy view */
  230. rtgui_widget_destroy(RTGUI_WIDGET(view));
  231. }