aboutsummaryrefslogblamecommitdiffstats
path: root/src/bin/miniview.c
blob: a526c17d48a76606c72f2247da53faf1728e0129 (plain) (tree)
1
2
3
4
5
6
7
8
9
10

                       
                   
                 
 
                    
                     


                    
                  
















                                                                       

                                                






                                                                  
                                                                           





                                     


                                                 




                                  
                     
                     
                    
                       
 




                                                                 
 

                                  

                              
                                






                         


                                 
 
           
                                                                                              

                            
                                   
                          
 

                                                                        









                           
                                                                        

                     

                                                      
      

                                                   





                                            


                                               

                                                                
                                                         
      

                                             

       
                
 

           
                                                   
                                                                   




                                
                                                      


      


                                    
                   
               

                                                  
 
                                   
                                                  
                                            


























                                                                            
           







                                                                          



                                                    


                                                   




                                           
                              






                                                                            
                     

 












                                                                               
                                                                   



















                                                                               

                     
                                                                        





                                                                         


      
         
                                                                    







                                                   
                                                   



                                                                        
                                                                  








                                                          









                                                                
                          



                                     









                                                               
                          




                         
           



                                                   


                                                  
                 






                                                            

                            
                                      





                                                                                         
                             


           



                                                    









                                                                    



                                                   























                                                                            
                             


           


                            
 


                                       
                  


           
                            

                                                  
 
                   
                                          
                             

                            


           









                                                               




                                                                        


                                                  


                                                                                     






                                               


                                      


                                                  
                   
                      
                     


           



                                                  
                   


                         
                         
                            
 
                          
                                   
      






                                                  
                   


                         
                                   
      

 
    
                                 
 
                
 

                                        
                                        
                     

 



















                                                                



                              
                             

                        
                           
                   
               
                            
                       
 


                                                               

                                     
                          
      
 
                                   
 
                                   
                                                  
                                                            
                                                                   
 
                                            
 


                                                            
 
                                                           

                                                
 




                                                          
 
                                  
      
                                                                 
                          
                                                                   
      
                                               

                                                            
 


                                                                               


                                                                       
      






















                                                                        
      



                                                                                         
 
                     
                                
                             
                     

 
 

                                                           
 
                                     


                                                  



                                                               
                                              



                         
                                                  
 
                      
                     

 


                 
 
                                                            
 





                                            


                                       

 
           



                                                 




                                          
                                                                     

 
             
                                                      

           
                        
                
                                               





                                                
 



                                          
                       






                                                                            
                                            
                                             
                                                         
                       

               




                                                               

                                                                             



                                                                            


              
#include <Elementary.h>
#include <stdio.h>
#include <assert.h>
#include <math.h>

#include "private.h"
#include "miniview.h"
#include "col.h"
#include "termpty.h"
#include "termio.h"
#include "utils.h"
#include "main.h"

/* specific log domain to help debug only miniview */
int _miniview_log_dom = -1;

#undef CRITICAL
#undef ERR
#undef WRN
#undef INF
#undef DBG

#define CRIT(...)     EINA_LOG_DOM_CRIT(_miniview_log_dom, __VA_ARGS__)
#define ERR(...)      EINA_LOG_DOM_ERR(_miniview_log_dom, __VA_ARGS__)
#define WRN(...)      EINA_LOG_DOM_WARN(_miniview_log_dom, __VA_ARGS__)
#define INF(...)      EINA_LOG_DOM_INFO(_miniview_log_dom, __VA_ARGS__)
#define DBG(...)      EINA_LOG_DOM_DBG(_miniview_log_dom, __VA_ARGS__)

static Eina_Bool _deferred_renderer(void *data);

void
miniview_init(void)
{
   if (_miniview_log_dom >= 0) return;

   _miniview_log_dom = eina_log_domain_register("miniview", NULL);
   if (_miniview_log_dom < 0)
     EINA_LOG_CRIT(_("Could not create logging domain '%s'."), "miniview");
}

void
miniview_shutdown(void)
{
   if (_miniview_log_dom < 0) return;
   eina_log_domain_unregister(_miniview_log_dom);
   _miniview_log_dom = -1;
}

typedef struct _Miniview Miniview;

struct _Miniview
{
   Evas_Object *self;
   Evas_Object *base;
   Evas_Object *img;
   Evas_Object *termio;

