demo_view_buffer_animation.c 3.4 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124
  1. #include <rtgui/dc.h>
  2. #include <rtgui/rtgui_system.h>
  3. #include <rtgui/widgets/view.h>
  4. #include "demo_view.h"
  5. /*
  6. * 直接在DC上绘图以实现动画效果
  7. *
  8. * 动画是依赖于定时器驱动的,会上下翻滚显示文字
  9. * "飞线乱飞"
  10. */
  11. static rt_int8_t dx = 1, dy = 1;
  12. static rtgui_rect_t text_rect;
  13. static rtgui_timer_t *timer;
  14. static struct rtgui_dc *dc_buffer;
  15. static void timeout(struct rtgui_timer* timer, void* parameter)
  16. {
  17. struct rtgui_dc* dc;
  18. rtgui_rect_t rect;
  19. rtgui_widget_t *widget;
  20. /* 控件(view)通过parameter参数传递给定时器 */
  21. widget = (rtgui_widget_t*)parameter;
  22. /* 获得控件所属的DC */
  23. dc = rtgui_dc_begin_drawing(widget);
  24. if (dc == RT_NULL) /* 如果不能正常获得DC,返回(如果控件或父控件是隐藏状态,DC是获取不成功的) */
  25. return ;
  26. /* 获得demo view允许绘图的区域,主要用于判断边界 */
  27. demo_view_get_logic_rect(RTGUI_VIEW(widget), &rect);
  28. rect.y2 -= 5;
  29. /* 判断是否是第一次绘图 */
  30. if ((text_rect.x1 == 0) && (text_rect.y1 == 0))
  31. {
  32. rtgui_rect_moveto(&text_rect, rect.x1, rect.y1);
  33. }
  34. /* 设置dx和dy */
  35. if (text_rect.x2 >= rect.x2) dx = -1;
  36. if (text_rect.x1 < rect.x1) dx = 1;
  37. if (text_rect.y2 >= rect.y2) dy = -1;
  38. if (text_rect.y1 < rect.y1) dy = 1;
  39. /* 移动文本框的位置 */
  40. text_rect.x1 += dx; text_rect.x2 += dx;
  41. text_rect.y1 += dy; text_rect.y2 += dy;
  42. /* 绘图 */
  43. rect = text_rect;
  44. rect.x2 += 2; rect.y2 += 2;
  45. rtgui_dc_blit(dc_buffer, NULL, dc, &rect);
  46. /* 绘图完成 */
  47. rtgui_dc_end_drawing(dc);
  48. }
  49. static rt_bool_t animation_event_handler(rtgui_widget_t* widget, rtgui_event_t *event)
  50. {
  51. if (event->type == RTGUI_EVENT_PAINT)
  52. {
  53. struct rtgui_dc* dc;
  54. rtgui_rect_t rect;
  55. /* 因为用的是demo view,上面本身有一部分控件,所以在绘图时先要让demo view先绘图 */
  56. rtgui_view_event_handler(widget, event);
  57. /* 获得控件所属的DC */
  58. dc = rtgui_dc_begin_drawing(widget);
  59. if (dc == RT_NULL) /* 如果不能正常获得DC,返回(如果控件或父控件是隐藏状态,DC是获取不成功的) */
  60. return RT_FALSE;
  61. /* 获得demo view允许绘图的区域 */
  62. demo_view_get_logic_rect(RTGUI_VIEW(widget), &rect);
  63. /* 绘图 */
  64. rect = text_rect;
  65. rtgui_rect_inflate(&rect, +1);
  66. rtgui_dc_blit(dc_buffer, NULL, dc, &rect);
  67. /* 绘图完成 */
  68. rtgui_dc_end_drawing(dc);
  69. }
  70. else
  71. {
  72. /* 调用默认的事件处理函数 */
  73. return rtgui_view_event_handler(widget, event);
  74. }
  75. return RT_FALSE;
  76. }
  77. rtgui_view_t *demo_view_buffer_animation(rtgui_workbench_t* workbench)
  78. {
  79. rtgui_view_t *view;
  80. view = demo_view(workbench, "DC 缓冲区动画");
  81. if (view != RT_NULL)
  82. rtgui_widget_set_event_handler(RTGUI_WIDGET(view), animation_event_handler);
  83. rtgui_font_get_metrics(RTGUI_WIDGET_FONT(RTGUI_WIDGET(view)), "缓冲动画", &text_rect);
  84. if (dc_buffer == RT_NULL)
  85. {
  86. rtgui_rect_t rect;
  87. rect.x1 = 0; rect.x2 = rtgui_rect_width(text_rect) + 2;
  88. rect.y1 = 0; rect.y2 = rtgui_rect_height(text_rect) + 2;
  89. /* 创建 DC Buffer,长 50,宽 50 */
  90. dc_buffer = rtgui_dc_buffer_create(rtgui_rect_width(rect), rtgui_rect_height(rect));
  91. RTGUI_DC_FC(dc_buffer) = RTGUI_WIDGET_BACKGROUND(RTGUI_WIDGET(view));
  92. rtgui_dc_fill_rect(dc_buffer, &rect);
  93. RTGUI_DC_FC(dc_buffer) = black;
  94. rect.x1 = 1; rect.y1 = 1;
  95. rtgui_dc_draw_text(dc_buffer, "缓冲动画", &rect);
  96. }
  97. /* 启动定时器以触发动画 */
  98. timer = rtgui_timer_create(1, RT_TIMER_FLAG_PERIODIC, timeout, (void*)view);
  99. rtgui_timer_start(timer);
  100. return view;
  101. }