enable click-pasting feature
[framework/uifw/cbhm.git] / src / clipdrawer.c
1 #include "common.h"
2 #include "cbhm_main.h"
3 #include "storage.h"
4 #include "xcnphandler.h"
5 #include "clipdrawer.h"
6
7 #define DELETE_ICON_PATH "/usr/share/icon/cbhm/05_delete.png"
8 #define IM      "/usr/share/icon/cbhm/"
9 static const char *g_images_path[] = {
10         IM"cbhm_default_img.png",
11 /*
12         IM"2_photo.jpg",
13         IM"3_photo.jpg",
14         IM"4_photo.jpg",
15         IM"5_photo.jpg",
16         IM"6_photo.jpg",
17 */
18 };
19 #define N_IMAGES (1)
20
21 #define GRID_ITEM_W 187
22 #define GRID_ITEM_H 151
23 #define GRID_IMAGE_LIMIT_W 91
24 #define GRID_IMAGE_LIMIT_H 113
25
26 // 0 - select mode, 1 - delete mode
27 static int g_clipdrawer_mode = 0;
28 // gic should live at gengrid callback functions
29 Elm_Gengrid_Item_Class gic;
30
31 typedef struct tag_griditem
32 {
33         Elm_Gengrid_Item *item;
34         const char *idata;
35         int itype;
36         Evas_Object *delbtn;
37 } griditem_t;
38
39 static int toggle_clipdrawer_mode()
40 {
41         g_clipdrawer_mode = !g_clipdrawer_mode;
42         return g_clipdrawer_mode;
43 }
44
45 static int get_clipdrawer_mode()
46 {
47         return g_clipdrawer_mode;
48 }
49
50 static void _list_click_paste(void *data, Evas_Object *obj, void *event_info)
51 {
52         struct appdata *ad = data;
53     Elm_List_Item *it = (Elm_List_Item *) elm_list_selected_item_get(obj);
54         if (it == NULL)
55                 return;
56
57     elm_list_item_selected_set(it, 0);
58
59         Elm_List_Item *item;
60         Eina_List *n;
61         int hc = 0;
62         EINA_LIST_FOREACH(elm_list_items_get(ad->txtlist), n, item)
63         {
64                 if (item == it)
65                         break;
66                 hc++;
67         }
68
69         fprintf(stderr, "## this c = %d, %d\n", hc, get_current_history_position());
70
71 //      int pos = get_current_history_position() - hc;
72
73         int     pos = get_current_history_position()-hc;
74         if (pos < 0)
75                 pos = pos+(HISTORY_QUEUE_MAX_TXT_ITEMS);
76
77         fprintf(stderr, "## pos = %d, %s\n", pos, get_item_contents_by_pos(pos));
78         char *p = strdup(get_item_contents_by_pos(pos));
79
80         elm_selection_set(1, obj, /*ELM_SEL_FORMAT_TEXT*/1, p);
81
82 /*
83         char *p = NULL;
84         int cplen;
85
86         char *cpdata = NULL;
87         cpdata = elm_entry_utf8_to_markup(elm_list_item_label_get(it));
88         if (cpdata == NULL)
89                 return;
90         cplen = strlen(cpdata);
91         p = malloc(cplen + 1);
92         snprintf(p, cplen+1, "%s", cpdata);
93 */
94 //      elm_selection_set(1, obj, /*ELM_SEL_FORMAT_TEXT*/1, p);
95 //      elm_selection_set(1, obj, /*ELM_SEL_FORMAT_MARKUP*/2, p);
96
97 //      clipdrawer_lower_view(ad);
98 }
99
100 int clipdrawer_update_contents(void *data)
101 {
102         struct appdata *ad = data;
103         int i, pos;
104         char *unesc = NULL;
105
106         // if delete mode, then back to normal mode
107 //      if (get_clipdrawer_mode())
108 //              clipdrawer_change_mode(ad);
109
110         for (i = 0; i < HISTORY_QUEUE_MAX_TXT_ITEMS; i++)
111         {
112                 pos = get_current_history_position() - i;
113                 if (pos < 0)
114                         pos = pos+HISTORY_QUEUE_MAX_TXT_ITEMS;
115
116                 if (get_item_contents_by_pos(pos) != NULL && strlen(get_item_contents_by_pos(pos)) > 0)
117                 {
118                         unesc = clipdrawer_get_plain_string_from_escaped(get_item_contents_by_pos(pos));
119                         unesc = unesc ? unesc : "";
120                         elm_list_item_append(ad->txtlist, unesc, NULL, NULL, NULL, ad);
121                         free(unesc);
122                 }
123         }
124
125         /* FIXME : sometimes when list update, screen isn't updated */
126
127         return 0;
128 }
129
130 const char* clipdrawer_get_plain_string_from_escaped(char *escstr)
131 {
132         /* TODO : is it should be here? besides, remove dependency for entry */
133         /* NOTE : return string should be freed */
134         return elm_entry_markup_to_utf8(escstr);
135 }
136
137 Evas_Object* _grid_icon_get(const void *data, Evas_Object *obj, const char *part)
138 {
139         int delete_mode = get_clipdrawer_mode();
140         griditem_t *ti = (griditem_t *)data;
141
142         if (!strcmp(part, "elm.swallow.icon"))
143         {
144                 if (ti->itype == GI_TEXT)
145                 {
146                         Evas_Object *ientry = elm_entry_add(obj);
147                         elm_entry_entry_set(ientry, eina_strbuf_string_get(ti->idata));
148                         elm_entry_background_color_set(ientry, 242, 233, 183, 255);
149                         elm_entry_editable_set(ientry, EINA_FALSE);
150 //                      evas_object_size_hint_aspect_set(ientry, EVAS_ASPECT_CONTROL_VERTICAL, 1, 1);
151                         evas_object_show(ientry);
152                         return ientry;
153                 }
154                 else// if (ti->itype == GI_IMAGE)
155                 {
156                         Evas_Object *icon = elm_icon_add(obj);
157                         elm_icon_file_set(icon, ti->idata, NULL);
158                         evas_object_size_hint_aspect_set(icon, EVAS_ASPECT_CONTROL_VERTICAL, 1, 1);
159                         evas_object_show(icon);
160
161 /*
162                         Ecore_Evas *my_ee;
163                         Evas *my_e;
164                         Evas_Object *fgimg;
165                         Evas_Object *bgrect;
166                         Evas_Object *delbtn;
167                         Evas_Object *icon;
168                         my_ee = ecore_evas_buffer_new(GRID_ITEM_W, GRID_ITEM_H);
169                         my_e = ecore_evas_get(my_ee);
170
171                         bgrect = evas_object_rectangle_add(my_e);
172                         evas_object_color_set(bgrect, 255, 255, 255, 255);
173                         evas_object_resize(bgrect, GRID_ITEM_W, GRID_ITEM_H);
174                         evas_object_move(bgrect, 0, 0);
175                         evas_object_show(bgrect);
176
177 #define BORDER_SIZE 1
178                         fgimg = evas_object_image_add(my_e);
179                         evas_object_image_load_size_set(fgimg, GRID_ITEM_W-BORDER_SIZE*2, GRID_ITEM_H-BORDER_SIZE*2);
180                         evas_object_image_file_set(fgimg, ti->idata, NULL);
181                         evas_object_image_fill_set(fgimg, 0, 0, GRID_ITEM_W-BORDER_SIZE*2, GRID_ITEM_H-BORDER_SIZE*2);
182                         evas_object_image_filled_set(fgimg, 1);
183                         int x,y;
184                         evas_object_image_size_get(fgimg, &x, &y);
185                         fprintf(stderr, "## img x = %d, y = %d\n", x, y);
186                         evas_object_resize(fgimg, GRID_ITEM_W-BORDER_SIZE*2, GRID_ITEM_H-BORDER_SIZE*2);
187                         evas_object_move(fgimg, BORDER_SIZE, BORDER_SIZE);
188                         evas_object_show(fgimg);
189
190                         icon = evas_object_image_add(evas_object_evas_get(obj));
191
192                         evas_object_image_data_set(icon, NULL);
193                         evas_object_image_size_set(icon, GRID_ITEM_W, GRID_ITEM_H);
194                         evas_object_image_fill_set(icon, 0, 0, GRID_ITEM_W, GRID_ITEM_H);
195                         evas_object_image_filled_set(icon, EINA_TRUE);
196                         evas_object_image_data_copy_set(icon, (int *)ecore_evas_buffer_pixels_get(my_ee));
197                         evas_object_image_data_update_add(icon, 0, 0, GRID_ITEM_W, GRID_ITEM_H);
198
199                         evas_object_del(bgrect);
200                         evas_object_del(fgimg);
201                         ecore_evas_free(my_ee);
202 */
203
204                         return icon;
205                 }
206
207 //              return icon;
208         }
209 /*
210         else if (!strcmp(part, "elm.swallow.end") && delete_mode)
211         {
212                 ti->delbtn = elm_check_add(obj);
213                 elm_object_style_set(ti->delbtn, "extended/imagegrid");
214                 elm_check_state_set(ti->delbtn, EINA_TRUE);
215                 evas_object_show(ti->delbtn);
216                 return ti->delbtn;
217         }
218 */
219            
220         return NULL;
221 }
222
223 static void _grid_longpress(void *data, Evas_Object *obj, void *event_info)
224 {
225         struct appdata *ad = data;
226         clipdrawer_change_mode(ad);
227 }
228
229
230 static void _grid_click_paste(void *data, Evas_Object *obj, void *event_info)
231 {
232         struct appdata *ad = data;
233         char *file, *p;
234         int len;
235         Elm_Gengrid_Item *sgobj = NULL;
236         sgobj = elm_gengrid_selected_item_get(ad->hig);
237         griditem_t *ti = NULL;
238         ti = elm_gengrid_item_data_get(sgobj);
239
240         if (!sgobj || !ti)
241         {
242                 DTRACE("ERR: cbhm can't get the selected image\n");
243                 return;
244         }
245         if (ti->itype == GI_TEXT)
246         {
247                 char *p = strdup(eina_strbuf_string_get(ti->idata));
248
249                 elm_selection_set(1, obj, /*ELM_SEL_FORMAT_TEXT*/1, p);
250         }
251         else //if (ti->itype == GI_IMAGE)
252         {
253                 len = strlen(ti->idata);
254                 p = malloc(len + 10);
255                 snprintf(p,len+10, "file:///%s", ti->idata);
256
257                 elm_selection_set(/*secondary*/1,obj,/*ELM_SEL_FORMAT_IMAGE*/4,p);
258
259                 elm_gengrid_item_selected_set(sgobj, EINA_FALSE);
260         }
261 }
262
263 static void _grid_del_response_cb(void *data, Evas_Object *obj, void *event_info)
264 {
265         Elm_Gengrid_Item *it = (Elm_Gengrid_Item *)data;
266         evas_object_del(obj);
267
268         if((int)event_info == ELM_POPUP_RESPONSE_OK)
269                 elm_gengrid_item_del(it);
270 }
271
272 static void _grid_click_delete(void *data, Evas_Object *obj, void *event_info)
273 {
274         struct appdata *ad = data;
275         Elm_Gengrid_Item *sgobj = NULL;
276         sgobj = elm_gengrid_selected_item_get(ad->hig);
277         griditem_t *ti = NULL;
278         ti = elm_gengrid_item_data_get(sgobj);
279
280         if (!sgobj || !ti)
281         {
282                 DTRACE("ERR: cbhm can't get the clicked del image\n");
283                 return;
284         }
285
286         Evas_Object *popup = elm_popup_add(ad->win_main);
287         elm_popup_timeout_set(popup, 5);
288         evas_object_size_hint_weight_set(popup, EVAS_HINT_EXPAND, EVAS_HINT_EXPAND);
289         elm_popup_desc_set(popup, "Are you sure delete this?");
290         elm_popup_buttons_add(popup, 2,
291                                                   "Yes", ELM_POPUP_RESPONSE_OK,
292                                                   "No", ELM_POPUP_RESPONSE_CANCEL,
293                                                   NULL);
294         evas_object_smart_callback_add(popup, "response", _grid_del_response_cb, sgobj);
295         evas_object_show(popup);
296 }
297
298 void _grid_del(const void *data, Evas_Object *obj)
299 {
300         griditem_t *ti = (griditem_t *)data;
301         if (ti->itype == GI_IMAGE)
302                 eina_stringshare_del(ti->idata);
303         else
304                 eina_strbuf_free(ti->idata);
305         free(ti);
306 }
307
308 int clipdrawer_refresh_history_item(void *data, int delete_mode)
309 {
310         struct appdata *ad = data;
311         Eina_List *oldlist = NULL;
312         const Eina_List *l;
313         Elm_Gengrid_Item *lgrid;
314         griditem_t *lgitem;
315         Evas_Object *ngg;
316         Evas_Object *oldgg;
317         
318         oldlist = elm_gengrid_items_get(ad->hig);
319         elm_layout_content_unset(ad->ly_main, "imagehistory/list");
320         ngg = elm_gengrid_add(ad->win_main);
321         elm_layout_content_set(ad->ly_main, "imagehistory/list", ngg);
322         oldgg = ad->hig;
323         ad->hig = ngg;
324         elm_gengrid_item_size_set(ad->hig, GRID_ITEM_W, GRID_ITEM_H+3);
325         elm_gengrid_align_set(ad->hig, 0.5, 0.5);
326 //      elm_gengrid_horizontal_set(ad->hig, EINA_TRUE);
327         elm_gengrid_bounce_set(ad->hig, EINA_TRUE, EINA_FALSE);
328         elm_gengrid_multi_select_set(ad->hig, EINA_FALSE);
329         if (delete_mode)
330                 evas_object_smart_callback_add(ad->hig, "selected", _grid_click_delete, ad);
331         else
332                 evas_object_smart_callback_add(ad->hig, "selected", _grid_click_paste, ad);
333         evas_object_smart_callback_add(ad->hig, "longpressed", _grid_longpress, ad);
334         evas_object_size_hint_weight_set(ad->hig, EVAS_HINT_EXPAND, EVAS_HINT_EXPAND);
335
336         gic.item_style = "default_grid";
337         gic.func.label_get = NULL;
338         gic.func.icon_get = _grid_icon_get;
339         gic.func.state_get = NULL;
340         gic.func.del = _grid_del;
341
342         EINA_LIST_REVERSE_FOREACH(oldlist, l, lgrid)
343         {
344                 lgitem = elm_gengrid_item_data_get(lgrid);
345                 clipdrawer_add_item(lgitem->idata, GI_IMAGE);
346         }
347
348         evas_object_show (ad->hig);
349
350         elm_gengrid_clear(oldgg);
351         evas_object_hide(oldgg);
352         evas_object_del(oldgg);
353
354         return 0;
355 }
356
357 int clipdrawer_change_mode(void *data)
358 {
359         struct appdata *ad = data;
360
361         toggle_clipdrawer_mode();
362
363         DTRACE("clipdrawer delete mode = %d\n", get_clipdrawer_mode());
364
365         clipdrawer_refresh_history_item(ad, get_clipdrawer_mode());
366
367         return 0;
368 }
369
370 // FIXME: how to remove calling g_get_main_appdata()? 
371 //        it's mainly used at 'clipdrawer_image_item'
372 int clipdrawer_add_item(char *idata, int type)
373 {
374         struct appdata *ad = g_get_main_appdata();
375         griditem_t *newgi = NULL;
376         char* filepath = NULL;
377         Eina_List *igl = NULL;
378         unsigned int igl_counter = 0;
379
380         // if delete mode, then back to normal mode
381 //      if (get_clipdrawer_mode())
382 //              clipdrawer_change_mode(ad);
383
384 /*
385         if (!check_regular_file(imagepath))
386         {
387                 DTRACE("Error : it isn't normal file = %s\n", imagepath);
388                 return -1;
389         }
390
391         igl = elm_gengrid_items_get(ad->hig);
392         igl_counter = eina_list_count(igl);
393
394         Eina_List *l;
395         Elm_Gengrid_Item *item;
396         griditem_t *ti = NULL;
397
398         EINA_LIST_FOREACH(igl, l, item)
399         {
400                 ti = elm_gengrid_item_data_get(item);
401                 if (!strcmp(ti->path, imagepath))
402                 {
403                         DTRACE("Error : duplicated file path = %s\n", imagepath);
404                         return -1;
405                 }
406         }
407
408         if (igl_counter >= HISTORY_QUEUE_MAX_IMG_ITEMS)
409         {
410                 elm_gengrid_item_del(eina_list_data_get(eina_list_last(igl)));
411         }
412
413         newgi = malloc(sizeof(griditem_t));
414         newgi->itype = GI_IMAGE;
415         newgi->idata = eina_stringshare_add(imagepath);
416         newgi->item = elm_gengrid_item_prepend(ad->hig, &gic, newgi, NULL, NULL);
417 */
418
419 /*
420         static int testmode = 0;
421         testmode++;
422
423         newgi = malloc(sizeof(griditem_t));
424         if (testmode % 3)
425         {
426         newgi->itype = GI_IMAGE;
427         newgi->idata = eina_stringshare_add(imagepath);
428         }
429         else
430         {
431         newgi->itype = GI_TEXT;
432         newgi->idata = eina_strbuf_new();
433         eina_strbuf_append(newgi->idata, "hello!! <item absize=40x30 href=file:///usr/share/icon/cbhm/cbhm_default_img.png></item>");
434
435         }
436         newgi->item = elm_gengrid_item_append(ad->hig, &gic, newgi, NULL, NULL);
437 */
438
439         newgi = malloc(sizeof(griditem_t));
440         newgi->itype = type;
441
442         fprintf(stderr, "## add %d : %s\n", newgi->itype, idata);
443         if (type == GI_TEXT)
444         {
445                 newgi->idata = eina_strbuf_new();
446                 eina_strbuf_append(newgi->idata, idata);
447
448         }
449         else //if (type == GI_IMAGE)
450         {
451                 newgi->idata = eina_stringshare_add(idata);
452         }
453         newgi->item = elm_gengrid_item_prepend(ad->hig, &gic, newgi, NULL, NULL);
454
455         return TRUE;
456 }
457
458 static void
459 clipdrawer_ly_clicked(void *data, Evas_Object *obj, const char *emission, const char *source)
460 {
461         struct appdata *ad = data;
462
463         #define EDJE_CLOSE_PART_PREFIX "background/close"
464         if (!strncmp(source, EDJE_CLOSE_PART_PREFIX, strlen(EDJE_CLOSE_PART_PREFIX)))
465         {
466                 clipdrawer_lower_view(ad);
467         }
468 }
469
470 int clipdrawer_init(void *data)
471 {
472         struct appdata *ad = data;
473         double cdy, cdw;
474
475         // for elm_check
476         elm_theme_extension_add(NULL, APP_EDJ_FILE);
477
478         cdy = (1.0*CLIPDRAWER_HEIGHT/SCREEN_HEIGHT)*ad->root_h;
479         cdw = (1.0*CLIPDRAWER_POS_Y/SCREEN_HEIGHT)*ad->root_h;
480
481         evas_object_resize(ad->win_main, ad->root_w, (int)cdy);
482         evas_object_move(ad->win_main, CLIPDRAWER_POS_X, (int)cdw);
483         evas_object_resize(ad->ly_main, ad->root_w, (int)cdy);
484         evas_object_move(ad->ly_main, CLIPDRAWER_POS_X, (int)cdw);
485
486         edje_object_signal_callback_add(elm_layout_edje_get(ad->ly_main), "mouse,up,1", "*", clipdrawer_ly_clicked, ad);
487
488         ad->hig = NULL;
489         ad->hig = elm_gengrid_add(ad->win_main);
490         elm_layout_content_set(ad->ly_main, "historyitems", ad->hig);
491         elm_gengrid_item_size_set(ad->hig, GRID_ITEM_W, GRID_ITEM_H);
492         elm_gengrid_align_set(ad->hig, 0.5, 0.5);
493         elm_gengrid_horizontal_set(ad->hig, EINA_TRUE);
494         elm_gengrid_bounce_set(ad->hig, EINA_TRUE, EINA_FALSE);
495         elm_gengrid_multi_select_set(ad->hig, EINA_FALSE);
496         evas_object_smart_callback_add(ad->hig, "selected", _grid_click_paste, ad);
497 //      evas_object_smart_callback_add(ad->hig, "longpressed", _grid_longpress, ad);
498         evas_object_size_hint_weight_set(ad->hig, EVAS_HINT_EXPAND, EVAS_HINT_EXPAND);
499
500         elm_gengrid_clear(ad->hig);
501
502         gic.item_style = "default_grid";
503         gic.func.label_get = NULL;
504         gic.func.icon_get = _grid_icon_get;
505         gic.func.state_get = NULL;
506         gic.func.del = _grid_del;
507
508         int i;
509         griditem_t *newgi;
510
511         for (i = 0; i < N_IMAGES; i++)
512 //      for (i = 0; i < 9; i++)
513         {
514                 clipdrawer_add_item(g_images_path[0], GI_IMAGE);
515         }
516
517         evas_object_show (ad->hig);
518
519 // for debugging, calc image history pos
520 /*
521    Evas_Coord x, y, w, h;
522    Evas_Coord vx, vy, vw, vh;
523
524    edje_object_part_geometry_get(elm_layout_edje_get(ad->ly_main),"imagehistory/list",&x,&y,&w,&h);
525    evas_object_geometry_get (ad->hig, &vx,&vy,&vw,&vh);
526    fprintf(stderr, "## x = %d, y = %d, w = %d, h = %d\n", x, y, w, h);
527    fprintf(stderr, "## vx = %d, vy = %d, vw = %d, vh = %d\n", vx, vy, vw, vh);
528 */
529
530 //      if (get_item_counts() != 0)
531 //              clipdrawer_update_contents(ad);
532
533         return 0;
534 }
535
536 int clipdrawer_create_view(void *data)
537 {
538         struct appdata *ad = data;
539
540         clipdrawer_init(ad);
541
542         // for debug
543         // at starting, showing app view
544         clipdrawer_activate_view(ad);
545
546         return 0;
547 }
548
549 void clipdrawer_activate_view(void *data)
550 {
551         struct appdata *ad = data;
552         
553         if (ad->win_main)
554         {
555                 evas_object_show(ad->win_main);
556                 elm_win_activate(ad->win_main);
557         }
558 }
559
560 void clipdrawer_lower_view(void *data)
561 {
562         struct appdata *ad = data;
563         
564         if (ad->win_main)
565         {
566                 evas_object_hide(ad->win_main);
567                 elm_win_lower(ad->win_main);
568
569                 // if delete mode, then back to normal mode
570                 if (get_clipdrawer_mode())
571                         clipdrawer_change_mode(ad);
572         }
573 }