   int img_hist; /* history rendered is between img_hist (<0) and
                    img_hist + img_h */
   unsigned img_h;
   unsigned rows;
   unsigned cols;

   Ecore_Timer *deferred_renderer;

   unsigned int is_shown : 1;
   unsigned int to_render : 1;
   unsigned int initial_pos : 1;

   Eina_Bool fits_to_img;

   struct _screen {
      double size;
      double pos_val;
   }screen;
};

static Evas_Smart *_smart = NULL;

static void
_draw_cell(const Termpty *ty, unsigned int *pixel, const Termcell *cell, unsigned int *colors)
{
   int fg, bg, fgext, bgext;
   int inv = ty->termstate.reverse;
   Eina_Unicode codepoint;

   codepoint = cell->codepoint;
   if ((codepoint == 0) || (cell->att.newline) || (cell->att.invisible))
     {
        *pixel = 0;
        return;
     }
   // colors
   fg = cell->att.fg;
   bg = cell->att.bg;
   fgext = cell->att.fg256;
   bgext = cell->att.bg256;

   if ((fg == COL_DEF) && (cell->att.inverse ^ inv)) fg = COL_INVERSEBG;
   if (bg == COL_DEF)
     {
        if (cell->att.inverse ^ inv) bg = COL_INVERSE;
        else if (!bgext) bg = COL_INVIS;
     }
   if ((cell->att.fgintense) && (!fgext)) fg += 48;
   if ((cell->att.bgintense) && (!bgext)) bg += 48;
   if (cell->att.inverse ^ inv)
     {
        int t;
        t = fgext; fgext = bgext; bgext = t;
        t = fg; fg = bg; bg = t;
     }
   if ((cell->att.bold) && (!fgext)) fg += 12;
   if ((cell->att.faint) && (!fgext)) fg += 24;
   
   if (bgext) *pixel = colors[bg + 256];
   else if (bg && ((bg % 12) != COL_INVIS)) *pixel = colors[bg];
   else if ((codepoint > 32) && (codepoint < 0x00110000))
     {
        if (fgext) *pixel = colors[fg + 256];
        else *pixel = colors[fg];
     }
   else
     *pixel = 0;
}

static void
_draw_line(const Termpty *ty, unsigned int *pixels,
           const Termcell *cells, int length, unsigned int *colors)
{
   int x;

   for (x = 0 ; x < length; x++)
     {
        _draw_cell(ty, pixels + x, cells + x, colors);
     }
}

Eina_Bool
_is_top_bottom_reached(Miniview *mv)
{
   int history_len;
   Termpty *ty;

   EINA_SAFETY_ON_NULL_RETURN_VAL(mv, EINA_FALSE);

   ty = termio_pty_get(mv->termio);
   EINA_SAFETY_ON_NULL_RETURN_VAL(ty, EINA_FALSE);
   history_len = termpty_backlog_length(ty);

   if (( (- mv->img_hist) > (int)(mv->img_h - mv->rows - (mv->rows / 2))) &&
       ( (- mv->img_hist) < (int)(history_len + (mv->rows / 2))))
          return EINA_FALSE;

   return EINA_TRUE;
}

static void
_screen_visual_bounds(Miniview *mv)
{
   if ((mv->screen.pos_val > 1) || (mv->screen.pos_val < 0))
     {
        edje_object_part_drag_value_set(mv->base, "miniview_screen",
                                        0.0, mv->screen.pos_val);
        edje_object_signal_emit(mv->base, "miniview_screen,outbounds",
                                "miniview");
     }
   else
     {
        edje_object_part_drag_value_set(mv->base, "miniview_screen",
                                        0.0, mv->screen.pos_val);
        edje_object_signal_emit(mv->base, "miniview_screen,inbounds",
                                "miniview");
     }
}

static void
_queue_render(Miniview *mv)
{
   mv->to_render = 1;
   if ((!mv->deferred_renderer) && (mv->is_shown))
     mv->deferred_renderer = ecore_timer_add(0.1, _deferred_renderer, mv);
}

