/src/ftk_check_button.c

http://ftk.googlecode.com/ · C · 326 lines · 248 code · 48 blank · 30 comment · 35 complexity · 79fdf8eaf668b4220f303252993e545c MD5 · raw file

  1. /*
  2. * File: ftk_check_button.c
  3. * Author: Li XianJing <xianjimli@hotmail.com>
  4. * Brief: check button widget
  5. *
  6. * Copyright (c) 2009 - 2010 Li XianJing <xianjimli@hotmail.com>
  7. *
  8. * Licensed under the Academic Free License version 2.1
  9. *
  10. * This program is free software; you can redistribute it and/or modify
  11. * it under the terms of the GNU General Public License as published by
  12. * the Free Software Foundation; either version 2 of the License, or
  13. * (at your option) any later version.
  14. *
  15. * This program is distributed in the hope that it will be useful,
  16. * but WITHOUT ANY WARRANTY; without even the implied warranty of
  17. * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
  18. * GNU General Public License for more details.
  19. *
  20. * You should have received a copy of the GNU General Public License
  21. * along with this program; if not, write to the Free Software
  22. * Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
  23. */
  24. /*
  25. * History:
  26. * ================================================================
  27. * 2012-06-01 woodysu@gmail.com modified
  28. * 2009-11-15 Li XianJing <xianjimli@hotmail.com> created
  29. *
  30. */
  31. #include "ftk_log.h"
  32. #include "ftk_canvas.h"
  33. #include "ftk_globals.h"
  34. #include "ftk_window.h"
  35. #include "ftk_icon_cache.h"
  36. #include "ftk_check_button.h"
  37. #include "ftk_group_box.h"
  38. typedef struct _CheckButtonPrivInfo
  39. {
  40. int checked;
  41. int is_radio;
  42. int icon_at_right;
  43. FtkListener listener;
  44. void* listener_ctx;
  45. FtkBitmap* onimg;
  46. FtkBitmap* offimg;
  47. }PrivInfo;
  48. static Ret ftk_check_button_check(FtkWidget* thiz)
  49. {
  50. Ret ret = RET_OK;
  51. DECL_PRIV0(thiz, priv);
  52. if(priv->is_radio && ftk_widget_type(ftk_widget_parent(thiz)) == FTK_GROUP_BOX)
  53. {
  54. ret = ftk_group_box_set_checked(ftk_widget_parent(thiz), thiz);
  55. }
  56. else
  57. {
  58. ret = ftk_check_button_set_checked(thiz, !priv->checked);
  59. }
  60. return ret;
  61. }
  62. static Ret ftk_check_button_on_event(FtkWidget* thiz, FtkEvent* event)
  63. {
  64. Ret ret = RET_OK;
  65. DECL_PRIV0(thiz, priv);
  66. switch(event->type)
  67. {
  68. case FTK_EVT_MOUSE_DOWN:
  69. {
  70. ftk_widget_set_active(thiz, 1);
  71. ftk_window_grab(ftk_widget_toplevel(thiz), thiz);
  72. break;
  73. }
  74. case FTK_EVT_MOUSE_UP:
  75. {
  76. int x = event->u.mouse.x;
  77. int y = event->u.mouse.y;
  78. if(ftk_widget_is_active(thiz))
  79. {
  80. ftk_widget_set_active(thiz, 0);
  81. ftk_window_ungrab(ftk_widget_toplevel(thiz), thiz);
  82. if(FTK_POINT_IN_WIDGET(x, y, thiz))
  83. {
  84. ret = FTK_CALL_LISTENER(priv->listener, priv->listener_ctx, thiz);
  85. ftk_check_button_check(thiz);
  86. }
  87. }
  88. break;
  89. }
  90. case FTK_EVT_KEY_DOWN:
  91. {
  92. if(FTK_IS_ACTIVE_KEY(event->u.key.code))
  93. {
  94. ftk_widget_set_active(thiz, 1);
  95. }
  96. break;
  97. }
  98. case FTK_EVT_KEY_UP:
  99. {
  100. if(FTK_IS_ACTIVE_KEY(event->u.key.code) && ftk_widget_is_active(thiz))
  101. {
  102. ftk_widget_set_active(thiz, 0);
  103. ftk_check_button_check(thiz);
  104. ret = FTK_CALL_LISTENER(priv->listener, priv->listener_ctx, thiz);
  105. }
  106. break;
  107. }
  108. default:break;
  109. }
  110. return ret;
  111. }
  112. static const char* check_bg_on_imgs[FTK_WIDGET_STATE_NR] =
  113. {
  114. "btn_check_on"FTK_STOCK_IMG_SUFFIX,
  115. "btn_check_on_selected"FTK_STOCK_IMG_SUFFIX,
  116. "btn_check_on_pressed"FTK_STOCK_IMG_SUFFIX,
  117. "btn_check_on_disable"FTK_STOCK_IMG_SUFFIX
  118. };
  119. static const char* check_bg_off_imgs[FTK_WIDGET_STATE_NR] =
  120. {
  121. "btn_check_off"FTK_STOCK_IMG_SUFFIX,
  122. "btn_check_off_selected"FTK_STOCK_IMG_SUFFIX,
  123. "btn_check_off_pressed"FTK_STOCK_IMG_SUFFIX,
  124. "btn_check_off_disable"FTK_STOCK_IMG_SUFFIX
  125. };
  126. static const char* radio_bg_on_imgs[FTK_WIDGET_STATE_NR] =
  127. {
  128. "btn_radio_on"FTK_STOCK_IMG_SUFFIX,
  129. "btn_radio_on_selected"FTK_STOCK_IMG_SUFFIX,
  130. "btn_radio_on_pressed"FTK_STOCK_IMG_SUFFIX,
  131. "btn_radio_on"FTK_STOCK_IMG_SUFFIX
  132. };
  133. static const char* radio_bg_off_imgs[FTK_WIDGET_STATE_NR] =
  134. {
  135. "btn_radio_off"FTK_STOCK_IMG_SUFFIX,
  136. "btn_radio_off_selected"FTK_STOCK_IMG_SUFFIX,
  137. "btn_radio_off_pressed"FTK_STOCK_IMG_SUFFIX,
  138. "btn_radio_off"FTK_STOCK_IMG_SUFFIX,
  139. };
  140. static Ret ftk_check_button_on_paint(FtkWidget* thiz)
  141. {
  142. int dx = 0;
  143. int dy = 0;
  144. int icon_w = 0;
  145. int icon_h = 0;
  146. FtkBitmap* bitmap = NULL;
  147. DECL_PRIV0(thiz, priv);
  148. const char** bg_imgs = NULL;
  149. FTK_BEGIN_PAINT(x, y, width, height, canvas);
  150. if (priv->onimg && priv->offimg) {
  151. if (priv->checked)
  152. bitmap = priv->onimg;
  153. else
  154. bitmap = priv->offimg;
  155. }
  156. else { // orignal code
  157. if(priv->is_radio)
  158. {
  159. bg_imgs = priv->checked ? radio_bg_on_imgs : radio_bg_off_imgs;
  160. }
  161. else
  162. {
  163. bg_imgs = priv->checked ? check_bg_on_imgs : check_bg_off_imgs;
  164. }
  165. bitmap = ftk_theme_load_image(ftk_default_theme(), bg_imgs[ftk_widget_state(thiz)]);
  166. }
  167. return_val_if_fail(bitmap != NULL, RET_FAIL);
  168. icon_w = ftk_bitmap_width(bitmap);
  169. icon_h = ftk_bitmap_height(bitmap);
  170. assert((icon_w) <= width && icon_h <= height);
  171. dy = (height - icon_h) / 2;
  172. dx = priv->icon_at_right ? width - icon_w : 0;
  173. ftk_canvas_draw_bitmap_simple(canvas, bitmap, 0, 0, icon_w, icon_h, x + dx, y + dy);
  174. if (NULL == priv->onimg)
  175. ftk_bitmap_unref(bitmap);
  176. if(ftk_widget_get_text(thiz) != NULL)
  177. {
  178. dy = height/2;
  179. dx = priv->icon_at_right ? 2 : icon_w;
  180. ftk_canvas_set_gc(canvas, ftk_widget_get_gc(thiz));
  181. ftk_canvas_draw_string(canvas, x + dx, y + dy, ftk_widget_get_text(thiz), -1, 1);
  182. }
  183. FTK_END_PAINT();
  184. }
  185. static void ftk_check_button_destroy(FtkWidget* thiz)
  186. {
  187. if(thiz != NULL)
  188. {
  189. DECL_PRIV0(thiz, priv);
  190. ftk_bitmap_unref(priv->onimg);
  191. ftk_bitmap_unref(priv->offimg);
  192. FTK_ZFREE(priv, sizeof(PrivInfo));
  193. }
  194. return;
  195. }
  196. FtkWidget* ftk_check_button_create_ex(FtkWidget* parent, int x, int y, int width, int height, int radio)
  197. {
  198. FtkWidget* thiz = (FtkWidget*)FTK_ZALLOC(sizeof(FtkWidget));
  199. return_val_if_fail(thiz != NULL, NULL);
  200. thiz->priv_subclass[0] = (PrivInfo*)FTK_ZALLOC(sizeof(PrivInfo));
  201. if(thiz->priv_subclass[0] != NULL)
  202. {
  203. DECL_PRIV0(thiz, priv);
  204. priv->is_radio = radio;
  205. thiz->on_event = ftk_check_button_on_event;
  206. thiz->on_paint = ftk_check_button_on_paint;
  207. thiz->destroy = ftk_check_button_destroy;
  208. ftk_widget_init(thiz, radio ? FTK_RADIO_BUTTON : FTK_CHECK_BUTTON, 0, x, y, width, height, FTK_ATTR_TRANSPARENT);
  209. ftk_widget_append_child(parent, thiz);
  210. }
  211. else
  212. {
  213. FTK_FREE(thiz);
  214. }
  215. return thiz;
  216. }
  217. FtkWidget* ftk_check_button_create(FtkWidget* parent, int x, int y, int width, int height)
  218. {
  219. return ftk_check_button_create_ex(parent, x, y, width, height, 0);
  220. }
  221. FtkWidget* ftk_check_button_create_radio(FtkWidget* parent, int x, int y, int width, int height)
  222. {
  223. return ftk_check_button_create_ex(parent, x, y, width, height, 1);
  224. }
  225. int ftk_check_button_get_checked(FtkWidget* thiz)
  226. {
  227. DECL_PRIV0(thiz, priv);
  228. return_val_if_fail(thiz != NULL, 0);
  229. return priv->checked;
  230. }
  231. Ret ftk_check_button_set_image(FtkWidget* thiz, FtkBitmap* onimg, FtkBitmap* offimg)
  232. {
  233. DECL_PRIV0(thiz, priv);
  234. return_val_if_fail(thiz != NULL, RET_FAIL);
  235. ftk_bitmap_unref(priv->onimg);
  236. ftk_bitmap_unref(priv->offimg);
  237. priv->onimg = onimg;
  238. ftk_bitmap_ref(priv->onimg);
  239. priv->offimg = offimg;
  240. ftk_bitmap_ref(priv->offimg);
  241. return RET_OK;
  242. }
  243. Ret ftk_check_button_set_icon_position(FtkWidget* thiz, int at_right)
  244. {
  245. DECL_PRIV0(thiz, priv);
  246. return_val_if_fail(thiz != NULL, RET_FAIL);
  247. if(priv->icon_at_right != at_right)
  248. {
  249. priv->icon_at_right = at_right;
  250. ftk_widget_invalidate(thiz);
  251. }
  252. return RET_OK;
  253. }
  254. Ret ftk_check_button_set_checked(FtkWidget* thiz, int checked)
  255. {
  256. DECL_PRIV0(thiz, priv);
  257. return_val_if_fail(thiz != NULL, RET_FAIL);
  258. if(priv->checked != checked)
  259. {
  260. priv->checked = checked;
  261. ftk_widget_invalidate(thiz);
  262. }
  263. return RET_OK;
  264. }
  265. int ftk_check_button_is_checked(FtkWidget* thiz)
  266. {
  267. DECL_PRIV0(thiz, priv);
  268. return_val_if_fail(thiz != NULL, RET_FAIL);
  269. return priv->checked;
  270. }
  271. Ret ftk_check_button_set_clicked_listener(FtkWidget* thiz, FtkListener listener, void* ctx)
  272. {
  273. DECL_PRIV0(thiz, priv);
  274. return_val_if_fail(thiz != NULL, RET_FAIL);
  275. priv->listener_ctx = ctx;
  276. priv->listener = listener;
  277. return RET_OK;
  278. }