/src/ftk_scroll_bar.c

http://ftk.googlecode.com/ · C · 391 lines · 292 code · 68 blank · 31 comment · 49 complexity · ff4184f3a8806b660a9d0d8fd4807a67 MD5 · raw file

  1. /*
  2. * File: ftk_scroll_bar.c
  3. * Author: Li XianJing <xianjimli@hotmail.com>
  4. * Brief: scroll bar
  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. * 2009-11-20 Li XianJing <xianjimli@hotmail.com> created
  28. *
  29. */
  30. #include "ftk_window.h"
  31. #include "ftk_globals.h"
  32. #include "ftk_scroll_bar.h"
  33. typedef struct _ScrollBarPrivInfo
  34. {
  35. int tracker_pos;
  36. int tracker_size;
  37. int drag_enabled;
  38. int mouse_pressed;
  39. int last_mouse_pos;
  40. int value;
  41. int vertical;
  42. int max_value;
  43. int page_delta;
  44. FtkBitmap* bitmap;
  45. void* listener_ctx;
  46. FtkListener listener;
  47. }PrivInfo;
  48. static Ret ftk_scroll_bar_on_event(FtkWidget* thiz, FtkEvent* event)
  49. {
  50. int x = 0;
  51. int y = 0;
  52. Ret ret = RET_OK;
  53. DECL_PRIV0(thiz, priv);
  54. if(priv->page_delta == priv->max_value)
  55. {
  56. /*one page, no scroll*/
  57. return RET_OK;
  58. }
  59. switch(event->type)
  60. {
  61. case FTK_EVT_KEY_DOWN:
  62. {
  63. if(event->u.key.code == FTK_KEY_DOWN)
  64. {
  65. ftk_scroll_bar_inc(thiz);
  66. ret = RET_REMOVE;
  67. }
  68. else if(event->u.key.code == FTK_KEY_UP)
  69. {
  70. ftk_scroll_bar_dec(thiz);
  71. ret = RET_REMOVE;
  72. }
  73. else if(event->u.key.code == FTK_KEY_PAGEDOWN)
  74. {
  75. ftk_scroll_bar_pagedown(thiz);
  76. }
  77. else if(event->u.key.code == FTK_KEY_PAGEUP)
  78. {
  79. ftk_scroll_bar_pageup(thiz);
  80. }
  81. break;
  82. }
  83. case FTK_EVT_MOUSE_DOWN:
  84. {
  85. int pos = 0;
  86. int value = 0;
  87. x = event->u.mouse.x;
  88. y = event->u.mouse.y;
  89. priv->mouse_pressed = 1;
  90. ftk_window_grab(ftk_widget_toplevel(thiz), thiz);
  91. pos = priv->vertical ? y : x;
  92. priv->drag_enabled = pos >= priv->tracker_pos && pos < (priv->tracker_pos + priv->tracker_size);
  93. if(!priv->drag_enabled)
  94. {
  95. value = pos < priv->tracker_pos ? priv->value - priv->page_delta : priv->value + priv->page_delta;
  96. ftk_scroll_bar_set_value(thiz, value);
  97. }
  98. priv->last_mouse_pos = pos;
  99. break;
  100. }
  101. case FTK_EVT_MOUSE_MOVE:
  102. {
  103. int pos = 0;
  104. int delta =0;
  105. int value = 0;
  106. int width = 0;
  107. int height = 0;
  108. if(!priv->drag_enabled) break;
  109. x = event->u.mouse.x;
  110. y = event->u.mouse.y;
  111. pos = priv->vertical ? y : x;
  112. delta = pos - priv->last_mouse_pos;
  113. width = ftk_widget_width(thiz);
  114. height = ftk_widget_height(thiz);
  115. value = priv->vertical ? priv->max_value * delta/height : priv->max_value * delta/width;
  116. if(value != 0)
  117. {
  118. value += priv->value;
  119. priv->last_mouse_pos = pos;
  120. ftk_scroll_bar_set_value(thiz, value);
  121. }
  122. break;
  123. }
  124. case FTK_EVT_MOUSE_UP:
  125. {
  126. priv->drag_enabled = 0;
  127. priv->mouse_pressed = 0;
  128. ftk_window_ungrab(ftk_widget_toplevel(thiz), thiz);
  129. break;
  130. }
  131. default:break;
  132. }
  133. return ret;
  134. }
  135. static Ret ftk_scroll_bar_on_paint(FtkWidget* thiz)
  136. {
  137. int i = 0;
  138. int dx = 0;
  139. int dy = 0;
  140. int fill_size = 0;
  141. int half_size = 0;
  142. int bitmap_width = 0;
  143. int bitmap_height = 0;
  144. DECL_PRIV0(thiz, priv);
  145. FTK_BEGIN_PAINT(x, y, width, height, canvas);
  146. return_val_if_fail(priv->bitmap != NULL, RET_FAIL);
  147. bitmap_width = ftk_bitmap_width(priv->bitmap);
  148. bitmap_height = ftk_bitmap_height(priv->bitmap);
  149. if(priv->vertical)
  150. {
  151. priv->tracker_size = height * priv->page_delta/priv->max_value;
  152. priv->tracker_size = priv->tracker_size < bitmap_height ? bitmap_height : priv->tracker_size;
  153. dy = height * priv->value / priv->max_value;
  154. dy = (dy + priv->tracker_size) < height ? dy : height - priv->tracker_size;
  155. priv->tracker_pos = dy + ftk_widget_top_abs(thiz);
  156. fill_size = priv->tracker_size - bitmap_height;
  157. dx += x;
  158. dy += y;
  159. half_size = bitmap_height/2;
  160. ftk_canvas_draw_bitmap_simple(canvas, priv->bitmap, 0, 0, bitmap_width, half_size, dx, dy);
  161. for(i = 0; i < fill_size; i++)
  162. {
  163. ftk_canvas_draw_bitmap_simple(canvas, priv->bitmap, 0, half_size, bitmap_width, 1, dx, dy + i + half_size);
  164. }
  165. ftk_canvas_draw_bitmap_simple(canvas, priv->bitmap, 0, half_size, bitmap_width, half_size, dx, dy + half_size + fill_size);
  166. }
  167. else
  168. {
  169. priv->tracker_size = width * priv->page_delta/priv->max_value;
  170. priv->tracker_size = priv->tracker_size < bitmap_width ? bitmap_width : priv->tracker_size;
  171. dx = width * priv->value / priv->max_value;
  172. dx = (dx + priv->tracker_size) < width ? dx : width - priv->tracker_size;
  173. priv->tracker_pos = dx + ftk_widget_left_abs(thiz);
  174. fill_size = priv->tracker_size - bitmap_width;
  175. dy += y;
  176. dx += x;
  177. half_size = bitmap_width/2;
  178. ftk_canvas_draw_bitmap_simple(canvas, priv->bitmap, 0, 0, half_size, bitmap_height, dx, dy);
  179. for(i = 0; i < fill_size; i++)
  180. {
  181. ftk_canvas_draw_bitmap_simple(canvas, priv->bitmap, half_size, 0, 1, bitmap_height, dx + i + half_size, dy);
  182. }
  183. ftk_canvas_draw_bitmap_simple(canvas, priv->bitmap, half_size, 0, half_size, bitmap_height, dx + fill_size + half_size, dy);
  184. }
  185. FTK_END_PAINT();
  186. }
  187. static void ftk_scroll_bar_destroy(FtkWidget* thiz)
  188. {
  189. if(thiz != NULL)
  190. {
  191. DECL_PRIV0(thiz, priv);
  192. ftk_bitmap_unref(priv->bitmap);
  193. FTK_ZFREE(priv, sizeof(PrivInfo));
  194. }
  195. return;
  196. }
  197. FtkWidget* ftk_scroll_bar_create(FtkWidget* parent, int x, int y, int width, int height)
  198. {
  199. FtkWidget* thiz = (FtkWidget*)FTK_ZALLOC(sizeof(FtkWidget));
  200. return_val_if_fail(thiz != NULL, NULL);
  201. thiz->priv_subclass[0] = (PrivInfo*)FTK_ZALLOC(sizeof(PrivInfo));
  202. if(thiz != NULL)
  203. {
  204. DECL_PRIV0(thiz, priv);
  205. thiz->on_event = ftk_scroll_bar_on_event;
  206. thiz->on_paint = ftk_scroll_bar_on_paint;
  207. thiz->destroy = ftk_scroll_bar_destroy;
  208. if(width < height)
  209. {
  210. /*vertical*/
  211. priv->vertical = 1;
  212. priv->bitmap = ftk_theme_load_image(ftk_default_theme(),
  213. "scrollbar_handle_vertical"FTK_STOCK_IMG_SUFFIX);
  214. width = ftk_bitmap_width(priv->bitmap);
  215. assert(width < height);
  216. }
  217. else
  218. {
  219. priv->vertical = 0;
  220. priv->bitmap = ftk_theme_load_image(ftk_default_theme(),
  221. "scrollbar_handle_horizontal"FTK_STOCK_IMG_SUFFIX);
  222. height = ftk_bitmap_height(priv->bitmap);
  223. assert(width > height);
  224. }
  225. ftk_widget_init(thiz, width < height ? FTK_SCROLL_VBAR : FTK_SCROLL_HBAR, 0,
  226. x, y, width, height, FTK_ATTR_TRANSPARENT);
  227. ftk_widget_append_child(parent, thiz);
  228. }
  229. return thiz;
  230. }
  231. Ret ftk_scroll_bar_set_bitmap(FtkWidget* thiz, FtkBitmap* bitmap)
  232. {
  233. DECL_PRIV0(thiz, priv);
  234. return_val_if_fail(priv != NULL, RET_FAIL);
  235. ftk_bitmap_unref(priv->bitmap);
  236. priv->bitmap = bitmap;
  237. ftk_bitmap_ref(priv->bitmap);
  238. return RET_OK;
  239. }
  240. Ret ftk_scroll_bar_set_param(FtkWidget* thiz, int value, int max_value, int page_delta)
  241. {
  242. DECL_PRIV0(thiz, priv);
  243. return_val_if_fail(priv != NULL, RET_FAIL);
  244. return_val_if_fail(value <= max_value && page_delta <= max_value, RET_FAIL);
  245. return_val_if_fail(max_value > 0 && page_delta > 0, RET_FAIL);
  246. if(priv->value == value
  247. && priv->max_value == max_value
  248. && priv->page_delta == page_delta)
  249. {
  250. return RET_OK;
  251. }
  252. priv->value = value < 0 ? 0 : value;
  253. priv->max_value = max_value;
  254. priv->page_delta = page_delta;
  255. ftk_widget_invalidate(thiz);
  256. return RET_OK;
  257. }
  258. Ret ftk_scroll_bar_set_listener(FtkWidget* thiz, FtkListener listener, void* ctx)
  259. {
  260. DECL_PRIV0(thiz, priv);
  261. return_val_if_fail(priv != NULL, RET_FAIL);
  262. priv->listener = listener;
  263. priv->listener_ctx = ctx;
  264. return RET_OK;
  265. }
  266. int ftk_scroll_bar_get_value(FtkWidget* thiz)
  267. {
  268. DECL_PRIV0(thiz, priv);
  269. return_val_if_fail(priv != NULL, 0);
  270. return priv->value;
  271. }
  272. int ftk_scroll_bar_get_max_value(FtkWidget* thiz)
  273. {
  274. DECL_PRIV0(thiz, priv);
  275. return_val_if_fail(priv != NULL, 0);
  276. return priv->max_value;
  277. }
  278. Ret ftk_scroll_bar_inc(FtkWidget* thiz)
  279. {
  280. int value = 0;
  281. DECL_PRIV0(thiz, priv);
  282. return_val_if_fail(priv != NULL, RET_FAIL);
  283. value = priv->value + 1;
  284. return ftk_scroll_bar_set_value(thiz, value);
  285. }
  286. Ret ftk_scroll_bar_dec(FtkWidget* thiz)
  287. {
  288. int value = 0;
  289. DECL_PRIV0(thiz, priv);
  290. return_val_if_fail(priv != NULL, RET_FAIL);
  291. value = priv->value - 1;
  292. return ftk_scroll_bar_set_value(thiz, value);
  293. }
  294. Ret ftk_scroll_bar_pageup(FtkWidget* thiz)
  295. {
  296. int value = 0;
  297. DECL_PRIV0(thiz, priv);
  298. return_val_if_fail(priv != NULL, RET_FAIL);
  299. value = priv->value - priv->page_delta;
  300. return ftk_scroll_bar_set_value(thiz, value);
  301. }
  302. Ret ftk_scroll_bar_pagedown(FtkWidget* thiz)
  303. {
  304. int value = 0;
  305. DECL_PRIV0(thiz, priv);
  306. return_val_if_fail(priv != NULL, RET_FAIL);
  307. value = priv->value + priv->page_delta;
  308. return ftk_scroll_bar_set_value(thiz, value);
  309. }
  310. Ret ftk_scroll_bar_set_value(FtkWidget* thiz, int value)
  311. {
  312. DECL_PRIV0(thiz, priv);
  313. return_val_if_fail(priv != NULL, RET_FAIL);
  314. value = value < 0 ? 0 : value;
  315. value = (value + priv->page_delta) < priv->max_value ? value : priv->max_value - priv->page_delta;
  316. if(value != priv->value)
  317. {
  318. priv->value = value;
  319. ftk_widget_invalidate(thiz);
  320. return FTK_CALL_LISTENER(priv->listener, priv->listener_ctx, thiz);
  321. }
  322. return RET_OK;
  323. }