static void
_smart_cb_mouse_wheel(void *data,
                      Evas *_e EINA_UNUSED,
                      Evas_Object *_obj EINA_UNUSED,
                      void *event)
{
   Evas_Event_Mouse_Wheel *ev = event;
   Miniview *mv = evas_object_smart_data_get(data);

   EINA_SAFETY_ON_NULL_RETURN(mv);

   /* do not handle horizontal scrolling */
   if (ev->direction) return;
   mv->img_hist += ev->z * 25;

   if (!mv->fits_to_img && !_is_top_bottom_reached(mv))
     {
        mv->screen.pos_val = mv->screen.pos_val -
                             (double) (ev->z * 25) / (mv->img_h - mv->rows);
        _screen_visual_bounds(mv);
     }
   _queue_render(mv);
}

void
miniview_position_offset(Evas_Object *obj, int by, Eina_Bool sanitize)
{
   Miniview *mv = evas_object_smart_data_get(obj);
   int remain = 0;

   termio_scroll_get(mv->termio);
   EINA_SAFETY_ON_NULL_RETURN(mv);
   if ((mv->screen.pos_val <= 1.0) && (mv->screen.pos_val >= 0.0))
     edje_object_signal_emit(mv->base, "miniview_screen,inbounds", "miniview");

   if (!mv->fits_to_img)
     {
        mv->screen.pos_val += (double) by / (mv->img_h - mv->rows);
        edje_object_part_drag_value_set(mv->base, "miniview_screen",
                                        0.0, mv->screen.pos_val);
        if ((mv->screen.pos_val <= 0) && (sanitize))
          {
             /* This is what remains when screen pos has to
                go negative by some portion "by" */
             remain = (int) round(mv->screen.pos_val * (mv->img_h - mv->rows));
             mv->img_hist += remain;
             mv->screen.pos_val = 0;
          }
        if ((mv->screen.pos_val > 1) && (sanitize))
          {
             remain = (int) round((1 - mv->screen.pos_val) *
                      (mv->img_h - mv->rows));
             mv->img_hist -= remain;
             mv->screen.pos_val = 1;
          }
     }
   else
     {
        if (sanitize)
          {
             mv->screen.pos_val += (double) by / (mv->img_h - mv->rows);
             edje_object_part_drag_value_set(mv->base, "miniview_screen",
                                             0.0, mv->screen.pos_val);

             if (mv->screen.pos_val < 0) mv->screen.pos_val = 0;
             if (mv->screen.pos_val > 1) mv->screen.pos_val = 1;
          }
     }
}

Eina_Bool
miniview_handle_key(Evas_Object *obj, const Evas_Event_Key_Down *ev)
{
   Miniview *mv;
   Evas_Coord ox, oy, ow, oh, mx, my;
   int z = 25;

   EINA_SAFETY_ON_NULL_RETURN_VAL(obj, EINA_FALSE);

   mv = evas_object_smart_data_get(obj);
   if ((!mv) || (!mv->is_shown)) return EINA_FALSE;

   evas_object_geometry_get(mv->img, &ox, &oy, &ow, &oh);
   evas_pointer_canvas_xy_get(evas_object_evas_get(mv->base), &mx, &my);

   if ((mx < ox) || (mx > ox + ow) || (my < oy) || (my > oy + oh))
     return EINA_FALSE;

   if (evas_key_modifier_is_set(ev->modifiers, "Alt") ||
       evas_key_modifier_is_set(ev->modifiers, "Shift") ||
       evas_key_modifier_is_set(ev->modifiers, "Control"))
     z = mv->img_h;

   if (!strcmp(ev->key, "Prior"))
     {
        if (!mv->fits_to_img)
          {
             mv->img_hist -= z;
             if (_is_top_bottom_reached(mv))
               {
                  miniview_position_offset(obj, -z, EINA_FALSE);
               }
             miniview_position_offset(obj, z, EINA_FALSE);
             _screen_visual_bounds(mv);
          }
        _queue_render(mv);
        return EINA_TRUE;
     }
   else if (!strcmp(ev->key, "Next"))
     {
        if (!mv->fits_to_img)
          {
             mv->img_hist += z;
             if (_is_top_bottom_reached(mv))
               {
                  miniview_position_offset(obj, z, EINA_FALSE);
               }
             miniview_position_offset(obj, -z, EINA_FALSE);
             _screen_visual_bounds(mv);
          }
        _queue_render(mv);
        return EINA_TRUE;
     }
   return EINA_FALSE;
}

