toplevel.c 5.0 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194
  1. /*
  2. * File : toplevel.c
  3. * This file is part of RT-Thread RTOS
  4. * COPYRIGHT (C) 2006 - 2009, 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. * 2009-10-16 Bernard first version
  13. */
  14. #include <rtgui/rtgui_system.h>
  15. #include <rtgui/widgets/toplevel.h>
  16. static void _rtgui_toplevel_constructor(rtgui_toplevel_t *toplevel)
  17. {
  18. /* set event handler */
  19. rtgui_widget_set_event_handler(RTGUI_WIDGET(toplevel), rtgui_toplevel_event_handler);
  20. /* set toplevel to self */
  21. RTGUI_WIDGET(toplevel)->toplevel = RTGUI_WIDGET(toplevel);
  22. /* init toplevel property */
  23. toplevel->drawing = 0;
  24. toplevel->external_clip_rect = RT_NULL;
  25. toplevel->external_clip_size = 0;
  26. toplevel->focus = RT_NULL;
  27. /* hide toplevel default */
  28. RTGUI_WIDGET_HIDE(RTGUI_WIDGET(toplevel));
  29. /* set server as RT_NULL (no connected) */
  30. toplevel->server = RT_NULL;
  31. }
  32. static void _rtgui_toplevel_destructor(rtgui_toplevel_t* toplevel)
  33. {
  34. /* release external clip info */
  35. rtgui_free(toplevel->external_clip_rect);
  36. toplevel->drawing = 0;
  37. rtgui_free(toplevel->external_clip_rect);
  38. toplevel->external_clip_rect = RT_NULL;
  39. toplevel->external_clip_size = 0;
  40. toplevel->focus = RT_NULL;
  41. }
  42. rtgui_type_t *rtgui_toplevel_type_get(void)
  43. {
  44. static rtgui_type_t *toplevel_type = RT_NULL;
  45. if (!toplevel_type)
  46. {
  47. toplevel_type = rtgui_type_create("toplevel", RTGUI_CONTAINER_TYPE,
  48. sizeof(rtgui_toplevel_t),
  49. RTGUI_CONSTRUCTOR(_rtgui_toplevel_constructor),
  50. RTGUI_DESTRUCTOR(_rtgui_toplevel_destructor));
  51. }
  52. return toplevel_type;
  53. }
  54. rt_bool_t rtgui_toplevel_event_handler(rtgui_widget_t* widget, rtgui_event_t* event)
  55. {
  56. rtgui_toplevel_t* toplevel = (rtgui_toplevel_t*)widget;
  57. switch (event->type)
  58. {
  59. case RTGUI_EVENT_KBD:
  60. if (toplevel->focus != RT_NULL)
  61. {
  62. toplevel->focus->event_handler(toplevel->focus, event);
  63. }
  64. break;
  65. case RTGUI_EVENT_CLIP_INFO:
  66. /* set toplevel external clip info */
  67. rtgui_toplevel_handle_clip(toplevel, (struct rtgui_event_clip_info*)event);
  68. /* update toplevel clip */
  69. rtgui_toplevel_update_clip(toplevel);
  70. break;
  71. case RTGUI_EVENT_TIMER:
  72. {
  73. struct rtgui_timer* timer;
  74. struct rtgui_event_timer* etimer = (struct rtgui_event_timer*) event;
  75. timer = etimer->timer;
  76. if (timer->timeout != RT_NULL)
  77. {
  78. /* call timeout function */
  79. timer->timeout(timer, timer->user_data);
  80. }
  81. }
  82. break;
  83. case RTGUI_EVENT_COMMAND:
  84. if (rtgui_container_dispatch_event(RTGUI_CONTAINER(widget), event) != RT_TRUE)
  85. {
  86. if (widget->on_command != RT_NULL)
  87. {
  88. widget->on_command(widget, event);
  89. }
  90. }
  91. break;
  92. default :
  93. return rtgui_container_event_handler(widget, event);
  94. }
  95. return RT_FALSE;
  96. }
  97. void rtgui_toplevel_handle_clip(struct rtgui_toplevel* top,
  98. struct rtgui_event_clip_info* info)
  99. {
  100. RT_ASSERT(top != RT_NULL);
  101. RT_ASSERT(info != RT_NULL);
  102. /* release old rect array */
  103. if (top->external_clip_size != 0)
  104. {
  105. rtgui_free(top->external_clip_rect);
  106. top->external_clip_rect = RT_NULL;
  107. top->external_clip_size = 0;
  108. }
  109. /* no rect info */
  110. if (info->num_rect == 0) return;
  111. top->external_clip_rect = (rtgui_rect_t*) rtgui_malloc(sizeof(rtgui_rect_t) *
  112. info->num_rect);
  113. top->external_clip_size = info->num_rect;
  114. /* copy rect array */
  115. rt_memcpy(top->external_clip_rect, info->rects, sizeof(rtgui_rect_t) * info->num_rect);
  116. }
  117. #include <rtgui/driver.h> /* to get screen rect */
  118. void rtgui_toplevel_update_clip(rtgui_toplevel_t* top)
  119. {
  120. rt_uint32_t idx;
  121. rtgui_container_t* container;
  122. struct rtgui_list_node* node;
  123. rtgui_rect_t screen_rect;
  124. if (top == RT_NULL) return;
  125. /* reset toplevel widget clip to extent */
  126. rtgui_region_reset(&(RTGUI_WIDGET(top)->clip), &(RTGUI_WIDGET(top)->extent));
  127. RTGUI_WIDGET(top)->clip_sync ++;
  128. /* subtract the screen rect */
  129. screen_rect.x1 = screen_rect.y1 = 0;
  130. screen_rect.x2 = rtgui_graphic_driver_get_default()->width;
  131. screen_rect.y2 = rtgui_graphic_driver_get_default()->height;
  132. rtgui_region_intersect_rect(&(RTGUI_WIDGET(top)->clip), &(RTGUI_WIDGET(top)->clip),
  133. &screen_rect);
  134. /* subtract the external rect */
  135. for (idx = 0; idx < top->external_clip_size; idx ++)
  136. {
  137. rtgui_region_subtract_rect(&(RTGUI_WIDGET(top)->clip), &(RTGUI_WIDGET(top)->clip),
  138. &(top->external_clip_rect[idx]));
  139. }
  140. /* update the clip info of each child */
  141. container = RTGUI_CONTAINER(top);
  142. rtgui_list_foreach(node, &(container->children))
  143. {
  144. rtgui_widget_t* child = rtgui_list_entry(node, rtgui_widget_t, sibling);
  145. rtgui_widget_update_clip(child);
  146. }
  147. }
  148. void rtgui_toplevel_set_focus(struct rtgui_toplevel* top, rtgui_widget_t* focus)
  149. {
  150. RT_ASSERT(top != RT_NULL);
  151. top->focus = focus;
  152. }
  153. rtgui_widget_t* rtgui_toplevel_get_focus(struct rtgui_toplevel* top)
  154. {
  155. RT_ASSERT(top != RT_NULL);
  156. return top->focus;
  157. }