filelist_view.c 21 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839
  1. /*
  2. * File : filelist_view.c
  3. * This file is part of RTGUI in RT-Thread RTOS
  4. * COPYRIGHT (C) 2010, 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. * 2010-01-06 Bernard first version
  13. */
  14. #include <rtgui/rtgui_object.h>
  15. #include <rtgui/rtgui_system.h>
  16. #include <rtgui/rtgui_theme.h>
  17. #include <rtgui/list.h>
  18. #include <rtgui/image.h>
  19. #include <rtgui/widgets/view.h>
  20. #include <rtgui/widgets/workbench.h>
  21. #include <rtgui/widgets/filelist_view.h>
  22. #include <rtgui/widgets/listbox.h>
  23. #include <rtgui/widgets/window.h>
  24. #if defined(RTGUI_USING_DFS_FILERW) || defined(RTGUI_USING_STDIO_FILERW)
  25. #ifdef _WIN32
  26. #include <io.h>
  27. #include <dirent.h>
  28. #include <sys/stat.h>
  29. #define PATH_SEPARATOR '\\'
  30. #define stat _stat
  31. #else
  32. #include <dfs_posix.h>
  33. #define PATH_SEPARATOR '/'
  34. #endif
  35. #include <string.h>
  36. #define RTGUI_FILELIST_MARGIN 5
  37. const static char * file_xpm[] = {
  38. "16 16 21 1",
  39. " c None",
  40. ". c #999999",
  41. "+ c #818181",
  42. "@ c #FFFFFF",
  43. "# c #ECECEC",
  44. "$ c #EAEAEA",
  45. "% c #EBEBEB",
  46. "& c #EDEDED",
  47. "* c #F0F0F0",
  48. "= c #C4C4C4",
  49. "- c #C5C5C5",
  50. "; c #C6C6C6",
  51. "> c #C7C7C7",
  52. ", c #EEEEEE",
  53. "' c #EDEDE5",
  54. ") c #EDEDE6",
  55. "! c #EFEFEF",
  56. "~ c #C8C8C8",
  57. "{ c #F1F1F1",
  58. "] c #F2F2F2",
  59. "^ c #959595",
  60. ".++++++++++++ ",
  61. "+@@@@@@@@@@@@+ ",
  62. "+@#$$%%%##&*@+ ",
  63. "+@$=--;;;;>*@+ ",
  64. "+@$%%###&&,*@+ ",
  65. "+@%-;;;;;;>*@+ ",
  66. "+@%%##&&'#,*@+ ",
  67. "+@%;;;;,,),*@+ ",
  68. "+@##&&,,!!!*@+ ",
  69. "+@#;;;>>~~~*@+ ",
  70. "+@#&,,!!*{{{@+ ",
  71. "+@&;>>~~~{{]@+ ",
  72. "+@&&,!!**{]]@+ ",
  73. "+@@@@@@@@@@@@+ ",
  74. "^++++++++++++^ ",
  75. " "};
  76. const static char * folder_xpm[] = {
  77. "16 16 121 2",
  78. " c None",
  79. ". c #D9B434",
  80. "+ c #E1C25E",
  81. "@ c #E2C360",
  82. "# c #E2C35F",
  83. "$ c #DBB63C",
  84. "% c #DAB336",
  85. "& c #FEFEFD",
  86. "* c #FFFFFE",
  87. "= c #FFFEFE",
  88. "- c #FFFEFD",
  89. "; c #FBF7EA",
  90. "> c #E4C76B",
  91. ", c #E3C76B",
  92. "' c #E6CD79",
  93. ") c #E5CA74",
  94. "! c #DAAF35",
  95. "~ c #FEFCF7",
  96. "{ c #F8E48E",
  97. "] c #F5DE91",
  98. "^ c #F5E09F",
  99. "/ c #F6E1AC",
  100. "( c #FEFBEF",
  101. "_ c #FEFDF4",
  102. ": c #FEFCF3",
  103. "< c #FEFCF1",
  104. "[ c #FEFBEE",
  105. "} c #FFFDFA",
  106. "| c #DAAF36",
  107. "1 c #DAAA36",
  108. "2 c #FDFAF1",
  109. "3 c #F5DE94",
  110. "4 c #F4DC93",
  111. "5 c #F2D581",
  112. "6 c #EDCA6A",
  113. "7 c #EACB6C",
  114. "8 c #EFD385",
  115. "9 c #EFD280",
  116. "0 c #EFD07A",
  117. "a c #EECF76",
  118. "b c #EECF72",
  119. "c c #FBF7E9",
  120. "d c #DAAE34",
  121. "e c #DAAB35",
  122. "f c #FBF6E8",
  123. "g c #EFD494",
  124. "h c #EECE88",
  125. "i c #E9C173",
  126. "j c #F6E9C9",
  127. "k c #FEFCF2",
  128. "l c #FEFCF0",
  129. "m c #DAAB36",
  130. "n c #DAA637",
  131. "o c #FFFDF8",
  132. "p c #FFFDF6",
  133. "q c #FFFCF5",
  134. "r c #FCF6D8",
  135. "s c #F8E694",
  136. "t c #F7E385",
  137. "u c #F6DF76",
  138. "v c #F5DB68",
  139. "w c #F4D85C",
  140. "x c #FCF4D7",
  141. "y c #DAA435",
  142. "z c #DAA136",
  143. "A c #FEFCF6",
  144. "B c #FCF2C8",
  145. "C c #FBEFB9",
  146. "D c #FAECAC",
  147. "E c #F9E89C",
  148. "F c #F7E38B",
  149. "G c #F6E07C",
  150. "H c #F6DC6C",
  151. "I c #F5D95D",
  152. "J c #F4D64F",
  153. "K c #F3D344",
  154. "L c #FCF3D0",
  155. "M c #DA9F35",
  156. "N c #DA9A36",
  157. "O c #FDFAF2",
  158. "P c #FAEDB3",
  159. "Q c #F9E9A4",
  160. "R c #F8E695",
  161. "S c #F7E285",
  162. "T c #F6DE76",
  163. "U c #F5DB65",
  164. "V c #F4D757",
  165. "W c #F3D449",
  166. "X c #F2D13B",
  167. "Y c #F1CE30",
  168. "Z c #FBF2CC",
  169. "` c #DA9835",
  170. " . c #DA9435",
  171. ".. c #FEFAEF",
  172. "+. c #F9E9A1",
  173. "@. c #F8E591",
  174. "#. c #F7E181",
  175. "$. c #F6DE72",
  176. "%. c #F5DA63",
  177. "&. c #F4D754",
  178. "*. c #F3D347",
  179. "=. c #F2D039",
  180. "-. c #F1CD2E",
  181. ";. c #F0CB26",
  182. ">. c #FBF2CA",
  183. ",. c #D98E33",
  184. "'. c #FAF0DC",
  185. "). c #F4DDA7",
  186. "!. c #F4DB9E",
  187. "~. c #F3DA96",
  188. "{. c #F3D88E",
  189. "]. c #F3D786",
  190. "^. c #F2D47F",
  191. "/. c #F2D379",
  192. "(. c #F1D272",
  193. "_. c #F1D06C",
  194. ":. c #F1CF69",
  195. "<. c #F8EAC2",
  196. "[. c #D8882D",
  197. "}. c #D8872D",
  198. "|. c #D8862C",
  199. " ",
  200. " ",
  201. " ",
  202. " . + @ @ @ # $ ",
  203. " % & * = - * ; > , , , ' ) ",
  204. " ! ~ { ] ^ / ( _ : < ( [ } | ",
  205. " 1 2 3 4 5 6 7 8 9 0 a b c d ",
  206. " e f g h i j k : k l ( [ * m ",
  207. " n * o p q : r s t u v w x y ",
  208. " z A B C D E F G H I J K L M ",
  209. " N O P Q R S T U V W X Y Z ` ",
  210. " ...+.@.#.$.%.&.*.=.-.;.>. . ",
  211. " ,.'.).!.~.{.].^./.(._.:.<.,. ",
  212. " [.}.[.[.[.[.[.[.[.[.}.[.|. ",
  213. " ",
  214. " "};
  215. /* image for file and folder */
  216. static rtgui_image_t *file_image, *folder_image;
  217. static struct rtgui_listbox_item items[] =
  218. {
  219. {"打开文件夹", RT_NULL},
  220. {"选择文件夹", RT_NULL},
  221. {"退出", RT_NULL}
  222. };
  223. static void rtgui_filelist_view_clear(rtgui_filelist_view_t* view);
  224. static void rtgui_filelist_view_on_folder_item(rtgui_widget_t* widget, struct rtgui_event* event)
  225. {
  226. rtgui_win_t *menu;
  227. rtgui_listbox_t *listbox;
  228. rtgui_filelist_view_t *view;
  229. listbox = RTGUI_LISTBOX(widget);
  230. menu = RTGUI_WIN(rtgui_widget_get_toplevel(widget));
  231. view = RTGUI_FILELIST_VIEW(menu->user_data);
  232. /* hide window */
  233. rtgui_win_hiden(menu);
  234. switch (listbox->current_item)
  235. {
  236. case 0:
  237. {
  238. char* dir_ptr;
  239. /* destroy menu window */
  240. rtgui_win_destroy(menu);
  241. dir_ptr = (char*) rtgui_malloc (256);
  242. rtgui_filelist_view_get_fullpath(view, dir_ptr, 256);
  243. rtgui_filelist_view_set_directory(view, dir_ptr);
  244. rt_free(dir_ptr);
  245. }
  246. break;
  247. case 1:
  248. /* destroy menu window */
  249. rtgui_win_destroy(menu);
  250. if (RTGUI_VIEW(view)->modal_show == RT_TRUE)
  251. {
  252. rtgui_view_end_modal(RTGUI_VIEW(view), RTGUI_MODAL_OK);
  253. }
  254. break;
  255. default:
  256. /* destroy menu window */
  257. rtgui_win_destroy(menu);
  258. break;
  259. }
  260. }
  261. static rt_bool_t rtgui_filelist_view_on_menu_deactivate(rtgui_widget_t* widget, struct rtgui_event* event)
  262. {
  263. rtgui_win_t *menu;
  264. menu = RTGUI_WIN(rtgui_widget_get_toplevel(widget));
  265. /* destroy menu window */
  266. rtgui_win_destroy(menu);
  267. return RT_TRUE;
  268. }
  269. static void rtgui_filelist_view_menu_pop(rtgui_widget_t *parent)
  270. {
  271. rtgui_win_t *menu;
  272. rtgui_listbox_t *listbox;
  273. rtgui_rect_t screen, rect = {0, 0, 140, 85};
  274. rtgui_graphic_driver_get_rect(rtgui_graphic_driver_get_default(), &screen);
  275. rtgui_rect_moveto_align(&screen, &rect, RTGUI_ALIGN_CENTER_HORIZONTAL | RTGUI_ALIGN_CENTER_VERTICAL);
  276. menu = rtgui_win_create(RTGUI_TOPLEVEL(rtgui_widget_get_toplevel(parent)),
  277. "Folder Menu", &rect, RTGUI_WIN_STYLE_DEFAULT);
  278. if (menu != RT_NULL)
  279. {
  280. /* set user data on menu window */
  281. menu->user_data = (rt_uint32_t)parent;
  282. rtgui_win_set_ondeactivate(menu, rtgui_filelist_view_on_menu_deactivate);
  283. listbox = rtgui_listbox_create(items, sizeof(items)/sizeof(items[0]), &rect);
  284. rtgui_listbox_set_onitem(listbox, rtgui_filelist_view_on_folder_item);
  285. rtgui_container_add_child(RTGUI_CONTAINER(menu), RTGUI_WIDGET(listbox));
  286. rtgui_widget_focus(RTGUI_WIDGET(listbox));
  287. rtgui_win_show(menu, RT_FALSE);
  288. }
  289. }
  290. static void _rtgui_filelist_view_constructor(struct rtgui_filelist_view *view)
  291. {
  292. /* default rect */
  293. struct rtgui_rect rect = {0, 0, 200, 200};
  294. /* set default widget rect and set event handler */
  295. rtgui_widget_set_event_handler(RTGUI_WIDGET(view), rtgui_filelist_view_event_handler);
  296. rtgui_widget_set_rect(RTGUI_WIDGET(view), &rect);
  297. RTGUI_WIDGET(view)->flag |= RTGUI_WIDGET_FLAG_FOCUSABLE;
  298. view->current_item = 0;
  299. view->items_count = 0;
  300. view->page_items = 0;
  301. view->current_directory = RT_NULL;
  302. view->pattern = RT_NULL;
  303. RTGUI_WIDGET_BACKGROUND(RTGUI_WIDGET(view)) = white;
  304. RTGUI_WIDGET_TEXTALIGN(RTGUI_WIDGET(view)) = RTGUI_ALIGN_CENTER_VERTICAL;
  305. file_image = rtgui_image_create_from_mem("xpm",
  306. (rt_uint8_t*)file_xpm, sizeof(file_xpm), RT_TRUE);
  307. folder_image = rtgui_image_create_from_mem("xpm",
  308. (rt_uint8_t*)folder_xpm, sizeof(folder_xpm), RT_TRUE);
  309. }
  310. rtgui_type_t *rtgui_filelist_view_type_get(void)
  311. {
  312. static rtgui_type_t *filelist_view_type = RT_NULL;
  313. if (!filelist_view_type)
  314. {
  315. filelist_view_type = rtgui_type_create("flview", RTGUI_VIEW_TYPE,
  316. sizeof(rtgui_filelist_view_t), RTGUI_CONSTRUCTOR(_rtgui_filelist_view_constructor), RT_NULL);
  317. }
  318. return filelist_view_type;
  319. }
  320. void rtgui_filelist_view_ondraw(struct rtgui_filelist_view* view)
  321. {
  322. struct rtgui_dc* dc;
  323. rt_uint16_t page_index, index;
  324. struct rtgui_file_item* item;
  325. struct rtgui_rect rect, item_rect, image_rect;
  326. dc = rtgui_dc_begin_drawing(RTGUI_WIDGET(view));
  327. if (dc == RT_NULL) return;
  328. rtgui_widget_get_rect(RTGUI_WIDGET(view), &rect);
  329. rtgui_dc_fill_rect(dc, &rect);
  330. /* get item base rect */
  331. item_rect = rect;
  332. item_rect.y1 += 1;
  333. item_rect.y2 = item_rect.y1 + (1 + rtgui_theme_get_selected_height());
  334. /* get image base rect */
  335. image_rect.x1 = RTGUI_FILELIST_MARGIN; image_rect.y1 = 0;
  336. image_rect.x2 = RTGUI_FILELIST_MARGIN + file_image->w; image_rect.y2 = file_image->h;
  337. rtgui_rect_moveto_align(&item_rect, &image_rect, RTGUI_ALIGN_CENTER_VERTICAL);
  338. /* get current page */
  339. page_index = (view->current_item / view->page_items) * view->page_items;
  340. for (index = 0; index < view->page_items; index ++)
  341. {
  342. if (page_index + index >= view->items_count) break;
  343. item = &(view->items[page_index + index]);
  344. if (page_index + index == view->current_item)
  345. {
  346. rtgui_theme_draw_selected(dc, &item_rect);
  347. }
  348. else
  349. {
  350. /* draw background */
  351. rtgui_dc_fill_rect(dc, &item_rect);
  352. }
  353. /* draw item */
  354. if (item->type == RTGUI_FITEM_FILE)
  355. rtgui_image_blit(file_image, dc, &image_rect);
  356. else
  357. rtgui_image_blit(folder_image, dc, &image_rect);
  358. /* draw text */
  359. item_rect.x1 += RTGUI_FILELIST_MARGIN + file_image->w + 2;
  360. rtgui_dc_draw_text(dc, item->name, &item_rect);
  361. item_rect.x1 -= RTGUI_FILELIST_MARGIN + file_image->w + 2;
  362. /* move to next item position */
  363. item_rect.y1 += (rtgui_theme_get_selected_height() + 1);
  364. item_rect.y2 += (rtgui_theme_get_selected_height() + 1);
  365. image_rect.y1 += (rtgui_theme_get_selected_height() + 1);
  366. image_rect.y2 += (rtgui_theme_get_selected_height() + 1);
  367. }
  368. rtgui_dc_end_drawing(dc);
  369. }
  370. void rtgui_filelist_view_update_current(struct rtgui_filelist_view* view, rt_uint16_t old_item)
  371. {
  372. struct rtgui_dc* dc;
  373. struct rtgui_file_item* item;
  374. rtgui_rect_t rect, item_rect, image_rect;
  375. if (old_item/view->page_items != view->current_item/view->page_items)
  376. {
  377. /* it's not a same page, update all */
  378. rtgui_widget_update(RTGUI_WIDGET(view));
  379. return;
  380. }
  381. dc = rtgui_dc_begin_drawing(RTGUI_WIDGET(view));
  382. if (dc == RT_NULL) return;
  383. rtgui_widget_get_rect(RTGUI_WIDGET(view), &rect);
  384. /* get old item rect */
  385. item_rect = rect;
  386. item_rect.y1 += 1;
  387. item_rect.y1 += (old_item % view->page_items) * (1 + rtgui_theme_get_selected_height());
  388. item_rect.y2 = item_rect.y1 + (1 + rtgui_theme_get_selected_height());
  389. /* get image rect */
  390. image_rect.x1 = RTGUI_FILELIST_MARGIN; image_rect.y1 = 0;
  391. image_rect.x2 = RTGUI_FILELIST_MARGIN + file_image->w; image_rect.y2 = file_image->h;
  392. rtgui_rect_moveto_align(&item_rect, &image_rect, RTGUI_ALIGN_CENTER_VERTICAL);
  393. /* draw old item */
  394. rtgui_dc_fill_rect(dc, &item_rect);
  395. item = &(view->items[old_item]);
  396. if (item->type == RTGUI_FITEM_FILE) /* draw item image */
  397. rtgui_image_blit(file_image, dc, &image_rect);
  398. else
  399. rtgui_image_blit(folder_image, dc, &image_rect);
  400. item_rect.x1 += RTGUI_FILELIST_MARGIN + file_image->w + 2;
  401. rtgui_dc_draw_text(dc, item->name, &item_rect);
  402. /* draw current item */
  403. item_rect = rect;
  404. item_rect.y1 += 1;
  405. item_rect.y1 += (view->current_item % view->page_items) * (1 + rtgui_theme_get_selected_height());
  406. item_rect.y2 = item_rect.y1 + (1 + rtgui_theme_get_selected_height());
  407. rtgui_theme_draw_selected(dc, &item_rect);
  408. /* get image base rect */
  409. image_rect.x1 = RTGUI_FILELIST_MARGIN; image_rect.y1 = 0;
  410. image_rect.x2 = RTGUI_FILELIST_MARGIN + file_image->w; image_rect.y2 = file_image->h;
  411. rtgui_rect_moveto_align(&item_rect, &image_rect, RTGUI_ALIGN_CENTER_VERTICAL);
  412. item = &(view->items[view->current_item]);
  413. if (item->type == RTGUI_FITEM_FILE) /* draw item image */
  414. rtgui_image_blit(file_image, dc, &image_rect);
  415. else
  416. rtgui_image_blit(folder_image, dc, &image_rect);
  417. item_rect.x1 += RTGUI_FILELIST_MARGIN + file_image->w + 2;
  418. rtgui_dc_draw_text(dc, item->name, &item_rect);
  419. rtgui_dc_end_drawing(dc);
  420. }
  421. static void rtgui_filelist_view_onenturn(struct rtgui_filelist_view* view)
  422. {
  423. if (view->items[view->current_item].type == RTGUI_FITEM_DIR)
  424. {
  425. char new_path[64];
  426. if (strcmp(view->items[view->current_item].name, ".") == 0) return ;
  427. if (strcmp(view->items[view->current_item].name, "..") == 0)
  428. {
  429. char *ptr;
  430. ptr = strrchr(view->current_directory, PATH_SEPARATOR);
  431. if (ptr == RT_NULL) return ;
  432. if (ptr == &(view->current_directory[0]))
  433. {
  434. /* it's root directory */
  435. new_path[0] = PATH_SEPARATOR;
  436. new_path[1] = '\0';
  437. }
  438. else
  439. {
  440. strncpy(new_path, view->current_directory, ptr - view->current_directory + 1);
  441. new_path[ptr - view->current_directory] = '\0';
  442. }
  443. }
  444. else if (view->current_item == 0 &&
  445. #ifdef _WIN32
  446. (view->current_directory[1] == ':') && (view->current_directory[2] == '\\'))
  447. #else
  448. (view->current_directory[0] == '/') && (view->current_directory[1] == '\0'))
  449. #endif
  450. {
  451. if (RTGUI_VIEW(view)->modal_show == RT_TRUE)
  452. {
  453. rtgui_view_end_modal(RTGUI_VIEW(view), RTGUI_MODAL_CANCEL);
  454. }
  455. else
  456. {
  457. rtgui_filelist_view_destroy(view);
  458. }
  459. return ;
  460. }
  461. else
  462. {
  463. rtgui_filelist_view_menu_pop(RTGUI_WIDGET(view));
  464. return ;
  465. }
  466. rtgui_filelist_view_set_directory(view, new_path);
  467. }
  468. else
  469. {
  470. if (RTGUI_VIEW(view)->modal_show == RT_TRUE)
  471. {
  472. rtgui_view_end_modal(RTGUI_VIEW(view), RTGUI_MODAL_OK);
  473. }
  474. }
  475. }
  476. rt_bool_t rtgui_filelist_view_event_handler(struct rtgui_widget* widget, struct rtgui_event* event)
  477. {
  478. struct rtgui_filelist_view* view = RT_NULL;
  479. view = RTGUI_FILELIST_VIEW(widget);
  480. switch (event->type)
  481. {
  482. case RTGUI_EVENT_PAINT:
  483. rtgui_filelist_view_ondraw(view);
  484. return RT_FALSE;
  485. case RTGUI_EVENT_RESIZE:
  486. {
  487. struct rtgui_event_resize* resize;
  488. resize = (struct rtgui_event_resize*)event;
  489. /* recalculate page items */
  490. if (file_image != RT_NULL)
  491. view->page_items = resize->h / (1 + rtgui_theme_get_selected_height());
  492. else
  493. view->page_items = resize->h / (2 + 14);
  494. }
  495. break;
  496. case RTGUI_EVENT_MOUSE_BUTTON:
  497. {
  498. rtgui_rect_t rect;
  499. struct rtgui_event_mouse* emouse;
  500. emouse = (struct rtgui_event_mouse*)event;
  501. /* calculate selected item */
  502. /* get physical extent information */
  503. rtgui_widget_get_rect(widget, &rect);
  504. rtgui_widget_rect_to_device(widget, &rect);
  505. if (rtgui_rect_contains_point(&rect, emouse->x, emouse->y) == RT_EOK)
  506. {
  507. rt_uint16_t index;
  508. rt_uint16_t current_page;
  509. rt_uint16_t old_item;
  510. index = (emouse->y - rect.y1) / (2 + rtgui_theme_get_selected_height());
  511. /* get current page */
  512. current_page = view->current_item/view->page_items;
  513. old_item = view->current_item;
  514. if (index + current_page * view->page_items < view->items_count)
  515. {
  516. /* set selected item */
  517. view->current_item = index + current_page * view->page_items;
  518. if (emouse->button & RTGUI_MOUSE_BUTTON_DOWN)
  519. {
  520. rtgui_filelist_view_update_current(view, old_item);
  521. }
  522. else
  523. {
  524. /* up event */
  525. rtgui_filelist_view_onenturn(view);
  526. }
  527. }
  528. return RT_TRUE;
  529. }
  530. }
  531. break;
  532. case RTGUI_EVENT_KBD:
  533. {
  534. struct rtgui_event_kbd* ekbd = (struct rtgui_event_kbd*)event;
  535. if (ekbd->type == RTGUI_KEYDOWN)
  536. {
  537. rt_uint16_t old_item;
  538. old_item = view->current_item;
  539. switch (ekbd->key)
  540. {
  541. case RTGUIK_UP:
  542. if (view->current_item > 0)
  543. view->current_item --;
  544. rtgui_filelist_view_update_current(view, old_item);
  545. return RT_TRUE;
  546. case RTGUIK_DOWN:
  547. if (view->current_item < view->items_count - 1)
  548. view->current_item ++;
  549. rtgui_filelist_view_update_current(view, old_item);
  550. return RT_TRUE;
  551. case RTGUIK_LEFT:
  552. if (view->current_item - view->page_items >= 0)
  553. view->current_item -= view->page_items;
  554. rtgui_filelist_view_update_current(view, old_item);
  555. return RT_TRUE;
  556. case RTGUIK_RIGHT:
  557. if (view->current_item + view->page_items < view->items_count - 1)
  558. view->current_item += view->page_items;
  559. rtgui_filelist_view_update_current(view, old_item);
  560. return RT_TRUE;
  561. case RTGUIK_RETURN:
  562. rtgui_filelist_view_onenturn(view);
  563. return RT_TRUE;
  564. default:
  565. break;
  566. }
  567. }
  568. }
  569. return RT_FALSE;
  570. }
  571. /* use view event handler */
  572. return rtgui_view_event_handler(widget, event);
  573. }
  574. rtgui_filelist_view_t* rtgui_filelist_view_create(rtgui_workbench_t* workbench,
  575. const char* directory, const char* pattern, const rtgui_rect_t* rect)
  576. {
  577. struct rtgui_filelist_view* view = RT_NULL;
  578. /* create a new view */
  579. view = (struct rtgui_filelist_view*) rtgui_widget_create(RTGUI_FILELIST_VIEW_TYPE);
  580. if (view != RT_NULL)
  581. {
  582. view->items = RT_NULL;
  583. view->pattern = rt_strdup(pattern);
  584. view->page_items = rtgui_rect_height(*rect) / (1 + rtgui_theme_get_selected_height());
  585. rtgui_filelist_view_set_directory(view, directory);
  586. rtgui_workbench_add_view(workbench, RTGUI_VIEW(view));
  587. }
  588. return view;
  589. }
  590. void rtgui_filelist_view_destroy(rtgui_filelist_view_t* view)
  591. {
  592. /* delete all file items */
  593. rtgui_filelist_view_clear(view);
  594. /* delete current directory and pattern */
  595. rtgui_free(view->current_directory); view->current_directory = RT_NULL;
  596. rtgui_free(view->pattern); view->pattern = RT_NULL;
  597. /* delete image */
  598. rtgui_image_destroy(file_image);
  599. rtgui_image_destroy(folder_image);
  600. /* destroy view */
  601. rtgui_widget_destroy(RTGUI_WIDGET(view));
  602. }
  603. /* clear all file items */
  604. static void rtgui_filelist_view_clear(rtgui_filelist_view_t* view)
  605. {
  606. rt_uint32_t index;
  607. struct rtgui_file_item* item;
  608. for (index = 0; index < view->items_count; index ++)
  609. {
  610. item = &(view->items[index]);
  611. /* release item name */
  612. rt_free(item->name);
  613. item->name = RT_NULL;
  614. }
  615. /* release items */
  616. rtgui_free(view->items);
  617. view->items = RT_NULL;
  618. view->items_count = 0;
  619. view->current_item = 0;
  620. }
  621. void rtgui_filelist_view_set_directory(rtgui_filelist_view_t* view, const char* directory)
  622. {
  623. char fullpath[256];
  624. struct rtgui_file_item *item;
  625. RT_ASSERT(view != RT_NULL);
  626. /* clear file items firstly */
  627. rtgui_filelist_view_clear(view);
  628. if (directory != RT_NULL)
  629. {
  630. DIR* dir;
  631. struct stat s;
  632. rt_uint32_t index;
  633. struct dirent* dirent;
  634. view->items_count = 0;
  635. dir = opendir(directory);
  636. if (dir == RT_NULL) goto __return;
  637. /* current directory exists, set it */
  638. if (view->current_directory != RT_NULL) rt_free(view->current_directory);
  639. view->current_directory = rt_strdup(directory);
  640. do
  641. {
  642. dirent = readdir(dir);
  643. if (dirent == RT_NULL) break;
  644. if (strcmp(dirent->d_name, ".") == 0) continue;
  645. if (strcmp(dirent->d_name, "..") == 0) continue;
  646. view->items_count ++;
  647. } while (dirent != RT_NULL);
  648. closedir(dir);
  649. view->items_count ++; /* root directory for [x] exit, others for .. */
  650. view->items = (struct rtgui_file_item*) rtgui_malloc(sizeof(struct rtgui_file_item) * view->items_count);
  651. if (view->items == RT_NULL) return; /* no memory */
  652. index = 0;
  653. if (directory[0] == '/' && directory[1] != '\0')
  654. {
  655. item = &(view->items[0]);
  656. /* add .. directory */
  657. item->name = rt_strdup("..");
  658. item->type = RTGUI_FITEM_DIR;
  659. item->size = 0;
  660. index ++;
  661. }
  662. else
  663. {
  664. item = &(view->items[0]);
  665. /* add .. directory */
  666. item->name = rt_strdup("退出文件浏览");
  667. item->type = RTGUI_FITEM_DIR;
  668. item->size = 0;
  669. index ++;
  670. }
  671. /* reopen directory */
  672. dir = opendir(directory);
  673. for (; index < view->items_count; index ++)
  674. {
  675. dirent = readdir(dir);
  676. if (dirent == RT_NULL) break;
  677. item = &(view->items[index]);
  678. item->name = rt_strdup(dirent->d_name);
  679. rt_memset(&s, 0, sizeof(struct stat));
  680. /* build full path for the file */
  681. if (directory[strlen(directory) - 1] != PATH_SEPARATOR)
  682. rt_sprintf(fullpath, "%s%c%s", directory, PATH_SEPARATOR, dirent->d_name);
  683. else
  684. rt_sprintf(fullpath, "%s%s", directory, dirent->d_name);
  685. stat(fullpath, &s);
  686. if ( s.st_mode & S_IFDIR )
  687. {
  688. item->type = RTGUI_FITEM_DIR;
  689. item->size = 0;
  690. }
  691. else
  692. {
  693. item->type = RTGUI_FITEM_FILE;
  694. item->size = s.st_size;
  695. }
  696. }
  697. closedir(dir);
  698. }
  699. view->current_item = 0;
  700. __return:
  701. /* update view */
  702. rtgui_widget_update(RTGUI_WIDGET(view));
  703. }
  704. void rtgui_filelist_view_get_fullpath(rtgui_filelist_view_t* view, char* path, rt_size_t len)
  705. {
  706. RT_ASSERT(view != RT_NULL);
  707. if (view->current_directory[strlen(view->current_directory) - 1] != PATH_SEPARATOR)
  708. rt_snprintf(path, len, "%s%c%s",view->current_directory, PATH_SEPARATOR,
  709. view->items[view->current_item].name);
  710. else
  711. rt_snprintf(path, len, "%s%s",view->current_directory,
  712. view->items[view->current_item].name);
  713. }
  714. #endif