static void
_smart_cb_mouse_down(void *data,
                     Evas *_e EINA_UNUSED,
                     Evas_Object *_obj EINA_UNUSED,
                     void *event)
{
   Evas_Event_Mouse_Down *ev = event;
   Miniview *mv= evas_object_smart_data_get(data);
   int pos, pos2;
   Evas_Coord oy;

   EINA_SAFETY_ON_NULL_RETURN(mv);

   evas_object_geometry_get(mv->img, NULL, &oy, NULL, NULL);
   pos = oy - ev->canvas.y;
   pos -= mv->img_hist;
   if (pos < 0) pos = 0;
   else pos += mv->rows / 2;
   termio_scroll_set(mv->termio, pos);

   pos2 = ev->canvas.y - oy - (mv->rows / 2);
   if (pos2 < 0) pos2 = 0;
   if (pos2 > -mv->img_hist) pos2 = -mv->img_hist;
   mv->screen.pos_val = (double) pos2 / (mv->img_h - mv->rows);
   edje_object_part_drag_value_set(mv->base, "miniview_screen", 0.0, mv->screen.pos_val);
   _screen_visual_bounds(mv);
}

static void
_on_screen_stoped(void *data,
                  Evas_Object *_o EINA_UNUSED,
                  const char *_emission EINA_UNUSED,
                  const char *_source EINA_UNUSED)
{
   Miniview *mv= evas_object_smart_data_get(data);

   EINA_SAFETY_ON_NULL_RETURN(mv);

   edje_object_part_drag_value_set(mv->base, "miniview_screen", 0.0,
                                   mv->screen.pos_val);
}

static void
_on_screen_moved(void *data,
                 Evas_Object *o,
                 const char *_emission EINA_UNUSED,
                 const char *_source EINA_UNUSED)
{
   Miniview *mv = evas_object_smart_data_get(data);
   double val = 0.0, pos = 0.0, bottom_bound = 0.0;

   edje_object_part_drag_value_get(o, "miniview_screen", NULL, &val);
   bottom_bound = ((double) (-mv->img_hist )) / (mv->img_h - mv->rows);
   if (!mv->fits_to_img)
     {
        pos = (bottom_bound - val) * (mv->img_h - mv->rows);
        mv->screen.pos_val = val;
     }
   else
     {
        bottom_bound = ((double) (-mv->img_hist )) / (mv->img_h - mv->rows);
        pos = (bottom_bound - val) * (mv->img_h - mv->rows);
        if (val < bottom_bound)
          mv->screen.pos_val = val;
        if (val > bottom_bound)
          {
             mv->screen.pos_val = bottom_bound;
             pos = 0;
          }
     }
   termio_scroll_set(mv->termio, (int) pos);
   _screen_visual_bounds(mv);
}

static void
_smart_add(Evas_Object *obj)
{
   Miniview *mv;

   mv = calloc(1, sizeof(Miniview));
   EINA_SAFETY_ON_NULL_RETURN(mv);
   evas_object_smart_data_set(obj, mv);
   mv->self = obj;
}

static void
_smart_del(Evas_Object *obj)
{
   Miniview *mv = evas_object_smart_data_get(obj);

   if (!mv) return;
   ecore_timer_del(mv->deferred_renderer);
   evas_object_del(mv->base);
   evas_object_del(mv->img);
   free(mv);
}

static void
_do_configure(Evas_Object *obj)
{
   Miniview *mv = evas_object_smart_data_get(obj);
   Evas_Coord ox, oy, ow, oh, font_w, font_h, w;

   if (!mv) return;

   evas_object_geometry_get(mv->termio, &ox, &oy, &ow, &oh);
   evas_object_size_hint_min_get(mv->termio, &font_w, &font_h);

   if ((font_w <= 0) || (font_h <= 0) || (ow == 0) || (oh == 0)) return;

   mv->img_h = oh;
   mv->rows = oh / font_h;
   mv->cols = ow / font_w;
   
   if ((mv->rows == 0) || (mv->cols == 0)) return;
   
   mv->screen.size = (double) mv->rows / (double) mv->img_h;
   edje_object_part_drag_size_set(mv->base, "miniview_screen", 1.0, mv->screen.size);

   w = (mv->cols * font_w) / font_h;
   
   evas_object_resize(mv->base, w, mv->img_h);
   evas_object_move(mv->base, ox + ow - w, oy);
}

