]> git.tdb.fi Git - libs/gltk.git/blobdiff - source/dropdown.cpp
Improve Dropdown list management
[libs/gltk.git] / source / dropdown.cpp
index fcf35babdf132234476f9989eeac7c37a83a583e..e48303317b809601514ce71db7dded49dffbc068 100644 (file)
@@ -28,22 +28,23 @@ void Dropdown::init()
        dropped = false;
 
        add(list);
+       list.set_visible(false);
+       list.set_view_all();
        list.signal_item_selected.connect(sigc::mem_fun(this, &Dropdown::list_item_selected));
        list.signal_autosize_changed.connect(sigc::mem_fun(this, &Dropdown::list_autosize_changed));
 }
 
-void Dropdown::autosize()
+void Dropdown::autosize_special(const Part &part, Geometry &ageom) const
 {
-       if(!style)
-               return;
-
-       Widget::autosize();
-       list.autosize_all();
-       geom.w = max(geom.w, list.get_geometry().w);
-
-       if(const Part *text_part = style->get_part("text"))
+       if(part.get_name()=="list")
+       {
+               Geometry lgeom;
+               list.autosize(lgeom);
+               ageom.w = max(ageom.w, list.get_geometry().w);
+       }
+       else if(part.get_name()=="text")
        {
-               const Sides &margin = text_part->get_margin();
+               const Sides &margin = part.get_margin();
                const GL::Font &font = style->get_font();
                float font_size = style->get_font_size();
 
@@ -54,25 +55,26 @@ void Dropdown::autosize()
                        unsigned w = static_cast<unsigned>(font.get_string_width(data.get_string(i))*font_size);
                        max_w = max(max_w, w);
                }
-               geom.w = max(geom.w, max_w+margin.left+margin.right);
+               ageom.w = max(ageom.w, max_w+margin.left+margin.right);
 
                unsigned line_height = static_cast<unsigned>((font.get_ascent()-font.get_descent())*font_size);
-               geom.h = max(geom.h, line_height+margin.top+margin.bottom);
+               ageom.h = max(ageom.h, line_height+margin.top+margin.bottom);
        }
+}
 
-       rebuild();
+void Dropdown::set_selected_index(int index)
+{
+       list.set_selected_index(index);
+       if(index<0)
+               text.set(string());
 }
 
-void Dropdown::rebuild_special(const Part &part, CachedPart &cache)
+void Dropdown::rebuild_special(const Part &part)
 {
        if(part.get_name()=="text")
-       {
-               int sel = list.get_selected_index();
-               if(sel>=0)
-                       Text(*style, list.get_data().get_string(sel)).build(part, geom, cache);
-               else
-                       cache.texture = 0;
-       }
+               text.build(part, state, geom, part_cache);
+       else
+               Widget::rebuild_special(part);
 }
 
 void Dropdown::render_special(const Part &part, GL::Renderer &renderer) const
@@ -89,6 +91,7 @@ void Dropdown::button_press(int x, int y, unsigned btn)
                if(!click_focus)
                {
                        dropped = false;
+                       list.set_visible(false);
                        clear_state(ACTIVE);
                        signal_ungrab_pointer.emit();
                }
@@ -96,6 +99,8 @@ void Dropdown::button_press(int x, int y, unsigned btn)
        else if(btn==1)
        {
                dropped = true;
+               list.set_visible(true);
+               resize_list();
                set_state(ACTIVE);
                signal_grab_pointer.emit();
        }
@@ -103,24 +108,28 @@ void Dropdown::button_press(int x, int y, unsigned btn)
 
 void Dropdown::on_geometry_change()
 {
-       resize_list();
+       if(dropped)
+               resize_list();
 }
 
 void Dropdown::on_style_change()
 {
-       resize_list();
+       text.set_style(style);
+       if(dropped)
+               resize_list();
 }
 
 void Dropdown::list_autosize_changed()
 {
-       resize_list();
+       if(dropped)
+               resize_list();
        signal_autosize_changed.emit();
 }
 
 void Dropdown::resize_list()
 {
-       list.autosize_all();
-       Geometry lgeom = list.get_geometry();
+       Geometry lgeom;
+       list.autosize(lgeom);
        lgeom.x = 0;
        lgeom.y = -lgeom.h;
        lgeom.w = max(geom.w, lgeom.w);
@@ -134,7 +143,10 @@ void Dropdown::resize_list()
                {
                        const Geometry &rgeom = root->get_geometry();
                        if(lgeom.h*2>rgeom.h)
+                       {
                                lgeom.h = rgeom.h/2;
+                               lgeom.y = -lgeom.h;
+                       }
                        if(root_y+lgeom.y<0)
                                lgeom.y = -root_y;
                        if(root_y+lgeom.y+lgeom.h>rgeom.h)
@@ -156,20 +168,22 @@ void Dropdown::list_item_selected(unsigned index)
                signal_ungrab_pointer.emit();
        }
 
+       text.set(list.get_data().get_string(index));
+
        signal_item_selected.emit(index);
        rebuild();
 }
 
 
 Dropdown::Loader::Loader(Dropdown &d):
-       Widget::Loader(d)
+       DataFile::DerivedObjectLoader<Dropdown, Widget::Loader>(d)
 {
        add("item", &Loader::item);
 }
 
 void Dropdown::Loader::item(const string &v)
 {
-       dynamic_cast<BasicListData<string> &>(dynamic_cast<Dropdown &>(obj).list.get_data()).append(v);
+       dynamic_cast<BasicListData<string> &>(obj.list.get_data()).append(v);
 }
 
 } // namespace GLtk