rtgui_theme.c 26 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972973974975976977978979980981982983984985986987988989990991992993994995996997
  1. /*
  2. * File : rtgui_theme.c
  3. * This file is part of RTGUI in 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-04 Bernard first version
  13. */
  14. #include <rtgui/rtgui.h>
  15. #include <rtgui/dc.h>
  16. #include <rtgui/widgets/widget.h>
  17. #include <rtgui/widgets/button.h>
  18. #include <rtgui/widgets/label.h>
  19. #include <rtgui/widgets/textbox.h>
  20. #include <rtgui/widgets/iconbox.h>
  21. #include <rtgui/widgets/title.h>
  22. #include <rtgui/rtgui_theme.h>
  23. #include <rtgui/rtgui_server.h>
  24. #include <rtgui/rtgui_system.h>
  25. #define SELECTED_HEIGHT 25
  26. const rtgui_color_t default_foreground = RTGUI_RGB(0x00, 0x00, 0x00);
  27. const rtgui_color_t default_background = RTGUI_RGB(212, 208, 200);
  28. const rtgui_color_t selected_color = RTGUI_RGB(0xc0, 0xc0, 0xc0);
  29. const rtgui_color_t disable_foreground = RTGUI_RGB(0x80, 0x80, 0x80);
  30. extern struct rtgui_font rtgui_font_asc16;
  31. extern struct rtgui_font rtgui_font_arial16;
  32. extern struct rtgui_font rtgui_font_asc12;
  33. extern struct rtgui_font rtgui_font_arial12;
  34. /* init theme */
  35. void rtgui_system_theme_init()
  36. {
  37. #if RTGUI_DEFAULT_FONT_SIZE == 16
  38. rtgui_font_set_defaut(&rtgui_font_asc16);
  39. #elif RTGUI_DEFAULT_FONT_SIZE == 12
  40. rtgui_font_set_defaut(&rtgui_font_asc12);
  41. #else
  42. rtgui_font_set_defaut(&rtgui_font_asc12);
  43. #endif
  44. }
  45. static const rt_uint8_t close_byte[14] = {0x06, 0x18, 0x03, 0x30, 0x01, 0xE0, 0x00,
  46. 0xC0, 0x01, 0xE0, 0x03, 0x30, 0x06, 0x18
  47. };
  48. /* window drawing */
  49. void rtgui_theme_draw_win(struct rtgui_topwin* win)
  50. {
  51. struct rtgui_dc* dc;
  52. rtgui_rect_t rect;
  53. if (win->title == RT_NULL) return; /* no title and no board */
  54. /* begin drawing */
  55. dc = rtgui_dc_begin_drawing(RTGUI_WIDGET(win->title));
  56. if (dc == RT_NULL) return;
  57. /* get rect */
  58. rtgui_widget_get_rect(RTGUI_WIDGET(win->title), &rect);
  59. /* draw border */
  60. if (win->flag & WINTITLE_BORDER)
  61. {
  62. rect.x2 -= 1; rect.y2 -= 1;
  63. RTGUI_WIDGET_FOREGROUND(win->title) = RTGUI_RGB(212, 208, 200);
  64. rtgui_dc_draw_hline(dc, rect.x1, rect.x2, rect.y1);
  65. rtgui_dc_draw_vline(dc, rect.x1, rect.y1, rect.y2);
  66. RTGUI_WIDGET_FOREGROUND(win->title) = white;
  67. rtgui_dc_draw_hline(dc, rect.x1 + 1, rect.x2 - 1, rect.y1 + 1);
  68. rtgui_dc_draw_vline(dc, rect.x1 + 1, rect.y1 + 1, rect.y2 - 1);
  69. RTGUI_WIDGET_FOREGROUND(win->title) = RTGUI_RGB(128, 128, 128);
  70. rtgui_dc_draw_hline(dc, rect.x1 + 1, rect.x2 - 1, rect.y2 - 1);
  71. rtgui_dc_draw_vline(dc, rect.x2 - 1, rect.y1 + 1, rect.y2);
  72. RTGUI_WIDGET_FOREGROUND(win->title) = RTGUI_RGB(64, 64, 64);
  73. rtgui_dc_draw_hline(dc, rect.x1, rect.x2, rect.y2);
  74. rtgui_dc_draw_vline(dc, rect.x2, rect.y1, rect.y2 + 1);
  75. /* shrink border */
  76. rtgui_rect_inflate(&rect, -WINTITLE_BORDER_SIZE);
  77. }
  78. /* draw title */
  79. if (!(win->flag & WINTITLE_NO))
  80. {
  81. rt_uint32_t index;
  82. float r, g, b, delta;
  83. if (win->flag & WINTITLE_ACTIVATE)
  84. {
  85. r = 10; g = 36; b = 106;
  86. delta = 150 / (float)(rect.x2 - rect.x1);
  87. }
  88. else
  89. {
  90. r = 128; g = 128; b = 128;
  91. delta = 64 / (float)(rect.x2 - rect.x1);
  92. }
  93. RTGUI_WIDGET_FOREGROUND(win->title) = RTGUI_RGB(r, g, b);
  94. for (index = rect.x1; index < rect.x2 + 1; index ++)
  95. {
  96. rtgui_dc_draw_vline(dc, index, rect.y1, rect.y2);
  97. r += delta; g += delta; b += delta;
  98. }
  99. if (win->flag & WINTITLE_ACTIVATE)
  100. {
  101. RTGUI_WIDGET_FOREGROUND(win->title) = white;
  102. }
  103. else
  104. {
  105. RTGUI_WIDGET_FOREGROUND(win->title) = RTGUI_RGB(212, 208, 200);
  106. }
  107. rect.x1 += 4;
  108. rect.y1 += 2; rect.y2 = rect.y1 + WINTITLE_CB_HEIGHT;
  109. rtgui_dc_draw_text(dc, rtgui_wintitle_get_title(win->title), &rect);
  110. if (win->flag & WINTITLE_CLOSEBOX)
  111. {
  112. /* get close button rect */
  113. rtgui_rect_t box_rect = {0, 0, WINTITLE_CB_WIDTH, WINTITLE_CB_HEIGHT};
  114. rtgui_rect_moveto_align(&rect, &box_rect, RTGUI_ALIGN_CENTER_VERTICAL | RTGUI_ALIGN_RIGHT);
  115. box_rect.x1 -= 3; box_rect.x2 -= 3;
  116. rtgui_dc_fill_rect(dc, &box_rect);
  117. /* draw close box */
  118. if (win->flag & WINTITLE_CB_PRESSED)
  119. {
  120. rtgui_dc_draw_border(dc, &box_rect, RTGUI_BORDER_SUNKEN);
  121. RTGUI_WIDGET_FOREGROUND(win->title) = red;
  122. rtgui_dc_draw_word(dc, box_rect.x1, box_rect.y1 + 6, 7, close_byte);
  123. }
  124. else
  125. {
  126. rtgui_dc_draw_border(dc, &box_rect, RTGUI_BORDER_RAISE);
  127. RTGUI_WIDGET_FOREGROUND(win->title) = black;
  128. rtgui_dc_draw_word(dc, box_rect.x1 - 1, box_rect.y1 + 5, 7, close_byte);
  129. }
  130. }
  131. }
  132. rtgui_dc_end_drawing(dc);
  133. }
  134. /* widget drawing */
  135. void rtgui_theme_draw_button(rtgui_button_t* btn)
  136. {
  137. /* draw button */
  138. struct rtgui_dc* dc;
  139. struct rtgui_rect rect;
  140. rtgui_color_t bc, fc;
  141. /* begin drawing */
  142. dc = rtgui_dc_begin_drawing(RTGUI_WIDGET(btn));
  143. if (dc == RT_NULL) return;
  144. /* get widget rect */
  145. rtgui_widget_get_rect(RTGUI_WIDGET(btn), &rect);
  146. /* get foreground color */
  147. bc = RTGUI_WIDGET_BACKGROUND(btn);
  148. fc = RTGUI_WIDGET_FOREGROUND(btn);
  149. if (btn->flag & RTGUI_BUTTON_FLAG_PRESS)
  150. {
  151. /* fill button rect with background color */
  152. rtgui_dc_fill_rect(dc, &rect);
  153. if (btn->pressed_image != RT_NULL)
  154. {
  155. rtgui_rect_t image_rect;
  156. image_rect.x1 = 0; image_rect.y1 = 0;
  157. image_rect.x2 = btn->unpressed_image->w;
  158. image_rect.y2 = btn->unpressed_image->h;
  159. rtgui_rect_moveto_align(&rect, &image_rect, RTGUI_ALIGN_CENTER);
  160. rtgui_image_blit(btn->pressed_image, dc, &image_rect);
  161. }
  162. else
  163. {
  164. rtgui_dc_draw_border(dc, &rect, RTGUI_BORDER_SUNKEN);
  165. }
  166. }
  167. else
  168. {
  169. /* fill button rect with background color */
  170. rtgui_dc_fill_rect(dc, &rect);
  171. if (btn->unpressed_image != RT_NULL)
  172. {
  173. rtgui_rect_t image_rect;
  174. image_rect.x1 = 0; image_rect.y1 = 0;
  175. image_rect.x2 = btn->unpressed_image->w;
  176. image_rect.y2 = btn->unpressed_image->h;
  177. rtgui_rect_moveto_align(&rect, &image_rect, RTGUI_ALIGN_CENTER);
  178. rtgui_image_blit(btn->unpressed_image, dc, &image_rect);
  179. }
  180. else
  181. {
  182. rtgui_dc_draw_border(dc, &rect, RTGUI_BORDER_RAISE);
  183. }
  184. }
  185. if (RTGUI_WIDGET_IS_FOCUSED(btn))
  186. {
  187. /* re-set foreground and get default rect */
  188. rtgui_widget_get_rect(RTGUI_WIDGET(btn), &rect);
  189. rtgui_rect_inflate(&rect, -2);
  190. RTGUI_WIDGET_FOREGROUND(btn) = black;
  191. rtgui_dc_draw_focus_rect(dc, &rect);
  192. }
  193. /* set forecolor */
  194. RTGUI_WIDGET_BACKGROUND(btn) = bc;
  195. RTGUI_WIDGET_FOREGROUND(btn) = fc;
  196. if (btn->pressed_image == RT_NULL)
  197. {
  198. /* re-set foreground and get default rect */
  199. rtgui_widget_get_rect(RTGUI_WIDGET(btn), &rect);
  200. /* remove border */
  201. rtgui_rect_inflate(&rect, -2);
  202. /* draw text */
  203. rtgui_dc_draw_text(dc, rtgui_label_get_text(RTGUI_LABEL(btn)), &rect);
  204. }
  205. /* end drawing */
  206. rtgui_dc_end_drawing(dc);
  207. }
  208. void rtgui_theme_draw_label(rtgui_label_t* label)
  209. {
  210. /* draw label */
  211. struct rtgui_dc* dc;
  212. struct rtgui_rect rect;
  213. /* begin drawing */
  214. dc = rtgui_dc_begin_drawing(RTGUI_WIDGET(label));
  215. if (dc == RT_NULL) return;
  216. rtgui_widget_get_rect(RTGUI_WIDGET(label), &rect);
  217. rtgui_dc_fill_rect(dc, &rect);
  218. /* default left and center draw */
  219. rtgui_dc_draw_text(dc, rtgui_label_get_text(label), &rect);
  220. /* end drawing */
  221. rtgui_dc_end_drawing(dc);
  222. }
  223. #define RTGUI_TEXTBOX_MARGIN 3
  224. void rtgui_theme_draw_textbox(rtgui_textbox_t* box)
  225. {
  226. /* draw button */
  227. struct rtgui_dc* dc;
  228. struct rtgui_rect rect;
  229. rtgui_color_t fc;
  230. /* begin drawing */
  231. dc = rtgui_dc_begin_drawing(RTGUI_WIDGET(box));
  232. if (dc == RT_NULL) return;
  233. /* get widget rect */
  234. rtgui_widget_get_rect(RTGUI_WIDGET(box), &rect);
  235. fc = RTGUI_WIDGET_FOREGROUND(box);
  236. /* fill widget rect with white color */
  237. RTGUI_WIDGET_BACKGROUND(box) = white;
  238. rtgui_dc_fill_rect(dc, &rect);
  239. /* draw border */
  240. RTGUI_WIDGET_FOREGROUND(box) = RTGUI_RGB(123, 158, 189);
  241. rtgui_dc_draw_rect(dc, &rect);
  242. /* draw text */
  243. RTGUI_WIDGET_FOREGROUND(box) = fc;
  244. if (box->text != RT_NULL)
  245. {
  246. rect.x1 += RTGUI_TEXTBOX_MARGIN;
  247. if (box->flag & RTGUI_TEXTBOX_MASK)
  248. {
  249. /* draw '*' */
  250. rt_size_t len = rt_strlen(box->text);
  251. if (len > 0)
  252. {
  253. char *text_mask = rtgui_malloc(len + 1);
  254. rt_memset(text_mask, '*', len + 1);
  255. text_mask[len] = 0;
  256. rtgui_dc_draw_text(dc, text_mask, &rect);
  257. rtgui_free(text_mask);
  258. }
  259. }
  260. else
  261. {
  262. rtgui_dc_draw_text(dc, box->text, &rect);
  263. }
  264. /* draw caret */
  265. if (box->flag & RTGUI_TEXTBOX_CARET_SHOW)
  266. {
  267. rect.x1 += box->position * box->font_width;
  268. rect.x2 = rect.x1 + box->font_width;
  269. rect.y2 -= 2;
  270. rect.y1 = rect.y2 - 3;
  271. RTGUI_WIDGET_BACKGROUND(box) = black;
  272. rtgui_dc_fill_rect(dc, &rect);
  273. }
  274. }
  275. /* end drawing */
  276. rtgui_dc_end_drawing(dc);
  277. }
  278. void rtgui_theme_draw_iconbox(rtgui_iconbox_t* iconbox)
  279. {
  280. struct rtgui_dc* dc;
  281. struct rtgui_rect rect;
  282. struct rtgui_rect text_rect;
  283. /* begin drawing */
  284. dc = rtgui_dc_begin_drawing(RTGUI_WIDGET(iconbox));
  285. if (dc == RT_NULL) return;
  286. /* get widget rect */
  287. rtgui_widget_get_rect(RTGUI_WIDGET(iconbox), &rect);
  288. /* draw icon */
  289. rtgui_image_blit(iconbox->image, dc, &rect);
  290. /* draw text */
  291. if (iconbox->text_position == RTGUI_ICONBOX_TEXT_BELOW && iconbox->text != RT_NULL)
  292. {
  293. rect.y1 = iconbox->image->h + RTGUI_WIDGET_DEFAULT_MARGIN;
  294. rtgui_font_get_metrics(rtgui_dc_get_gc(dc)->font, iconbox->text, &text_rect);
  295. rtgui_rect_moveto_align(&rect, &text_rect, RTGUI_ALIGN_CENTER);
  296. rtgui_dc_draw_text(dc, iconbox->text, &text_rect);
  297. }
  298. else if (iconbox->text_position == RTGUI_ICONBOX_TEXT_RIGHT && iconbox->text != RT_NULL)
  299. {
  300. rect.x1 = iconbox->image->w + RTGUI_WIDGET_DEFAULT_MARGIN;
  301. rtgui_font_get_metrics(rtgui_dc_get_gc(dc)->font, iconbox->text, &text_rect);
  302. rtgui_rect_moveto_align(&rect, &text_rect, RTGUI_ALIGN_CENTER);
  303. rtgui_dc_draw_text(dc, iconbox->text, &text_rect);
  304. }
  305. /* end drawing */
  306. rtgui_dc_end_drawing(dc);
  307. }
  308. static const rt_uint8_t checked_byte[7] = {0x02, 0x06, 0x8E, 0xDC, 0xF8, 0x70, 0x20};
  309. void rtgui_theme_draw_checkbox(struct rtgui_checkbox* checkbox)
  310. {
  311. struct rtgui_dc* dc;
  312. struct rtgui_rect rect, box_rect;
  313. rtgui_color_t bc, fc;
  314. fc = RTGUI_WIDGET_FOREGROUND(checkbox);
  315. bc = RTGUI_WIDGET_BACKGROUND(checkbox);
  316. /* begin drawing */
  317. dc = rtgui_dc_begin_drawing(RTGUI_WIDGET(checkbox));
  318. if (dc == RT_NULL) return;
  319. /* get rect */
  320. rtgui_widget_get_rect(RTGUI_WIDGET(checkbox), &rect);
  321. /* fill rect */
  322. rtgui_dc_fill_rect(dc, &rect);
  323. if (RTGUI_WIDGET_IS_FOCUSED(checkbox))
  324. {
  325. RTGUI_WIDGET_FOREGROUND(checkbox) = black;
  326. /* draw focused border */
  327. rtgui_rect_inflate(&rect, -1);
  328. rtgui_dc_draw_focus_rect(dc, &rect);
  329. rtgui_rect_inflate(&rect, 1);
  330. }
  331. /* draw check box */
  332. box_rect.x1 = 0;
  333. box_rect.y1 = 0;
  334. box_rect.x2 = CHECK_BOX_W;
  335. box_rect.y2 = CHECK_BOX_H;
  336. rtgui_rect_moveto_align(&rect, &box_rect, RTGUI_ALIGN_CENTER_VERTICAL);
  337. box_rect.x1 += 2; box_rect.x2 += 2;
  338. rtgui_dc_draw_border(dc, &box_rect, RTGUI_BORDER_BOX);
  339. rtgui_rect_inflate(&box_rect, -1);
  340. RTGUI_WIDGET_BACKGROUND(checkbox) = RTGUI_RGB(247, 247, 246);
  341. rtgui_dc_fill_rect(dc, &box_rect);
  342. if (checkbox->status_down == RTGUI_CHECKBOX_STATUS_CHECKED)
  343. {
  344. RTGUI_WIDGET_FOREGROUND(checkbox) = RTGUI_RGB(33, 161, 33);
  345. rtgui_dc_draw_byte(dc, box_rect.x1 + 2, box_rect.y1 + 2, 7, checked_byte);
  346. }
  347. /* restore saved color */
  348. RTGUI_WIDGET_BACKGROUND(checkbox) = bc;
  349. RTGUI_WIDGET_FOREGROUND(checkbox) = fc;
  350. /* draw text */
  351. rect.x1 += rtgui_rect_height(rect) - 4 + 5;
  352. rtgui_dc_draw_text(dc, rtgui_label_get_text(RTGUI_LABEL(checkbox)), &rect);
  353. /* end drawing */
  354. rtgui_dc_end_drawing(dc);
  355. return;
  356. }
  357. static const rt_uint8_t radio_unchecked_byte[] =
  358. {
  359. 0x0f, 0x00, 0x30, 0xc0, 0x40, 0x20,
  360. 0x40, 0x20, 0x80, 0x10, 0x80, 0x10,
  361. 0x80, 0x10, 0x80, 0x10, 0x40, 0x20,
  362. 0x40, 0x20, 0x30, 0xc0, 0x0f, 0x00,
  363. };
  364. static const rt_uint8_t radio_checked_byte[] =
  365. {
  366. 0x0f, 0x00, 0x30, 0xc0, 0x40, 0x20,
  367. 0x40, 0x20, 0x86, 0x10, 0x8f, 0x10,
  368. 0x8f, 0x10, 0x86, 0x10, 0x40, 0x20,
  369. 0x40, 0x20, 0x30, 0xc0, 0x0f, 0x00,
  370. };
  371. void rtgui_theme_draw_radiobutton(struct rtgui_radiobox* radiobox, rt_uint16_t item)
  372. {
  373. struct rtgui_dc* dc;
  374. struct rtgui_rect rect, item_rect;
  375. int item_size, bord_size;
  376. /* begin drawing */
  377. dc = rtgui_dc_begin_drawing(RTGUI_WIDGET(radiobox));
  378. if (dc == RT_NULL) return;
  379. /* get widget rect */
  380. rtgui_widget_get_rect(RTGUI_WIDGET(radiobox), &rect);
  381. item_size = radiobox->item_size;
  382. /* get board size */
  383. if (radiobox->orient == RTGUI_VERTICAL)
  384. bord_size = item_size;
  385. else
  386. {
  387. rtgui_font_get_metrics(RTGUI_DC_FONT(dc), "H", &item_rect);
  388. bord_size = rtgui_rect_height(item_rect);
  389. }
  390. item_rect = rect;
  391. rtgui_rect_inflate(&item_rect, - bord_size);
  392. if (radiobox->orient == RTGUI_VERTICAL)
  393. {
  394. /* set the first text rect */
  395. item_rect.y1 += item * item_size;
  396. item_rect.y2 = item_rect.y1 + item_size;
  397. /* draw radio */
  398. if (radiobox->item_selection == item)
  399. {
  400. if (RTGUI_WIDGET_IS_FOCUSED(radiobox))
  401. rtgui_dc_draw_focus_rect(dc, &item_rect);
  402. rtgui_dc_draw_word(dc, item_rect.x1, item_rect.y1 + (item_size - RADIO_BOX_H) / 2,
  403. RADIO_BOX_H, radio_checked_byte);
  404. }
  405. else
  406. {
  407. item_rect.x2 += 1; item_rect.y2 += 1;
  408. rtgui_dc_fill_rect(dc, &item_rect);
  409. item_rect.x2 -= 1; item_rect.y2 -= 1;
  410. rtgui_dc_draw_word(dc, item_rect.x1, item_rect.y1 + (item_size - RADIO_BOX_H) / 2,
  411. RADIO_BOX_H, radio_unchecked_byte);
  412. }
  413. /* draw text */
  414. item_rect.x1 += item_size + 3;
  415. rtgui_dc_draw_text(dc, radiobox->items[item], &item_rect);
  416. }
  417. else
  418. {
  419. item_rect.x1 += item * item_size;
  420. /* set the first text rect */
  421. item_rect.x2 = item_rect.x1 + item_size - 1;
  422. item_rect.y2 = item_rect.y1 + bord_size;
  423. /* draw radio */
  424. if (radiobox->item_selection == item)
  425. {
  426. if (RTGUI_WIDGET_IS_FOCUSED(radiobox))
  427. rtgui_dc_draw_focus_rect(dc, &item_rect);
  428. rtgui_dc_draw_word(dc, item_rect.x1, item_rect.y1, RADIO_BOX_H, radio_checked_byte);
  429. }
  430. else
  431. {
  432. item_rect.x2 += 1; item_rect.y2 += 1;
  433. rtgui_dc_fill_rect(dc, &item_rect);
  434. item_rect.x2 -= 1; item_rect.y2 -= 1;
  435. rtgui_dc_draw_word(dc, item_rect.x1, item_rect.y1, RADIO_BOX_H, radio_unchecked_byte);
  436. }
  437. /* draw text */
  438. item_rect.x1 += bord_size + 3;
  439. rtgui_dc_draw_text(dc, radiobox->items[item], &item_rect);
  440. }
  441. /* end drawing */
  442. rtgui_dc_end_drawing(dc);
  443. }
  444. void rtgui_theme_draw_radiobox(struct rtgui_radiobox* radiobox)
  445. {
  446. struct rtgui_dc* dc;
  447. struct rtgui_rect rect, item_rect;
  448. int item_size, bord_size, index;
  449. rtgui_color_t fc;
  450. /* begin drawing */
  451. dc = rtgui_dc_begin_drawing(RTGUI_WIDGET(radiobox));
  452. if (dc == RT_NULL) return;
  453. /* get widget rect */
  454. rtgui_widget_get_rect(RTGUI_WIDGET(radiobox), &rect);
  455. rtgui_dc_fill_rect(dc, &rect);
  456. item_size = radiobox->item_size;
  457. /* get board size */
  458. if (radiobox->orient == RTGUI_VERTICAL)
  459. bord_size = item_size;
  460. else
  461. {
  462. rtgui_font_get_metrics(RTGUI_DC_FONT(dc), "H", &item_rect);
  463. bord_size = rtgui_rect_height(item_rect);
  464. }
  465. /* draw box */
  466. rtgui_rect_inflate(&rect, -bord_size/2);
  467. fc = RTGUI_WIDGET_FOREGROUND(radiobox);
  468. RTGUI_WIDGET_FOREGROUND(radiobox) = white;
  469. rect.x1 ++; rect.y1 ++; rect.x2 ++; rect.y2 ++;
  470. rtgui_dc_draw_rect(dc, &rect);
  471. RTGUI_WIDGET_FOREGROUND(radiobox) = RTGUI_RGB(128, 128, 128);
  472. rect.x1 --; rect.y1 --; rect.x2 --; rect.y2 --;
  473. rtgui_dc_draw_rect(dc, &rect);
  474. RTGUI_WIDGET_FOREGROUND(radiobox) = fc;
  475. rtgui_rect_inflate(&rect, bord_size/2);
  476. if (radiobox->text != RT_NULL)
  477. {
  478. struct rtgui_rect text_rect;
  479. /* draw group text */
  480. rtgui_font_get_metrics(RTGUI_DC_FONT(dc), radiobox->text, &text_rect);
  481. rtgui_rect_moveto(&text_rect, rect.x1 + bord_size + 5, rect.y1);
  482. rect.x1 -= 5; rect.x2 += 5;
  483. rtgui_dc_fill_rect(dc, &text_rect);
  484. rect.x1 += 5; rect.x2 -= 5;
  485. rtgui_dc_draw_text(dc, radiobox->text, &text_rect);
  486. }
  487. /* set init item rect */
  488. item_rect = rect;
  489. rtgui_rect_inflate(&item_rect, - bord_size);
  490. if (radiobox->orient == RTGUI_VERTICAL)
  491. {
  492. rt_uint16_t offset;
  493. /* set the first text rect */
  494. item_rect.y2 = item_rect.y1 + item_size;
  495. offset = (item_size - RADIO_BOX_H) / 2;
  496. /* draw each radio button */
  497. for (index = 0; index < radiobox->item_count; index ++)
  498. {
  499. if (item_rect.y2 > rect.y2 - item_size) break;
  500. /* draw radio */
  501. if (radiobox->item_selection == index)
  502. {
  503. if (RTGUI_WIDGET_IS_FOCUSED(radiobox))
  504. rtgui_dc_draw_focus_rect(dc, &item_rect);
  505. rtgui_dc_draw_word(dc, item_rect.x1, item_rect.y1 + offset, RADIO_BOX_H, radio_checked_byte);
  506. }
  507. else
  508. {
  509. rtgui_dc_draw_word(dc, item_rect.x1, item_rect.y1 + offset, RADIO_BOX_H, radio_unchecked_byte);
  510. }
  511. /* draw text */
  512. item_rect.x1 += item_size + 3;
  513. rtgui_dc_draw_text(dc, radiobox->items[index], &item_rect);
  514. item_rect.x1 -= item_size + 3;
  515. item_rect.y1 += item_size;
  516. item_rect.y2 += item_size;
  517. }
  518. }
  519. else
  520. {
  521. /* set the first text rect */
  522. item_rect.x2 = item_rect.x1 + item_size;
  523. item_rect.y2 = item_rect.y1 + bord_size;
  524. /* draw each radio button */
  525. for (index = 0; index < radiobox->item_count; index ++)
  526. {
  527. if (item_rect.x2 > rect.x2 - item_size) break;
  528. /* draw radio */
  529. if (radiobox->item_selection == index)
  530. {
  531. if (RTGUI_WIDGET_IS_FOCUSED(radiobox))
  532. rtgui_dc_draw_focus_rect(dc, &item_rect);
  533. rtgui_dc_draw_word(dc, item_rect.x1, item_rect.y1, RADIO_BOX_H, radio_checked_byte);
  534. }
  535. else
  536. {
  537. rtgui_dc_draw_word(dc, item_rect.x1, item_rect.y1, RADIO_BOX_H, radio_unchecked_byte);
  538. }
  539. /* draw text */
  540. item_rect.x1 += bord_size + 3;
  541. rtgui_dc_draw_text(dc, radiobox->items[index], &item_rect);
  542. item_rect.x1 -= bord_size + 3;
  543. item_rect.x1 += item_size;
  544. item_rect.x2 += (item_size - 1);
  545. }
  546. }
  547. /* end drawing */
  548. rtgui_dc_end_drawing(dc);
  549. }
  550. void rtgui_theme_draw_slider(struct rtgui_slider* slider)
  551. {
  552. /* draw button */
  553. struct rtgui_dc* dc;
  554. int i, xsize, x0;
  555. rtgui_rect_t r, focus_rect, slider_rect, slot_rect;
  556. /* begin drawing */
  557. dc = rtgui_dc_begin_drawing(RTGUI_WIDGET(slider));
  558. if (dc == RT_NULL) return;
  559. /* get widget rect */
  560. rtgui_widget_get_rect(RTGUI_WIDGET(slider), &focus_rect);
  561. /* fill widget rect with background color */
  562. rtgui_dc_fill_rect(dc, &focus_rect);
  563. r = focus_rect;
  564. if (slider->orient == RTGUI_VERTICAL)
  565. {
  566. rtgui_rect_inflate(&r, -1);
  567. xsize = r.y2 - r.y1 + 1 - slider->thumb_width;
  568. x0 = r.y1 + slider->thumb_width / 2;
  569. /* calculate thumb position */
  570. slider_rect = r;
  571. slider_rect.x1 = 5;
  572. slider_rect.y1 = x0 + xsize * (slider->value - slider->min) / (slider->max - slider->min) - slider->thumb_width/2;
  573. slider_rect.y2 = slider_rect.y1 + slider->thumb_width;
  574. /* calculate slot position */
  575. slot_rect.y1 = x0;
  576. slot_rect.y2 = x0 + xsize;
  577. slot_rect.x1 = (slider_rect.x1 + slider_rect.x2) /2 -1;
  578. slot_rect.x2 = slot_rect.x1 +3;
  579. /* draw slot */
  580. rtgui_dc_draw_border(dc, &slot_rect, RTGUI_BORDER_RAISE);
  581. /* draw the ticks */
  582. for (i = 0; i <= slider->ticks; i++)
  583. {
  584. int x = x0 + xsize * i / slider->ticks;
  585. rtgui_dc_draw_hline(dc, 1, 3, x);
  586. }
  587. /* draw the thumb */
  588. rtgui_dc_fill_rect(dc, &slider_rect);
  589. rtgui_dc_draw_border(dc, &slider_rect, RTGUI_BORDER_RAISE);
  590. }
  591. else
  592. {
  593. rtgui_rect_inflate(&r, -1);
  594. xsize = r.x2 - r.x1 + 1 - slider->thumb_width;
  595. x0 = r.x1 + slider->thumb_width / 2;
  596. /* calculate thumb position */
  597. slider_rect = r;
  598. slider_rect.y1 = 5;
  599. slider_rect.x1 = x0 + xsize * (slider->value - slider->min) / (slider->max - slider->min) - slider->thumb_width/2;
  600. slider_rect.x2 = slider_rect.x1 + slider->thumb_width;
  601. /* calculate slot position */
  602. slot_rect.x1 = x0;
  603. slot_rect.x2 = x0 + xsize;
  604. slot_rect.y1 = (slider_rect.y1 + slider_rect.y2) /2 -1;
  605. slot_rect.y2 = slot_rect.y1 +3;
  606. /* draw slot */
  607. rtgui_dc_draw_border(dc, &slot_rect, RTGUI_BORDER_RAISE);
  608. /* draw the ticks */
  609. for (i = 0; i <= slider->ticks; i++)
  610. {
  611. int x = x0 + xsize * i / slider->ticks;
  612. rtgui_dc_draw_vline(dc, x, 1, 3);
  613. }
  614. /* draw the thumb */
  615. rtgui_dc_fill_rect(dc, &slider_rect);
  616. rtgui_dc_draw_border(dc, &slider_rect, RTGUI_BORDER_RAISE);
  617. }
  618. /* draw focus */
  619. if (RTGUI_WIDGET_IS_FOCUSED(slider))
  620. {
  621. rtgui_dc_draw_focus_rect(dc, &focus_rect);
  622. }
  623. /* end drawing */
  624. rtgui_dc_end_drawing(dc);
  625. return;
  626. }
  627. const static rt_uint8_t _up_arrow[] = {0x10, 0x38, 0x7C, 0xFE};
  628. const static rt_uint8_t _down_arrow[] = {0xFE,0x7C, 0x38, 0x10};
  629. const static rt_uint8_t _left_arrow[] = {0x10, 0x30, 0x70, 0xF0, 0x70, 0x30, 0x10};
  630. const static rt_uint8_t _right_arrow[] = {0x80, 0xC0, 0xE0, 0xF0, 0xE0, 0xC0, 0x80};
  631. void rtgui_theme_draw_scrollbar(struct rtgui_scrollbar* bar)
  632. {
  633. /* draw scroll bar */
  634. struct rtgui_dc* dc;
  635. rtgui_rect_t rect, btn_rect, thum_rect, arrow_rect;
  636. rtgui_color_t bc, fc;
  637. /* begin drawing */
  638. dc = rtgui_dc_begin_drawing(&(bar->parent));
  639. if (dc == RT_NULL) return;
  640. rtgui_widget_get_rect(RTGUI_WIDGET(bar), &rect);
  641. /* draw background */
  642. fc = RTGUI_WIDGET_FOREGROUND(bar);
  643. if (!RTGUI_WIDGET_IS_ENABLE(bar))
  644. RTGUI_WIDGET_FOREGROUND(bar) = RTGUI_RGB(128, 128, 128);
  645. bc = RTGUI_WIDGET_BACKGROUND(bar);
  646. RTGUI_WIDGET_BACKGROUND(bar) = white;
  647. rtgui_dc_fill_rect(dc, &rect);
  648. RTGUI_WIDGET_BACKGROUND(bar) = bc;
  649. if (bar->orient == RTGUI_VERTICAL)
  650. {
  651. btn_rect = rect;
  652. btn_rect.y2 = btn_rect.y1 + (rect.x2 - rect.x1);
  653. /* draw up button */
  654. rtgui_dc_fill_rect(dc, &btn_rect);
  655. if (bar->status & SBS_UPARROW) rtgui_dc_draw_border(dc, &btn_rect, RTGUI_BORDER_SUNKEN);
  656. else rtgui_dc_draw_border(dc, &btn_rect, RTGUI_BORDER_RAISE);
  657. /* draw arrow */
  658. arrow_rect.x1 = 0; arrow_rect.y1 = 0;
  659. arrow_rect.x2 = 7; arrow_rect.y2 = 4;
  660. rtgui_rect_moveto_align(&btn_rect, &arrow_rect,
  661. RTGUI_ALIGN_CENTER_HORIZONTAL | RTGUI_ALIGN_CENTER_VERTICAL);
  662. rtgui_dc_draw_byte(dc, arrow_rect.x1, arrow_rect.y1,
  663. rtgui_rect_height(arrow_rect), _up_arrow);
  664. /* draw thumb */
  665. if (RTGUI_WIDGET_IS_ENABLE(bar))
  666. {
  667. rtgui_scrollbar_get_thumb_rect(bar, &thum_rect);
  668. rtgui_dc_fill_rect(dc, &thum_rect);
  669. rtgui_dc_draw_border(dc, &thum_rect, RTGUI_BORDER_RAISE);
  670. }
  671. /* draw down button */
  672. btn_rect.y1 = rect.y2 - (rect.x2 - rect.x1);
  673. btn_rect.y2 = rect.y2;
  674. rtgui_dc_fill_rect(dc, &btn_rect);
  675. if (bar->status & SBS_DOWNARROW) rtgui_dc_draw_border(dc, &btn_rect, RTGUI_BORDER_SUNKEN);
  676. else rtgui_dc_draw_border(dc, &btn_rect, RTGUI_BORDER_RAISE);
  677. arrow_rect.x1 = 0; arrow_rect.y1 = 0;
  678. arrow_rect.x2 = 7; arrow_rect.y2 = 4;
  679. rtgui_rect_moveto_align(&btn_rect, &arrow_rect,
  680. RTGUI_ALIGN_CENTER_HORIZONTAL | RTGUI_ALIGN_CENTER_VERTICAL);
  681. rtgui_dc_draw_byte(dc, arrow_rect.x1, arrow_rect.y1,
  682. rtgui_rect_height(arrow_rect), _down_arrow);
  683. }
  684. else
  685. {
  686. btn_rect.x1 = rect.x1;
  687. btn_rect.y1 = rect.y1;
  688. btn_rect.x2 = rect.y2;
  689. btn_rect.y2 = rect.y2;
  690. /* draw left button */
  691. rtgui_dc_fill_rect(dc, &btn_rect);
  692. if (bar->status & SBS_LEFTARROW) rtgui_dc_draw_border(dc, &btn_rect, RTGUI_BORDER_SUNKEN);
  693. else rtgui_dc_draw_border(dc, &btn_rect, RTGUI_BORDER_RAISE);
  694. arrow_rect.x1 = 0; arrow_rect.y1 = 0;
  695. arrow_rect.x2 = 4; arrow_rect.y2 = 7;
  696. rtgui_rect_moveto_align(&btn_rect, &arrow_rect,
  697. RTGUI_ALIGN_CENTER_HORIZONTAL | RTGUI_ALIGN_CENTER_VERTICAL);
  698. rtgui_dc_draw_byte(dc, arrow_rect.x1, arrow_rect.y1,
  699. rtgui_rect_height(arrow_rect), _left_arrow);
  700. /* draw thumb */
  701. if (RTGUI_WIDGET_IS_ENABLE(bar))
  702. {
  703. rtgui_scrollbar_get_thumb_rect(bar, &thum_rect);
  704. rtgui_dc_fill_rect(dc, &thum_rect);
  705. rtgui_dc_draw_border(dc, &thum_rect, RTGUI_BORDER_RAISE);
  706. }
  707. btn_rect.x1 = rect.x2 - rect.y2;
  708. btn_rect.x2 = rect.x2;
  709. /* draw right button */
  710. rtgui_dc_fill_rect(dc, &btn_rect);
  711. if (bar->status & SBS_RIGHTARROW) rtgui_dc_draw_border(dc, &btn_rect, RTGUI_BORDER_SUNKEN);
  712. else rtgui_dc_draw_border(dc, &btn_rect, RTGUI_BORDER_RAISE);
  713. arrow_rect.x1 = 0; arrow_rect.y1 = 0;
  714. arrow_rect.x2 = 4; arrow_rect.y2 = 7;
  715. rtgui_rect_moveto_align(&btn_rect, &arrow_rect,
  716. RTGUI_ALIGN_CENTER_HORIZONTAL | RTGUI_ALIGN_CENTER_VERTICAL);
  717. rtgui_dc_draw_byte(dc, arrow_rect.x1, arrow_rect.y1,
  718. rtgui_rect_height(arrow_rect), _right_arrow);
  719. }
  720. /* end drawing */
  721. rtgui_dc_end_drawing(dc);
  722. RTGUI_WIDGET_FOREGROUND(bar) = fc;
  723. return;
  724. }
  725. void rtgui_theme_draw_progressbar(struct rtgui_progressbar* bar)
  726. {
  727. /* draw progress bar */
  728. struct rtgui_dc* dc;
  729. struct rtgui_rect rect;
  730. int max = bar->range;
  731. int pos = bar->position;
  732. int left;
  733. rtgui_color_t bc;
  734. /* begin drawing */
  735. dc = rtgui_dc_begin_drawing(&(bar->parent));
  736. if (dc == RT_NULL) return;
  737. bc = RTGUI_DC_BC(dc);
  738. rtgui_widget_get_rect(&(bar->parent), &rect);
  739. /* fill button rect with background color */
  740. RTGUI_WIDGET_BACKGROUND(bar) = RTGUI_RGB(212, 208, 200);
  741. /* draw border */
  742. rtgui_dc_draw_border(dc, &rect, RTGUI_BORDER_SUNKEN);
  743. /* Nothing to draw */
  744. if (max == 0)
  745. {
  746. rtgui_dc_end_drawing(dc);
  747. return;
  748. }
  749. rect.x2 ++; rect.y2 ++;
  750. left = max - pos;
  751. rtgui_rect_inflate(&rect, -2);
  752. RTGUI_WIDGET_BACKGROUND(bar) = RTGUI_RGB(0, 0, 255);
  753. rect.y2 --; rect.x2 --;
  754. if (bar->orient == RTGUI_VERTICAL)
  755. {
  756. /* Vertical bar grows from bottom to top */
  757. int dy = (rtgui_rect_height(rect) * left) / max;
  758. rect.y1 += dy;
  759. rtgui_dc_fill_rect(dc, &rect);
  760. RTGUI_DC_BC(dc) = bc;
  761. rect.y1 -= dy; rect.y2 = dy;
  762. rtgui_dc_fill_rect(dc, &rect);
  763. }
  764. else
  765. {
  766. /* Horizontal bar grows from left to right */
  767. int dx = (rtgui_rect_width(rect) * left) / max;
  768. rect.x2 -= dx;
  769. rtgui_dc_fill_rect(dc, &rect);
  770. RTGUI_DC_BC(dc) = bc;
  771. rect.x1 = rect.x2; rect.x2 += dx;
  772. rtgui_dc_fill_rect(dc, &rect);
  773. }
  774. /* end drawing */
  775. rtgui_dc_end_drawing(dc);
  776. return;
  777. }
  778. void rtgui_theme_draw_staticline(struct rtgui_staticline* staticline)
  779. {
  780. struct rtgui_dc* dc;
  781. struct rtgui_rect rect;
  782. /* begin drawing */
  783. dc = rtgui_dc_begin_drawing(RTGUI_WIDGET(staticline));
  784. if (dc == RT_NULL) return ;
  785. rtgui_widget_get_rect(RTGUI_WIDGET(staticline), &rect);
  786. rtgui_dc_fill_rect(dc, &rect);
  787. if (staticline->orient == RTGUI_HORIZONTAL)
  788. {
  789. rtgui_dc_draw_horizontal_line(dc, rect.x1, rect.x2, rect.y1);
  790. }
  791. else
  792. {
  793. rtgui_dc_draw_vertical_line(dc, rect.x1, rect.y1, rect.y2);
  794. }
  795. rtgui_dc_end_drawing(dc);
  796. }
  797. rt_uint16_t rtgui_theme_get_selected_height()
  798. {
  799. return SELECTED_HEIGHT;
  800. }
  801. void rtgui_theme_draw_selected(struct rtgui_dc* dc, rtgui_rect_t *rect)
  802. {
  803. rtgui_color_t bc;
  804. rt_uint16_t index;
  805. bc = RTGUI_DC_FC(dc);
  806. RTGUI_DC_FC(dc) = selected_color;
  807. rtgui_dc_draw_hline(dc, rect->x1 + 3, rect->x2 - 2, rect->y1 + 1);
  808. rtgui_dc_draw_hline(dc, rect->x1 + 3, rect->x2 - 2, rect->y2 - 2);
  809. rtgui_dc_draw_vline(dc, rect->x1 + 2, rect->y1 + 2, rect->y2 - 2);
  810. rtgui_dc_draw_vline(dc, rect->x2 - 2, rect->y1 + 2, rect->y2 - 2);
  811. for (index = rect->y1 + 1; index < rect->y2 - 2; index ++)
  812. rtgui_dc_draw_hline(dc, rect->x1 + 3, rect->x2 - 2, index);
  813. RTGUI_DC_FC(dc) = bc;
  814. }
  815. /* get default background color */
  816. rtgui_color_t rtgui_theme_default_bc()
  817. {
  818. return default_background;
  819. }
  820. /* get default foreground color */
  821. rtgui_color_t rtgui_theme_default_fc()
  822. {
  823. return default_foreground;
  824. }