static void
_smart_move(Evas_Object *obj,
            Evas_Coord _x EINA_UNUSED,
            Evas_Coord _y EINA_UNUSED)
{
   Miniview *mv = evas_object_smart_data_get(obj);

   if (!mv) return;
   _do_configure(obj);
   _queue_render(mv);
}

static void
_smart_show(Evas_Object *obj)
{
   Miniview *mv = evas_object_smart_data_get(obj);

   if (!mv) return;
   if (!mv->is_shown)
     {
        mv->is_shown = 1;
        mv->img_hist = 0;
        mv->initial_pos = 1;

        _queue_render(mv);
        evas_object_show(mv->base);
     }
}

static void
_smart_hide(Evas_Object *obj)
{
   Miniview *mv = evas_object_smart_data_get(obj);

   if (!mv) return;
   if (mv->is_shown)
     {
        mv->is_shown = 0;
        evas_object_hide(mv->base);
     }
}

void
miniview_redraw(Evas_Object *obj)
{
   Miniview *mv;

   if (!obj) return;
   mv = evas_object_smart_data_get(obj);
   if ((!mv) || (!mv->is_shown)) return;
   _queue_render(mv);
}

static void
miniview_colors_get(Miniview *mv, unsigned int *colors)
{
   Evas_Object *tg = termio_textgrid_get(mv->termio);
   int r, g, b, a, c;
   
   for (c = 0; c < 256; c++)
     {
        evas_object_textgrid_palette_get
        (tg, EVAS_TEXTGRID_PALETTE_STANDARD, c, &r, &g, &b, &a);
        colors[c] = (a << 24) | (r << 16) | (g << 8) | b;
     }
   for (c = 0; c < 256; c++)
     {
        evas_object_textgrid_palette_get
        (tg, EVAS_TEXTGRID_PALETTE_EXTENDED, c, &r, &g, &b, &a);
        colors[c + 256] = (a << 24) | (r << 16) | (g << 8) | b;
     }
}

static Eina_Bool
_deferred_renderer(void *data)
{
   Miniview *mv = data;
   Evas_Coord ox, oy, ow, oh;
   int history_len, pos;
   ssize_t wret;
   unsigned int *pixels, y;
   Termcell *cells;
   Termpty *ty;
   unsigned int colors[512];
   double bottom_bound;

   if (!mv) return EINA_FALSE;

   if ((!mv->is_shown) || (!mv->to_render) || (mv->img_h == 0))
     {
        mv->deferred_renderer = NULL;
        return EINA_FALSE;
     }

   miniview_colors_get(mv, colors);

   ty = termio_pty_get(mv->termio);
   EINA_SAFETY_ON_NULL_RETURN_VAL(ty, EINA_FALSE);
   evas_object_geometry_get(mv->termio, &ox, &oy, &ow, &oh);
   if ((ow == 0) || (oh == 0) || (mv->cols == 1)) return EINA_TRUE;

   history_len = termpty_backlog_length(ty);

   evas_object_image_size_set(mv->img, mv->cols, mv->img_h);
   ow = mv->cols;
   oh = mv->img_h;

   pixels = evas_object_image_data_get(mv->img, EINA_TRUE);
   memset(pixels, 0, sizeof(*pixels) * ow * oh);
   mv->img_h = oh;

   /* "current"? */
   if (mv->img_hist >= - ((int)mv->img_h - (int)mv->rows))
     mv->img_hist = -((int)mv->img_h - (int)mv->rows);
   if (mv->img_hist < -history_len)
     mv->img_hist = -history_len;

   for (y = 0; y < mv->img_h; y++)
     {
        cells = termpty_cellrow_get(ty, mv->img_hist + y, &wret);
        if (!cells) break;
        _draw_line(ty, &pixels[y * mv->cols], cells, wret, colors);
     }
   evas_object_image_data_set(mv->img, pixels);
   evas_object_image_pixels_dirty_set(mv->img, EINA_FALSE);
   evas_object_image_data_update_add(mv->img, 0, 0, ow, oh);

   if (history_len > (int)(mv->img_h - mv->rows)) mv->fits_to_img = EINA_FALSE;
   else mv->fits_to_img = EINA_TRUE;

   bottom_bound = ((double) (-mv->img_hist )) / (mv->img_h - mv->rows);
   pos = termio_scroll_get(mv->termio);
   if ((pos != 0) && (mv->initial_pos))
     {
        double val;
        val = (double) pos / (mv->img_h - mv->rows);
        if ((mv->fits_to_img))
          {
             mv->screen.pos_val = bottom_bound - val;
             mv->initial_pos = 0;
          }
        else
          {
             if (val > 1)
               {
                  mv->img_hist += (mv->img_hist * (int)val);
                  mv->screen.pos_val = (double)(1 - ( val - (int) val));
                  mv->initial_pos = 0;
                  mv->to_render = 1;
                  return EINA_TRUE;
               }
             else
               {
                  mv->screen.pos_val = bottom_bound - val;
                  mv->initial_pos = 0;
               }
          }
     }
   if (pos == 0)
     mv->screen.pos_val = bottom_bound;

   edje_object_part_drag_value_set(mv->base, "miniview_screen", 0.0, mv->screen.pos_val);

   mv->to_render = 0;
   mv->deferred_renderer = NULL;
   _screen_visual_bounds(mv);
   return EINA_FALSE;
}


static void
_smart_resize(Evas_Object *obj, Evas_Coord w, Evas_Coord h)
{
   Evas_Coord font_w, font_h, ox, oy;
   Miniview *mv = evas_object_smart_data_get(obj);
   if (!mv) return;

   mv->img_h = h;

   evas_object_geometry_get(mv->termio, &ox, &oy, NULL, NULL);
   evas_object_size_hint_min_get(mv->termio, &font_w, &font_h);
   if ((font_w <= 0) || (font_h <= 0)) return;

   mv->rows = h / font_h;
   mv->cols = w / font_w;

   if ((mv->rows == 0) || (mv->cols == 0)) return;

   _do_configure(obj);
   _queue_render(mv);
}


static void
_smart_init(void)
{
    static Evas_Smart_Class sc = EVAS_SMART_CLASS_INIT_NULL;

    sc.name      = "miniview";
    sc.version   = EVAS_SMART_CLASS_VERSION;
    sc.add       = _smart_add;
    sc.del       = _smart_del;
    sc.resize    = _smart_resize;
    sc.move      = _smart_move;
    sc.show      = _smart_show;
    sc.hide      = _smart_hide;
    _smart = evas_smart_class_new(&sc);
}

static void
_cb_miniview_close(void *data,
                   Evas_Object *_obj EINA_UNUSED,
                   const char *_sig EINA_UNUSED,
                   const char *_src EINA_UNUSED)
{
   Miniview *mv = data;

   EINA_SAFETY_ON_NULL_RETURN(mv);
   EINA_SAFETY_ON_NULL_RETURN(mv->termio);
   if (mv->is_shown) term_miniview_hide(termio_term_get(mv->termio));
}

Evas_Object *
miniview_add(Evas_Object *parent, Evas_Object *termio)
{
   Evas *e;
   Evas_Object *obj, *o;
   Miniview *mv;
   Evas *canvas = evas_object_evas_get(parent);

   EINA_SAFETY_ON_NULL_RETURN_VAL(parent, NULL);
   e = evas_object_evas_get(parent);
   if (!e) return NULL;

   if (!_smart) _smart_init();

   obj = evas_object_smart_add(e, _smart);
   mv = evas_object_smart_data_get(obj);
   if (!mv) return obj;

   mv->termio = termio;

   mv->base = edje_object_add(canvas);
   theme_apply(mv->base, termio_config_get(termio), "terminology/miniview");

   evas_object_smart_member_add(mv->base, obj);

   /* miniview output widget  */
   o = evas_object_image_filled_add(canvas);
   evas_object_image_alpha_set(o, EINA_TRUE);
   edje_object_part_swallow(mv->base, "miniview.img", o);
   evas_object_show(o);

   mv->img = o;

   evas_object_event_callback_add(o, EVAS_CALLBACK_MOUSE_WHEEL,
                                  _smart_cb_mouse_wheel, obj);
   evas_object_event_callback_add(o, EVAS_CALLBACK_MOUSE_DOWN,
                                  _smart_cb_mouse_down, obj);
   edje_object_signal_callback_add(mv->base, "miniview,close", "terminology",
                                   _cb_miniview_close, mv);
   edje_object_signal_callback_add(mv->base, "drag", "miniview_screen",
                                   _on_screen_moved, obj);
   edje_object_signal_callback_add(mv->base, "drag,stop", "miniview_screen",
                                   _on_screen_stoped, obj);
   return obj;
}