// Copyright (C) 2005 Davis E. King (davis@dlib.net), Keita Mochizuki // License: Boost Software License See LICENSE.txt for the full license. #ifndef DLIB_WIDGETs_ #define DLIB_WIDGETs_ #include #include #include #include #include #include #include "../algs.h" #include "widgets_abstract.h" #include "drawable.h" #include "../gui_core.h" #include "fonts.h" #include "../timer.h" #include "base_widgets.h" #include "../member_function_pointer.h" #include "../array.h" #include "../array2d.h" #include "../sequence.h" #include "../dir_nav.h" #include "../queue.h" #include "style.h" #include "../string.h" #include "../misc_api.h" #include "../any.h" #include "../image_processing/full_object_detection.h" #include "../geometry/line.h" #ifdef _MSC_VER // This #pragma directive is also located in the algs.h file but for whatever // reason visual studio 9 just ignores it when it is only there. // this is to disable the "'this' : used in base member initializer list" // warning you get from some of the GUI objects since all the objects // require that their parent class be passed into their constructor. // In this case though it is totally safe so it is ok to disable this warning. #pragma warning(disable : 4355) #endif namespace dlib { // ---------------------------------------------------------------------------------------- // ---------------------------------------------------------------------------------------- // class label // ---------------------------------------------------------------------------------------- // ---------------------------------------------------------------------------------------- class label : public drawable { public: label( drawable_window& w ) : drawable(w), text_color_(0,0,0) { enable_events(); } ~label() { disable_events(); parent.invalidate_rectangle(rect); } void set_text ( const std::string& text ); void set_text ( const std::wstring& text ); void set_text ( const dlib::ustring& text ); const std::string text ( ) const; const std::wstring wtext ( ) const; const dlib::ustring utext ( ) const; void set_text_color ( const rgb_pixel color ); const rgb_pixel text_color ( ) const; void set_main_font ( const std::shared_ptr& f ); private: dlib::ustring text_; rgb_pixel text_color_; // restricted functions label(label&); // copy constructor label& operator=(label&); // assignment operator protected: void draw ( const canvas& c ) const; }; // ---------------------------------------------------------------------------------------- // ---------------------------------------------------------------------------------------- // class toggle_button // ---------------------------------------------------------------------------------------- // ---------------------------------------------------------------------------------------- class toggle_button : public button_action { /*! INITIAL VALUE - checked == false CONVENTION - is_checked() == checked !*/ public: toggle_button( drawable_window& w ) : button_action(w), btn_tooltip(w), checked(false) { style.reset(new toggle_button_style_default()); enable_events(); } ~toggle_button() { disable_events(); parent.invalidate_rectangle(rect); } void set_name ( const std::string& name ); void set_name ( const std::wstring& name ); void set_name ( const dlib::ustring& name ); void set_size ( unsigned long width_, unsigned long height_ ); void set_tooltip_text ( const std::string& text ); void set_tooltip_text ( const std::wstring& text ); void set_tooltip_text ( const ustring& text ); const std::string tooltip_text ( ) const; const std::wstring tooltip_wtext ( ) const; const dlib::ustring tooltip_utext ( ) const; bool is_checked ( ) const; const std::string name ( ) const; const std::wstring wname ( ) const; const dlib::ustring uname ( ) const; void set_checked ( ); void set_unchecked ( ); void show ( ); void hide ( ); void enable ( ); void disable ( ); void set_main_font ( const std::shared_ptr& f ); void set_pos ( long x, long y ); template < typename style_type > void set_style ( const style_type& style_ ) { auto_mutex M(m); style.reset(new style_type(style_)); rect = move_rect(style->get_min_size(name_,*mfont), rect.left(), rect.top()); parent.invalidate_rectangle(rect); } template < typename T > void set_click_handler ( T& object, void (T::*event_handler_)() ) { auto_mutex M(m); event_handler = make_mfp(object,event_handler_); event_handler_self.clear(); } void set_click_handler ( const any_function& event_handler_ ) { auto_mutex M(m); event_handler = event_handler_; event_handler_self.clear(); } template < typename T > void set_click_handler ( T& object, void (T::*event_handler_)(toggle_button&) ) { auto_mutex M(m); event_handler_self = make_mfp(object,event_handler_); event_handler.clear(); } void set_sourced_click_handler ( const any_function& event_handler_ ) { auto_mutex M(m); event_handler_self = event_handler_; event_handler.clear(); } private: // restricted functions toggle_button(toggle_button&); // copy constructor toggle_button& operator=(toggle_button&); // assignment operator dlib::ustring name_; tooltip btn_tooltip; bool checked; any_function event_handler; any_function event_handler_self; std::unique_ptr style; protected: void draw ( const canvas& c ) const { style->draw_toggle_button(c,rect,enabled,*mfont,lastx,lasty,name_,is_depressed(),checked); } void on_button_up ( bool mouse_over ); void on_mouse_over ( ){ if (style->redraw_on_mouse_over()) parent.invalidate_rectangle(rect); } void on_mouse_not_over ( ){ if (style->redraw_on_mouse_over()) parent.invalidate_rectangle(rect); } }; // ---------------------------------------------------------------------------------------- // ---------------------------------------------------------------------------------------- // class text_field // ---------------------------------------------------------------------------------------- // ---------------------------------------------------------------------------------------- class text_field : public drawable { /*! INITIAL VALUE text_color_ == rgb_pixel(0,0,0) bg_color_ == rgb_pixel(255,255,255) cursor_pos == 0 text_width == 0 text_ == "" has_focus == false cursor_visible == false recent_movement == false highlight_start == 0 highlight_end == -1 shift_pos == -1 text_pos == 0 CONVENTION - cursor_pos == the position of the cursor in the string text_. The cursor appears before the letter text_[cursor_pos] - cursor_x == the x coordinate of the cursor relative to the left side of rect. i.e. the number of pixels that separate the cursor from the left side of the text_field. - has_focus == true if this text field has keyboard input focus - cursor_visible == true if the cursor should be painted - text_ == text() - text_pos == the index of the first letter in text_ that appears in this text field. - text_width == the width of text_[text_pos] though text_[text.size()-1] - if (has_focus && the user has recently moved the cursor) then - recent_movement == true - else - recent_movement == false - if (highlight_start <= highlight_end) then - text[highlight_start] though text[highlight_end] should be highlighted - if (shift_pos != -1) then - has_focus == true - the shift key is being held down or the left mouse button is being held down. - shift_pos == the position of the cursor when the shift or mouse key was first pressed. - text_color() == text_color_ - background_color() == bg_color_ !*/ public: text_field( drawable_window& w ) : drawable(w,MOUSE_CLICK | KEYBOARD_EVENTS | MOUSE_MOVE | STRING_PUT), text_color_(0,0,0), bg_color_(255,255,255), text_width(0), text_pos(0), recent_movement(false), has_focus(false), cursor_visible(false), cursor_pos(0), highlight_start(0), highlight_end(-1), shift_pos(-1), t(*this,&text_field::timer_action), right_click_menu(w) { style.reset(new text_field_style_default()); rect.set_bottom(mfont->height()+ (style->get_padding(*mfont))*2); rect.set_right((style->get_padding(*mfont))*2); cursor_x = style->get_padding(*mfont); right_click_menu.menu().add_menu_item(menu_item_text("Cut",*this,&text_field::on_cut,'t')); right_click_menu.menu().add_menu_item(menu_item_text("Copy",*this,&text_field::on_copy,'C')); right_click_menu.menu().add_menu_item(menu_item_text("Paste",*this,&text_field::on_paste,'P')); right_click_menu.menu().add_menu_item(menu_item_text("Delete",*this,&text_field::on_delete_selected,'D')); right_click_menu.menu().add_menu_item(menu_item_separator()); right_click_menu.menu().add_menu_item(menu_item_text("Select All",*this,&text_field::on_select_all,'A')); right_click_menu.set_rect(get_text_rect()); enable_events(); t.set_delay_time(500); } ~text_field ( ) { disable_events(); parent.invalidate_rectangle(rect); t.stop_and_wait(); } template < typename style_type > void set_style ( const style_type& style_ ) { auto_mutex M(m); style.reset(new style_type(style_)); // call this just so that this widget redraws itself with the new style set_main_font(mfont); } void set_text ( const std::string& text_ ); void set_text ( const std::wstring& text_ ); void give_input_focus ( ); bool has_input_focus ( ) const; void select_all_text ( ); void set_text ( const dlib::ustring& text_ ); const std::string text ( ) const; const std::wstring wtext ( ) const; const dlib::ustring utext ( ) const; void set_text_color ( const rgb_pixel color ); const rgb_pixel text_color ( ) const; void set_background_color ( const rgb_pixel color ); const rgb_pixel background_color ( ) const; void set_width ( unsigned long width ); void set_pos ( long x, long y ); void set_main_font ( const std::shared_ptr& f ); int next_free_user_event_number ( ) const { return drawable::next_free_user_event_number()+1; } void disable ( ); void enable ( ); void hide ( ); void show ( ); template < typename T > void set_text_modified_handler ( T& object, void (T::*event_handler)() ) { auto_mutex M(m); text_modified_handler = make_mfp(object,event_handler); } template < typename T > void set_enter_key_handler ( T& object, void (T::*event_handler)() ) { auto_mutex M(m); enter_key_handler = make_mfp(object,event_handler); } void set_text_modified_handler ( const any_function& event_handler ) { auto_mutex M(m); text_modified_handler = event_handler; } void set_enter_key_handler ( const any_function& event_handler ) { auto_mutex M(m); enter_key_handler = event_handler; } template < typename T > void set_focus_lost_handler ( T& object, void (T::*event_handler)() ) { auto_mutex M(m); focus_lost_handler = make_mfp(object,event_handler); } void set_focus_lost_handler ( const any_function& event_handler ) { auto_mutex M(m); focus_lost_handler = event_handler; } private: void on_cut ( ); void on_copy ( ); void on_paste ( ); void on_select_all ( ); void on_delete_selected ( ); void on_text_is_selected ( ); void on_no_text_selected ( ); void on_user_event ( int num ) { // ignore this user event if it isn't for us if (num != drawable::next_free_user_event_number()) return; if (recent_movement == false) { cursor_visible = !cursor_visible; parent.invalidate_rectangle(rect); } else { if (cursor_visible == false) { cursor_visible = true; parent.invalidate_rectangle(rect); } recent_movement = false; } } void timer_action ( ) { parent.trigger_user_event(this,drawable::next_free_user_event_number()); } /*! ensures - flips the state of cursor_visible !*/ void move_cursor ( unsigned long pos ); /*! requires - pos <= text_.size() ensures - moves the cursor to the position given by pos and moves the text in the text box if necessary - if the position changes then the parent window will be updated !*/ rectangle get_text_rect ( ) const; /*! ensures - returns the rectangle that should contain the text in this widget !*/ dlib::ustring text_; rgb_pixel text_color_; rgb_pixel bg_color_; unsigned long text_width; unsigned long text_pos; bool recent_movement; bool has_focus; bool cursor_visible; long cursor_pos; unsigned long cursor_x; // this tells you what part of the text is highlighted long highlight_start; long highlight_end; long shift_pos; any_function text_modified_handler; any_function enter_key_handler; any_function focus_lost_handler; std::unique_ptr style; timer t; popup_menu_region right_click_menu; // restricted functions text_field(text_field&); // copy constructor text_field& operator=(text_field&); // assignment operator protected: void draw ( const canvas& c ) const; void on_mouse_down ( unsigned long btn, unsigned long state, long x, long y, bool is_double_click ); void on_mouse_up ( unsigned long btn, unsigned long state, long x, long y ); void on_mouse_move ( unsigned long state, long x, long y ); void on_keydown ( unsigned long key, bool is_printable, unsigned long state ); void on_string_put ( const std::wstring &str ); }; // ---------------------------------------------------------------------------------------- // ---------------------------------------------------------------------------------------- // class text_box // ---------------------------------------------------------------------------------------- // ---------------------------------------------------------------------------------------- class text_box : public scrollable_region { /*! INITIAL VALUE text_color_ == rgb_pixel(0,0,0) bg_color_ == rgb_pixel(255,255,255) cursor_pos == 0 text_ == "" has_focus == false cursor_visible == false recent_movement == false highlight_start == 0 highlight_end == -1 shift_pos == -1 CONVENTION - cursor_pos == the position of the cursor in the string text_. The cursor appears before the letter text_[cursor_pos] - cursor_rect == The rectangle that should be drawn for the cursor. The position is relative to total_rect(). - has_focus == true if this text field has keyboard input focus - cursor_visible == true if the cursor should be painted - text_ == text() - if (has_focus && the user has recently moved the cursor) then - recent_movement == true - else - recent_movement == false - if (highlight_start <= highlight_end) then - text[highlight_start] though text[highlight_end] should be highlighted - if (shift_pos != -1) then - has_focus == true - the shift key is being held down or the left mouse button is being held down. - shift_pos == the position of the cursor when the shift or mouse key was first pressed. - text_color() == text_color_ - background_color() == bg_color_ !*/ public: text_box( drawable_window& w ) : scrollable_region(w,MOUSE_CLICK | KEYBOARD_EVENTS | MOUSE_MOVE | STRING_PUT), text_color_(0,0,0), bg_color_(255,255,255), recent_movement(false), has_focus(false), cursor_visible(false), cursor_pos(0), highlight_start(0), highlight_end(-1), shift_pos(-1), t(*this,&text_box::timer_action), right_click_menu(w) { style.reset(new text_box_style_default()); const long padding = static_cast(style->get_padding(*mfont)); cursor_rect = mfont->compute_cursor_rect(rectangle(padding,padding,1000000,1000000), text_, 0); adjust_total_rect(); set_vertical_mouse_wheel_scroll_increment(mfont->height()); set_horizontal_mouse_wheel_scroll_increment(mfont->height()); right_click_menu.menu().add_menu_item(menu_item_text("Cut",*this,&text_box::on_cut,'t')); right_click_menu.menu().add_menu_item(menu_item_text("Copy",*this,&text_box::on_copy,'C')); right_click_menu.menu().add_menu_item(menu_item_text("Paste",*this,&text_box::on_paste,'P')); right_click_menu.menu().add_menu_item(menu_item_text("Delete",*this,&text_box::on_delete_selected,'D')); right_click_menu.menu().add_menu_item(menu_item_separator()); right_click_menu.menu().add_menu_item(menu_item_text("Select All",*this,&text_box::on_select_all,'A')); right_click_menu.set_rect(get_text_rect()); set_size(100,100); enable_events(); t.set_delay_time(500); } ~text_box ( ) { disable_events(); parent.invalidate_rectangle(rect); t.stop_and_wait(); } template < typename style_type > void set_style ( const style_type& style_ ) { auto_mutex M(m); style.reset(new style_type(style_)); scrollable_region::set_style(style_.get_scrollable_region_style()); // call this just so that this widget redraws itself with the new style set_main_font(mfont); } void set_text ( const std::string& text_ ); void set_text ( const std::wstring& text_ ); void set_text ( const dlib::ustring& text_ ); const std::string text ( ) const; const std::wstring wtext ( ) const; const dlib::ustring utext ( ) const; void set_text_color ( const rgb_pixel color ); const rgb_pixel text_color ( ) const; void set_background_color ( const rgb_pixel color ); const rgb_pixel background_color ( ) const; void set_size ( unsigned long width, unsigned long height ); void set_pos ( long x, long y ); void set_main_font ( const std::shared_ptr& f ); int next_free_user_event_number ( ) const { return scrollable_region::next_free_user_event_number()+1; } void disable ( ); void enable ( ); void hide ( ); void show ( ); template < typename T > void set_text_modified_handler ( T& object, void (T::*event_handler)() ) { auto_mutex M(m); text_modified_handler = make_mfp(object,event_handler); } void set_text_modified_handler ( const any_function& event_handler ) { auto_mutex M(m); text_modified_handler = event_handler; } template < typename T > void set_enter_key_handler ( T& object, void (T::*event_handler)() ) { auto_mutex M(m); enter_key_handler = make_mfp(object,event_handler); } void set_enter_key_handler ( const any_function& event_handler ) { auto_mutex M(m); enter_key_handler = event_handler; } template < typename T > void set_focus_lost_handler ( T& object, void (T::*event_handler)() ) { auto_mutex M(m); focus_lost_handler = make_mfp(object,event_handler); } void set_focus_lost_handler ( const any_function& event_handler ) { auto_mutex M(m); focus_lost_handler = event_handler; } private: void on_cut ( ); void on_copy ( ); void on_paste ( ); void on_select_all ( ); void on_delete_selected ( ); void on_text_is_selected ( ); void on_no_text_selected ( ); void on_user_event ( int num ) { // ignore this user event if it isn't for us if (num != scrollable_region::next_free_user_event_number()) return; if (recent_movement == false) { cursor_visible = !cursor_visible; parent.invalidate_rectangle(rect); } else { if (cursor_visible == false) { cursor_visible = true; parent.invalidate_rectangle(rect); } recent_movement = false; } } // The reason for using user actions here rather than just having the timer just call // what it needs directly is to avoid a potential deadlock during destruction of this widget. void timer_action ( ) { parent.trigger_user_event(this,scrollable_region::next_free_user_event_number()); } /*! ensures - flips the state of cursor_visible !*/ void move_cursor ( unsigned long pos ); /*! requires - pos <= text_.size() ensures - moves the cursor to the position given by pos and moves the text in the text box if necessary - if the position changes then the parent window will be updated !*/ rectangle get_text_rect ( ) const; /*! ensures - returns the rectangle that should contain the text in this widget !*/ void adjust_total_rect ( ); /*! ensures - adjusts total_rect() so that it is big enough to contain the text currently in this object. !*/ dlib::ustring text_; rgb_pixel text_color_; rgb_pixel bg_color_; bool recent_movement; bool has_focus; bool cursor_visible; long cursor_pos; rectangle cursor_rect; // this tells you what part of the text is highlighted long highlight_start; long highlight_end; long shift_pos; any_function text_modified_handler; any_function enter_key_handler; any_function focus_lost_handler; std::unique_ptr style; timer t; popup_menu_region right_click_menu; // restricted functions text_box(text_box&); // copy constructor text_box& operator=(text_box&); // assignment operator protected: void draw ( const canvas& c ) const; void on_mouse_down ( unsigned long btn, unsigned long state, long x, long y, bool is_double_click ); void on_mouse_up ( unsigned long btn, unsigned long state, long x, long y ); void on_mouse_move ( unsigned long state, long x, long y ); void on_keydown ( unsigned long key, bool is_printable, unsigned long state ); void on_string_put ( const std::wstring &str ); }; // ---------------------------------------------------------------------------------------- // ---------------------------------------------------------------------------------------- // class check_box // ---------------------------------------------------------------------------------------- // ---------------------------------------------------------------------------------------- class check_box : public toggle_button { public: check_box( drawable_window& w ) : toggle_button(w) { set_style(toggle_button_style_check_box()); } }; // ---------------------------------------------------------------------------------------- // ---------------------------------------------------------------------------------------- // class radio_button // ---------------------------------------------------------------------------------------- // ---------------------------------------------------------------------------------------- class radio_button : public toggle_button { public: radio_button ( drawable_window& w ) : toggle_button(w) { set_style(toggle_button_style_radio_button()); } }; // ---------------------------------------------------------------------------------------- // ---------------------------------------------------------------------------------------- // class tabbed_display // ---------------------------------------------------------------------------------------- // ---------------------------------------------------------------------------------------- class tabbed_display : public drawable { /*! INITIAL VALUE - tabs.size() == 0 - selected_tab_ == 0 CONVENTION - number_of_tabs() == tabs.size() - tab_name(idx) == tabs[idx] - if (tabs.size() > 0) then - selected_tab_ == the index of the tab that is currently selected - for all valid i: - tabs[i].width == mfont->compute_size(tabs[i].name) - tabs[i].rect == the rectangle that defines where this tab is - if (tabs[i].group != 0) then - tabs[i].group == a pointer to the widget_group for this tab. - left_pad == the amount of padding in a tab to the left of the name string. - right_pad == the amount of padding in a tab to the right of the name string. - top_pad == the amount of padding in a tab to the top of the name string. - bottom_pad == the amount of padding in a tab to the bottom of the name string. - if (event_handler.is_set()) then - event_handler() is what is called to process click events on this object. !*/ public: tabbed_display( drawable_window& w ); virtual ~tabbed_display( ); void set_size ( unsigned long width, unsigned long height ); void set_number_of_tabs ( unsigned long num ); unsigned long selected_tab ( ) const; unsigned long number_of_tabs ( ) const; const std::string tab_name ( unsigned long idx ) const; const std::wstring tab_wname ( unsigned long idx ) const; const dlib::ustring& tab_uname ( unsigned long idx ) const; void set_tab_name ( unsigned long idx, const std::string& new_name ); void set_tab_name ( unsigned long idx, const std::wstring& new_name ); void set_tab_name ( unsigned long idx, const dlib::ustring& new_name ); void set_pos ( long x, long y ); template < typename T > void set_click_handler ( T& object, void (T::*eh)(unsigned long new_idx,unsigned long old_idx) ) { auto_mutex M(m); event_handler = make_mfp(object,eh); } void set_click_handler ( const any_function& eh ) { auto_mutex M(m); event_handler = eh; } void set_tab_group ( unsigned long idx, widget_group& group ); void show ( ); void hide ( ); void enable ( ); void disable ( ); void set_main_font ( const std::shared_ptr& f ); void fit_to_contents ( ); protected: void on_mouse_down ( unsigned long btn, unsigned long state, long x, long y, bool is_double_click ); void draw ( const canvas& c ) const; private: void recompute_tabs ( ); /*! ensures - recomputes the rectangles for all the tabs and makes this object wider if needed !*/ void draw_tab ( const rectangle& tab, const canvas& c ) const; /*! ensures - draws the outline of a tab as given by the rectangle onto c !*/ struct tab_data { tab_data() : width(0), group(0) {} dlib::ustring name; unsigned long width; rectangle rect; widget_group* group; }; unsigned long selected_tab_; array tabs; const long left_pad; const long right_pad; const long top_pad; const long bottom_pad; any_function event_handler; // restricted functions tabbed_display(tabbed_display&); // copy constructor tabbed_display& operator=(tabbed_display&); // assignment operator }; // ---------------------------------------------------------------------------------------- // ---------------------------------------------------------------------------------------- // class named_rectangle // ---------------------------------------------------------------------------------------- // ---------------------------------------------------------------------------------------- class named_rectangle : public drawable { /*! INITIAL VALUE name == "" CONVENTION name_ == name() !*/ public: named_rectangle( drawable_window& w ); virtual ~named_rectangle( ); void set_size ( unsigned long width, unsigned long height ); void set_name ( const std::string& name ); void set_name ( const std::wstring& name ); void set_name ( const dlib::ustring& name ); const std::string name ( ) const; const std::wstring wname ( ) const; const dlib::ustring uname ( ) const; void wrap_around ( const rectangle& rect ); void set_main_font ( const std::shared_ptr& f ); protected: void draw ( const canvas& c ) const; private: void make_name_fit_in_rect ( ); dlib::ustring name_; unsigned long name_width; unsigned long name_height; // restricted functions named_rectangle(named_rectangle&); // copy constructor named_rectangle& operator=(named_rectangle&); // assignment operator }; // ---------------------------------------------------------------------------------------- // ---------------------------------------------------------------------------------------- // class mouse_tracker // ---------------------------------------------------------------------------------------- // ---------------------------------------------------------------------------------------- class mouse_tracker : public draggable { public: mouse_tracker( drawable_window& w ); ~mouse_tracker( ); void show ( ); void hide ( ); void enable ( ); void disable ( ); void set_pos ( long x, long y ); void set_main_font ( const std::shared_ptr& f ); protected: void on_mouse_move ( unsigned long state, long x, long y ); void on_drag ( ); void draw ( const canvas& c ) const; void on_mouse_down ( unsigned long btn, unsigned long state, long x, long y, bool is_double_click ); private: const long offset; named_rectangle nr; label x_label; label y_label; std::ostringstream sout; long click_x, click_y; // restricted functions mouse_tracker(mouse_tracker&); // copy constructor mouse_tracker& operator=(mouse_tracker&); // assignment operator }; // ---------------------------------------------------------------------------------------- // ---------------------------------------------------------------------------------------- // function message_box() // ---------------------------------------------------------------------------------------- // ---------------------------------------------------------------------------------------- namespace message_box_helper { class box_win : public drawable_window { void initialize ( ); public: box_win ( const std::string& title_, const std::string& message_ ); box_win ( const std::wstring& title_, const std::wstring& message_ ); box_win ( const dlib::ustring& title_, const dlib::ustring& message_ ); ~box_win ( ); void set_click_handler ( const any_function& event_handler_ ) { auto_mutex M(wm); event_handler = event_handler_; } private: static void deleter_thread ( void* param ); void on_click ( ); on_close_return_code on_window_close ( ); const std::wstring title; const std::wstring message; label msg; button btn_ok; any_function event_handler; }; class blocking_box_win : public drawable_window { void initialize ( ); public: blocking_box_win ( const std::string& title_, const std::string& message_ ); blocking_box_win ( const std::wstring& title_, const std::wstring& message_ ); blocking_box_win ( const dlib::ustring& title_, const dlib::ustring& message_ ); ~blocking_box_win ( ); private: void on_click ( ); const std::wstring title; const std::wstring message; label msg; button btn_ok; }; } template < typename T > void message_box ( const std::string& title, const std::string& message, T& object, void (T::*event_handler)() ) { using namespace message_box_helper; box_win* win = new box_win(title,message); win->set_click_handler(make_mfp(object,event_handler)); } inline void message_box ( const std::string& title, const std::string& message, const any_function& event_handler ) { using namespace message_box_helper; box_win* win = new box_win(title,message); win->set_click_handler(event_handler); } inline void message_box ( const std::string& title, const std::string& message ) { using namespace message_box_helper; new box_win(title,message); } inline void message_box_blocking ( const std::string& title, const std::string& message ) { using namespace message_box_helper; blocking_box_win w(title,message); w.wait_until_closed(); } // ---------------------------------------------------------------------------------------- // ---------------------------------------------------------------------------------------- // class list_box // ---------------------------------------------------------------------------------------- // ---------------------------------------------------------------------------------------- namespace list_box_helper{ template class list_box : public scrollable_region, public enumerable { /*! INITIAL VALUE - ms_enabled == false - items.size() == 0 - last_selected = 0 CONVENTION - size() == items.size() - (*this)[i] == items[i].name - is_selected(i) == items[i].is_selected - ms_enabled == multiple_select_enabled() - items[i].width == the width of items[i].name as given by font::compute_size() - items[i].height == the height of items[i].name as given by font::compute_size() - last_selected == the last item the user selected !*/ public: list_box( drawable_window& w ); ~list_box( ); bool is_selected ( unsigned long index ) const; void select ( unsigned long index ); void unselect ( unsigned long index ); template < typename style_type > void set_style ( const style_type& style_ ) { auto_mutex M(m); style.reset(new style_type(style_)); scrollable_region::set_style(style_.get_scrollable_region_style()); parent.invalidate_rectangle(rect); } template void get_selected ( T& list ) const { auto_mutex M(m); list.clear(); for (unsigned long i = 0; i < items.size(); ++i) { if (items[i].is_selected) { unsigned long idx = i; list.enqueue(idx); } } } template void load ( const T& list ) { auto_mutex M(m); items.clear(); unsigned long i = 0; items.set_max_size(list.size()); items.set_size(list.size()); list.reset(); unsigned long max_width = 0; unsigned long total_height = 0; while (list.move_next()) { items[i].is_selected = false; items[i].name = list.element(); mfont->compute_size(items[i].name,items[i].width, items[i].height); if (items[i].width > max_width) max_width = items[i].width; total_height += items[i].height; ++i; } set_total_rect_size(max_width, total_height); parent.invalidate_rectangle(rect); last_selected = 0; } const S& operator[] ( unsigned long index ) const; bool multiple_select_enabled ( ) const; void enable_multiple_select ( ); void disable_multiple_select ( ); template < typename T > void set_double_click_handler ( T& object, void (T::*eh)(unsigned long index) ) { auto_mutex M(m); event_handler = make_mfp(object,eh); } void set_double_click_handler ( const any_function& eh ) { auto_mutex M(m); event_handler = eh; } template < typename T > void set_click_handler ( T& object, void (T::*eh)(unsigned long index) ) { auto_mutex M(m); single_click_event_handler = make_mfp(object,eh); } void set_click_handler ( const any_function& eh ) { auto_mutex M(m); single_click_event_handler = eh; } bool at_start ( ) const; void reset ( ) const; bool current_element_valid ( ) const; const S& element ( ) const; const S& element ( ); bool move_next ( ) const; size_t size ( ) const; unsigned long get_selected ( ) const; void set_main_font ( const std::shared_ptr& f ); private: void on_mouse_down ( unsigned long btn, unsigned long state, long x, long y, bool is_double_click ); void draw ( const canvas& c ) const; template struct data { SS name; bool is_selected; unsigned long width; unsigned long height; }; bool ms_enabled; array > items; any_function event_handler; any_function single_click_event_handler; unsigned long last_selected; std::unique_ptr style; // restricted functions list_box(list_box&); // copy constructor list_box& operator=(list_box&); // assignment operator }; } typedef list_box_helper::list_box list_box; typedef list_box_helper::list_box wlist_box; typedef list_box_helper::list_box ulist_box; // ---------------------------------------------------------------------------------------- // ---------------------------------------------------------------------------------------- // function open_file_box() // ---------------------------------------------------------------------------------------- // ---------------------------------------------------------------------------------------- namespace open_file_box_helper { class box_win : public drawable_window { public: box_win ( const std::string& title, bool has_text_field = false ); ~box_win ( ); void set_click_handler ( const any_function& event_handler_ ) { auto_mutex M(wm); event_handler = event_handler_; } private: void set_sizes( ); void on_window_resized ( ); void deleter_thread ( ); void enter_folder ( const std::string& folder_name ); void on_dirs_click ( unsigned long idx ); void on_files_click ( unsigned long idx ); void on_files_double_click ( unsigned long ); void on_cancel_click ( ); void on_open_click ( ); void on_path_button_click ( toggle_button& btn ); bool set_dir ( const std::string& dir ); void on_root_click ( ); on_close_return_code on_window_close ( ); label lbl_dirs; label lbl_files; label lbl_file_name; list_box lb_dirs; list_box lb_files; button btn_ok; button btn_cancel; toggle_button btn_root; text_field tf_file_name; std::string path; std::string prefix; int cur_dir; any_function event_handler; sequence >::kernel_2a_c sob; }; } template < typename T > void open_file_box ( T& object, void (T::*event_handler)(const std::string&) ) { using namespace open_file_box_helper; box_win* win = new box_win("Open File",true); win->set_click_handler(make_mfp(object,event_handler)); } inline void open_file_box ( const any_function& event_handler ) { using namespace open_file_box_helper; box_win* win = new box_win("Open File",true); win->set_click_handler(event_handler); } template < typename T > void open_existing_file_box ( T& object, void (T::*event_handler)(const std::string&) ) { using namespace open_file_box_helper; box_win* win = new box_win("Open File"); win->set_click_handler(make_mfp(object,event_handler)); } inline void open_existing_file_box ( const any_function& event_handler ) { using namespace open_file_box_helper; box_win* win = new box_win("Open File"); win->set_click_handler(event_handler); } template < typename T > void save_file_box ( T& object, void (T::*event_handler)(const std::string&) ) { using namespace open_file_box_helper; box_win* win = new box_win("Save File",true); win->set_click_handler(make_mfp(object,event_handler)); } inline void save_file_box ( const any_function& event_handler ) { using namespace open_file_box_helper; box_win* win = new box_win("Save File",true); win->set_click_handler(event_handler); } // ---------------------------------------------------------------------------------------- // ---------------------------------------------------------------------------------------- // class menu_bar // ---------------------------------------------------------------------------------------- // ---------------------------------------------------------------------------------------- class menu_bar : public drawable { /*! INITIAL VALUE - menus.size() == 0 - open_menu == 0 CONVENTION - size() == menus.size() - all menu data is stored in menus - menus[x].name == the name of the xth menu - if (menus[x].underline_pos != std::string::npos) then - menus[x].underline_pos == the position of the character in the menu name that should be underlined - menus[x].underline_p1 != menus[x].underline_p2 and these two points define the underline bar - else - menus[x].underline_p1 == menus[x].underline_p2 - menus[x].menu == menu(x) - menus[x].rect == the rectangle in which menus[x].name is drawn - menus[x].bgrect == the rectangle for the xth menu button - if (there is an open menu on the screen) then - open_menu == the index of the open menu from menus - else - open_menu == menus.size() !*/ public: menu_bar( drawable_window& w ); ~menu_bar(); // this function does nothing void set_pos(long,long){} void set_main_font ( const std::shared_ptr& f ); void set_number_of_menus ( unsigned long num ); unsigned long number_of_menus ( ) const; void set_menu_name ( unsigned long idx, const std::string name, char underline_ch = '\0' ); void set_menu_name ( unsigned long idx, const std::wstring name, char underline_ch = '\0' ); void set_menu_name ( unsigned long idx, const dlib::ustring name, char underline_ch = '\0' ); const std::string menu_name ( unsigned long idx ) const; const std::wstring menu_wname ( unsigned long idx ) const; const dlib::ustring menu_uname ( unsigned long idx ) const; popup_menu& menu ( unsigned long idx ); const popup_menu& menu ( unsigned long idx ) const; protected: void on_window_resized ( ); void draw ( const canvas& c ) const; void on_window_moved ( ); void on_focus_lost ( ); void on_mouse_down ( unsigned long btn, unsigned long , long x, long y, bool ); void on_mouse_move ( unsigned long , long x, long y ); void on_keydown ( unsigned long key, bool is_printable, unsigned long state ); private: void show_menu ( unsigned long i ); void hide_menu ( ); void on_popup_hide ( ); void compute_menu_geometry ( ); void adjust_position ( ); struct menu_data { menu_data():underline_pos(dlib::ustring::npos){} dlib::ustring name; dlib::ustring::size_type underline_pos; popup_menu menu; rectangle rect; rectangle bgrect; point underline_p1; point underline_p2; }; array menus; unsigned long open_menu; // restricted functions menu_bar(menu_bar&); // copy constructor menu_bar& operator=(menu_bar&); // assignment operator }; // ---------------------------------------------------------------------------------------- // ---------------------------------------------------------------------------------------- // class directed_graph_drawer // ---------------------------------------------------------------------------------------- // ---------------------------------------------------------------------------------------- template class directed_graph_drawer : public zoomable_region { /*! INITIAL VALUE - edge_selected == false - mouse_drag == false - selected_node == 0 - graph_.number_of_nodes() == 0 - external_graph.number_of_nodes() == 0 - radius == 25 - last_mouse_click_in_display == false CONVENTION - radius == the radius of the nodes when they aren't zoomed - external_graph and graph_ have the same graph structure - external_graph == graph() - external_graph.node(i) == graph_node(i) - if (one of the nodes is selected) then - selected_node < graph_.number_of_nodes() - graph_.node(selected_node) == the selected node - else - selected_node == graph_.number_of_nodes() - if (the user is dragging a node with the mouse) then - mouse_drag == true - drag_offset == the vector from the mouse position to the center of the node - else - mouse_drag == false - if (the user has selected an edge) then - edge_selected == true - the parent node is graph_.node(selected_edge_parent) - the child node is graph_.node(selected_edge_parent) - else - edge_selected == false - for all valid i: - graph_.node(i).data.p == the center of the node in graph space - graph_.node(i).data.name == node_label(i) - graph_.node(i).data.color == node_color(i) - graph_.node(i).data.str_rect == a rectangle sized to contain graph_.node(i).data.name - if (the last mouse click in our parent window as in our display_rect_ ) then - last_mouse_click_in_display == true - else - last_mouse_click_in_display == false !*/ public: directed_graph_drawer ( drawable_window& w ) : zoomable_region(w,MOUSE_CLICK | MOUSE_WHEEL | KEYBOARD_EVENTS), radius(25), edge_selected(false), last_mouse_click_in_display(false) { mouse_drag = false; selected_node = 0; // Whenever you make your own drawable (or inherit from draggable or button_action) // you have to remember to call this function to enable the events. The idea // here is that you can perform whatever setup you need to do to get your // object into a valid state without needing to worry about event handlers // triggering before you are ready. enable_events(); } ~directed_graph_drawer ( ) { // Disable all further events for this drawable object. We have to do this // because we don't want draw() events coming to this object while or after // it has been destructed. disable_events(); // Tell the parent window to redraw its area that previously contained this // drawable object. parent.invalidate_rectangle(rect); } void clear_graph ( ) { auto_mutex M(m); graph_.clear(); external_graph.clear(); parent.invalidate_rectangle(display_rect()); } const typename graph_type::node_type& graph_node ( unsigned long i ) const { DLIB_ASSERT ( i < number_of_nodes() , "\tgraph_type::node_type& directed_graph_drawer::graph_node(i)" << "\n\ti: " << i << "\n\tnumber_of_nodes(): " << number_of_nodes() ); return external_graph.node(i); } typename graph_type::node_type& graph_node ( unsigned long i ) { DLIB_ASSERT ( i < number_of_nodes() , "\tgraph_type::node_type& directed_graph_drawer::graph_node(i)" << "\n\ti: " << i << "\n\tnumber_of_nodes(): " << number_of_nodes() ); return external_graph.node(i); } const graph_type& graph ( ) const { return external_graph; } void save_graph ( std::ostream& out ) { auto_mutex M(m); serialize(external_graph, out); serialize(graph_, out); parent.invalidate_rectangle(display_rect()); } void load_graph ( std::istream& in ) { auto_mutex M(m); deserialize(external_graph, in); deserialize(graph_, in); parent.invalidate_rectangle(display_rect()); } unsigned long number_of_nodes ( ) const { auto_mutex M(m); return graph_.number_of_nodes(); } void set_node_label ( unsigned long i, const std::string& label ) { set_node_label(i, convert_mbstring_to_wstring(label)); } void set_node_label ( unsigned long i, const std::wstring& label ) { set_node_label(i, convert_wstring_to_utf32(label)); } void set_node_label ( unsigned long i, const dlib::ustring& label ) { auto_mutex M(m); DLIB_ASSERT ( i < number_of_nodes() , "\tvoid directed_graph_drawer::set_node_label(i,label)" << "\n\ti: " << i << "\n\tlabel: " << narrow(label) << "\n\tnumber_of_nodes(): " << number_of_nodes() ); graph_.node(i).data.name = label.c_str(); unsigned long width, height; mfont->compute_size(label,width,height); graph_.node(i).data.str_rect = rectangle(width,height); parent.invalidate_rectangle(display_rect()); } void set_node_color ( unsigned long i, rgb_pixel color ) { auto_mutex M(m); DLIB_ASSERT ( i < number_of_nodes() , "\tvoid directed_graph_drawer::set_node_color(i,label)" << "\n\ti: " << i << "\n\tnumber_of_nodes(): " << number_of_nodes() ); graph_.node(i).data.color = color; parent.invalidate_rectangle(display_rect()); } rgb_pixel node_color ( unsigned long i ) const { auto_mutex M(m); DLIB_ASSERT ( i < number_of_nodes() , "\trgb_pixel directed_graph_drawer::node_color(i)" << "\n\ti: " << i << "\n\tnumber_of_nodes(): " << number_of_nodes() ); return graph_.node(i).data.color; } const std::string node_label ( unsigned long i ) const { auto_mutex M(m); DLIB_ASSERT ( i < number_of_nodes() , "\tconst std::ustring directed_graph_drawer::node_label(i)" << "\n\ti: " << i << "\n\tnumber_of_nodes(): " << number_of_nodes() ); return narrow(graph_.node(i).data.name); } const std::wstring node_wlabel ( unsigned long i ) const { return convert_utf32_to_wstring(node_ulabel(i)); } const dlib::ustring node_ulabel ( unsigned long i ) const { auto_mutex M(m); DLIB_ASSERT ( i < number_of_nodes() , "\tconst std::ustring directed_graph_drawer::node_label(i)" << "\n\ti: " << i << "\n\tnumber_of_nodes(): " << number_of_nodes() ); return graph_.node(i).data.name.c_str(); } template < typename T > void set_node_selected_handler ( T& object, void (T::*event_handler_)(unsigned long) ) { auto_mutex M(m); node_selected_handler = make_mfp(object,event_handler_); } void set_node_selected_handler ( const any_function& event_handler_ ) { auto_mutex M(m); node_selected_handler = event_handler_; } template < typename T > void set_node_deselected_handler ( T& object, void (T::*event_handler_)(unsigned long) ) { auto_mutex M(m); node_deselected_handler = make_mfp(object,event_handler_); } void set_node_deselected_handler ( const any_function& event_handler_ ) { auto_mutex M(m); node_deselected_handler = event_handler_; } template < typename T > void set_node_deleted_handler ( T& object, void (T::*event_handler_)() ) { auto_mutex M(m); node_deleted_handler = make_mfp(object,event_handler_); } void set_node_deleted_handler ( const any_function& event_handler_ ) { auto_mutex M(m); node_deleted_handler = event_handler_; } template < typename T > void set_graph_modified_handler ( T& object, void (T::*event_handler_)() ) { auto_mutex M(m); graph_modified_handler = make_mfp(object,event_handler_); } void set_graph_modified_handler ( const any_function& event_handler_ ) { auto_mutex M(m); graph_modified_handler = event_handler_; } protected: void on_keydown ( unsigned long key, bool , unsigned long ) { // ignore all keyboard input if the last thing the user clicked on // wasn't the display area if (last_mouse_click_in_display == false) return; // if a node is selected if (selected_node != graph_.number_of_nodes()) { // deselect the node if the user hits escape if (key == base_window::KEY_ESC) { parent.invalidate_rectangle(display_rect()); if (node_deselected_handler.is_set()) node_deselected_handler(selected_node); selected_node = graph_.number_of_nodes(); } // delete the node if the user hits delete if (key == base_window::KEY_DELETE || key == base_window::KEY_BACKSPACE) { parent.invalidate_rectangle(display_rect()); graph_.remove_node(selected_node); external_graph.remove_node(selected_node); selected_node = graph_.number_of_nodes(); mouse_drag = false; if (graph_modified_handler.is_set()) graph_modified_handler(); if (node_deleted_handler.is_set()) node_deleted_handler(); } } // if an edge is selected if (edge_selected) { // deselect the node if the user hits escape if (key == base_window::KEY_ESC) { parent.invalidate_rectangle(display_rect()); edge_selected = false; } // delete the node if the user hits delete if (key == base_window::KEY_DELETE || key == base_window::KEY_BACKSPACE) { parent.invalidate_rectangle(display_rect()); graph_.remove_edge(selected_edge_parent, selected_edge_child); external_graph.remove_edge(selected_edge_parent, selected_edge_child); edge_selected = false; if (graph_modified_handler.is_set()) graph_modified_handler(); } } } void on_mouse_move ( unsigned long state, long x, long y ) { if (mouse_drag) { const point p(nearest_point(display_rect(),point(x,y))); point center = drag_offset + p; graph_.node(selected_node).data.p = gui_to_graph_space(center); parent.invalidate_rectangle(display_rect()); } else { zoomable_region::on_mouse_move(state,x,y); } // check if the mouse isn't being dragged anymore if ((state & base_window::LEFT) == 0) { mouse_drag = false; } } void on_mouse_up ( unsigned long btn, unsigned long state, long x, long y ) { mouse_drag = false; zoomable_region::on_mouse_up(btn,state,x,y); } void on_mouse_down ( unsigned long btn, unsigned long state, long x, long y, bool is_double_click ) { bool redraw = false; if (display_rect().contains(x,y) && (btn == base_window::RIGHT || btn == base_window::LEFT) && (state & base_window::SHIFT) == 0 ) { // start out saying no edge is selected if (edge_selected) { edge_selected = false; redraw = true; } bool click_hit_node = false; dlib::vector p(gui_to_graph_space(point(x,y))); // check if this click is on an existing node for (unsigned long i = 0; i < graph_.number_of_nodes(); ++i) { dlib::vector n(graph_.node(i).data.p); if ((p-n).length() < radius) { click_hit_node = true; point center = graph_to_gui_space(graph_.node(i).data.p); mouse_drag = true; drag_offset = center - point(x,y); // only do something if the click isn't on the currently // selected node if (selected_node != i) { // send out the deselected event if appropriate if (selected_node != graph_.number_of_nodes() && node_deselected_handler.is_set()) node_deselected_handler(selected_node); selected_node = i; redraw = true; if (node_selected_handler.is_set()) node_selected_handler(selected_node); } break; } } // if the click didn't hit any node then make sure nothing is selected if (click_hit_node == false && selected_node != graph_.number_of_nodes()) { if (node_deselected_handler.is_set()) node_deselected_handler(selected_node); selected_node = graph_.number_of_nodes(); redraw = true; } // check if this click is on an edge if we didn't click on a node if (click_hit_node == false) { for (unsigned long n = 0; n < graph_.number_of_nodes() && edge_selected == false; ++n) { const dlib::vector parent_center(graph_to_gui_space(graph_.node(n).data.p)); for (unsigned long e = 0; e < graph_.node(n).number_of_children() && edge_selected == false; ++e) { const dlib::vector child_center(graph_to_gui_space(graph_.node(n).child(e).data.p)); rectangle area; area += parent_center; area += child_center; // if the point(x,y) is between the two nodes then lets consider it further if (area.contains(point(x,y))) { p = point(x,y); const dlib::vector z(0,0,1); // find the distance from the line between the two nodes const dlib::vector perpendicular(z.cross(parent_center-child_center).normalize()); double distance = std::abs((child_center-p).dot(perpendicular)); if (distance < 8) { edge_selected = true; selected_edge_parent = n; selected_edge_child = graph_.node(n).child(e).index(); redraw = true; } } } } } // if the click didn't land on any node then add a new one if this was // a right mouse button click if (click_hit_node == false && btn == base_window::RIGHT) { const unsigned long n = graph_.add_node(); external_graph.add_node(); graph_.node(n).data.p = gui_to_graph_space(point(x,y)); redraw = true; selected_node = n; mouse_drag = false; if (graph_modified_handler.is_set()) graph_modified_handler(); if (node_selected_handler.is_set()) node_selected_handler(selected_node); } else if (selected_node == graph_.number_of_nodes()) { // in this case the click landed in the white area between nodes zoomable_region::on_mouse_down( btn, state, x, y, is_double_click); } } // If the user is shift clicking with the mouse then see if we // should add a new edge. if (display_rect().contains(x,y) && btn == base_window::LEFT && (state & base_window::SHIFT) && selected_node != graph_.number_of_nodes() ) { dlib::vector p(gui_to_graph_space(point(x,y))); // check if this click is on an existing node for (unsigned long i = 0; i < graph_.number_of_nodes(); ++i) { dlib::vector n(graph_.node(i).data.p); if ((p-n).length() < radius) { // add the edge if it doesn't already exist and isn't an edge back to // the same node if (graph_.has_edge(selected_node,i) == false && selected_node != i && graph_.has_edge(i, selected_node) == false) { graph_.add_edge(selected_node,i); external_graph.add_edge(selected_node,i); redraw = true; if (graph_modified_handler.is_set()) graph_modified_handler(); } break; } } } if (redraw) parent.invalidate_rectangle(display_rect()); if (display_rect().contains(x,y) == false) last_mouse_click_in_display = false; else last_mouse_click_in_display = true; } void draw ( const canvas& c ) const { zoomable_region::draw(c); rectangle area = c.intersect(display_rect()); if (area.is_empty() == true) return; if (enabled) fill_rect(c,display_rect(),255); else fill_rect(c,display_rect(),128); const unsigned long rad = static_cast(radius*zoom_scale()); point center; // first draw all the edges for (unsigned long i = 0; i < graph_.number_of_nodes(); ++i) { center = graph_to_gui_space(graph_.node(i).data.p); const rectangle circle_area(centered_rect(center,2*(rad+8),2*(rad+8))); // draw lines to all this node's parents const dlib::vector z(0,0,1); for (unsigned long j = 0; j < graph_.node(i).number_of_parents(); ++j) { point p(graph_to_gui_space(graph_.node(i).parent(j).data.p)); rgb_pixel color(0,0,0); // if this is the selected edge then draw it with red instead of black if (edge_selected && selected_edge_child == i && selected_edge_parent == graph_.node(i).parent(j).index()) { color.red = 255; // we need to be careful when drawing this line to not draw it over the node dots since it // has a different color from them and would look weird dlib::vector v(p-center); v = v.normalize()*rad; draw_line(c,center+v,p-v ,color, area); } else { draw_line(c,center,p ,color, area); } // draw the triangle pointing to this node if (area.intersect(circle_area).is_empty() == false) { dlib::vector v(p-center); v = v.normalize(); dlib::vector cross = z.cross(v).normalize(); dlib::vector r(center + v*rad); for (double i = 0; i < 8*zoom_scale(); i += 0.1) draw_line(c,(r+v*i)+cross*i, (r+v*i)-cross*i,color,area); } } } // now draw all the node dots for (unsigned long i = 0; i < graph_.number_of_nodes(); ++i) { center = graph_to_gui_space(graph_.node(i).data.p); const rectangle circle_area(centered_rect(center,2*(rad+8),2*(rad+8))); // draw the actual dot for this node if (area.intersect(circle_area).is_empty()==false) { rgb_alpha_pixel color; assign_pixel(color, graph_.node(i).data.color); // this node is in area so lets draw it and all of its edges as well draw_solid_circle(c,center,rad-3,color,area); color.alpha = 240; draw_circle(c,center,rad-3,color,area); color.alpha = 200; draw_circle(c,center,rad-2.5,color,area); color.alpha = 160; draw_circle(c,center,rad-2.0,color,area); color.alpha = 120; draw_circle(c,center,rad-1.5,color,area); color.alpha = 80; draw_circle(c,center,rad-1.0,color,area); color.alpha = 40; draw_circle(c,center,rad-0.5,color,area); } if (i == selected_node) draw_circle(c,center,rad+5,rgb_pixel(0,0,255),area); } // now draw all the strings last for (unsigned long i = 0; i < graph_.number_of_nodes(); ++i) { center = graph_to_gui_space(graph_.node(i).data.p); rectangle circle_area(centered_rect(center,2*rad+3,2*rad+3)); if (area.intersect(circle_area).is_empty()==false) { rgb_pixel color = graph_.node(i).data.color; // invert this color color.red = 255-color.red; color.green = 255-color.green; color.blue = 255-color.blue; sout << i; unsigned long width, height; mfont->compute_size(sout.str(),width,height); rectangle str_rect(centered_rect(center, width,height)); if (circle_area.contains(str_rect)) { mfont->draw_string(c,str_rect,sout.str(),color,0,std::string::npos,area); // draw the label for this node if it isn't empty if(graph_.node(i).data.name.size() > 0) { rectangle str_rect(graph_.node(i).data.str_rect); str_rect = centered_rect(center.x(), center.y()-rad-mfont->height(), str_rect.width(), str_rect.height()); mfont->draw_string(c,str_rect,graph_.node(i).data.name,0,0,std::string::npos,area); } } sout.str(""); } } } private: struct data { data() : color(0,0,0) {} vector p; dlib::ustring name; rectangle str_rect; rgb_pixel color; }; friend void serialize(const data& item, std::ostream& out) { serialize(item.p, out); serialize(item.name, out); serialize(item.str_rect, out); serialize(item.color, out); } friend void deserialize(data& item, std::istream& in) { deserialize(item.p, in); deserialize(item.name, in); deserialize(item.str_rect, in); deserialize(item.color, in); } mutable std::ostringstream sout; const double radius; unsigned long selected_node; bool mouse_drag; // true if the user is dragging a node point drag_offset; bool edge_selected; unsigned long selected_edge_parent; unsigned long selected_edge_child; any_function node_selected_handler; any_function node_deselected_handler; any_function node_deleted_handler; any_function graph_modified_handler; graph_type external_graph; // rebind the graph_ type to make us a graph_ of data structs typename graph_type::template rebind::other graph_; bool last_mouse_click_in_display; }; // ---------------------------------------------------------------------------------------- // ---------------------------------------------------------------------------------------- // class text_grid // ---------------------------------------------------------------------------------------- // ---------------------------------------------------------------------------------------- class text_grid : public scrollable_region { /*! INITIAL VALUE - has_focus == false - vertical_scroll_increment() == 10 - horizontal_scroll_increment() == 10 - border_color_ == rgb_pixel(128,128,128) CONVENTION - grid.nr() == row_height.size() - grid.nc() == col_width.size() - border_color() == border_color_ - text(r,c) == grid[r][c].text - text_color(r,c) == grid[r][c].text_color - background_color(r,c) == grid[r][c].bg_color - if (the user has clicked on this widget and caused one of the boxes to have input focus) then - has_focus == true - grid[active_row][active_col] == the active text box - cursor_pos == the position of the cursor in the above box - if (the cursor should be displayed) then - show_cursor == true - else - show_cursor == false - else - has_focus == false !*/ public: text_grid ( drawable_window& w ); ~text_grid ( ); void set_grid_size ( unsigned long rows, unsigned long cols ); unsigned long number_of_columns ( ) const; unsigned long number_of_rows ( ) const; int next_free_user_event_number ( ) const; rgb_pixel border_color ( ) const; void set_border_color ( rgb_pixel color ); const std::string text ( unsigned long row, unsigned long col ) const; const std::wstring wtext ( unsigned long row, unsigned long col ) const; const dlib::ustring utext ( unsigned long row, unsigned long col ) const; void set_text ( unsigned long row, unsigned long col, const std::string& str ); void set_text ( unsigned long row, unsigned long col, const std::wstring& str ); void set_text ( unsigned long row, unsigned long col, const dlib::ustring& str ); const rgb_pixel text_color ( unsigned long row, unsigned long col ) const; void set_text_color ( unsigned long row, unsigned long col, const rgb_pixel color ); const rgb_pixel background_color ( unsigned long row, unsigned long col ) const; void set_background_color ( unsigned long row, unsigned long col, const rgb_pixel color ); bool is_editable ( unsigned long row, unsigned long col ) const; void set_editable ( unsigned long row, unsigned long col, bool editable ); void set_column_width ( unsigned long col, unsigned long width ); void set_row_height ( unsigned long row, unsigned long height ); void disable ( ); void hide ( ); template < typename T > void set_text_modified_handler ( T& object, void (T::*eh)(unsigned long, unsigned long) ) { text_modified_handler = make_mfp(object,eh); } void set_text_modified_handler ( const any_function& eh ) { text_modified_handler = eh; } private: void on_user_event ( int num ); void timer_action ( ); /*! ensures - flips the state of show_cursor !*/ void compute_bg_rects ( ); void compute_total_rect ( ); void on_keydown ( unsigned long key, bool is_printable, unsigned long state ); void on_mouse_down ( unsigned long btn, unsigned long state, long x, long y, bool is_double_click ); void on_mouse_up ( unsigned long btn, unsigned long state, long x, long y ); void on_focus_lost ( ); void draw ( const canvas& c ) const; rectangle get_text_rect ( unsigned long row, unsigned long col ) const; rectangle get_bg_rect ( unsigned long row, unsigned long col ) const; struct data_type { data_type(): text_color(0,0,0), bg_color(255,255,255), first(0), is_editable(true) {} dlib::ustring text; rgb_pixel text_color; rgb_pixel bg_color; rectangle bg_rect; dlib::ustring::size_type first; bool is_editable; }; void drop_input_focus ( ); void move_cursor ( long row, long col, long new_cursor_pos ); array2d grid; array col_width; array row_height; bool has_focus; long active_col; long active_row; long cursor_pos; bool show_cursor; bool recent_cursor_move; timer cursor_timer; rgb_pixel border_color_; any_function text_modified_handler; }; // ---------------------------------------------------------------------------------------- class image_display : public scrollable_region { /*! INITIAL VALUE - img.size() == 0 - overlay_rects.size() == 0 - overlay_lines.size() == 0 - drawing_rect == false - rect_is_selected == false CONVENTION - img == the image this object displays - overlay_rects == the overlay rectangles this object displays - overlay_lines == the overlay lines this object displays - if (drawing_rect) then - the user is drawing a rectangle on the screen and is thus holding down CTRL and the left mouse button. - rect_anchor == the point on the screen where the user clicked to begin drawing the rectangle. - rect_to_draw == the rectangle which should appear on the screen. - if (rect_is_selected) then - selected_rect == the index in overlay_rects of the user selected rectangle. - last_right_click_pos == the last place we saw the user right click the mouse. - parts_menu.is_enabled() == true - if (it is actually a part of this rect that is selected) then - selected_part_name == the name of the part in overlay_rects[selected_rect].parts that is selected. - else - selected_part_name.size() == 0 - else - parts_menu.is_enabled() == false - selected_part_name.size() == 0 - if (moving_overlay) then - moving_rect == the index in overlay_rects that the move applies to. - if (moving_what == MOVING_PART) then - moving_part_name == the name of the part in overlay_rects[moving_rect] that is being moved around with the mouse. - else - moving_what will tell us which side of the rectangle in overlay_rects[moving_rect] is being moved by the mouse. !*/ public: image_display( drawable_window& w ); ~image_display( ); template < typename image_type > void set_image ( const image_type& new_img ) { auto_mutex M(m); // if the new image has a different size when compared to the previous image // then we should readjust the total rectangle size. if (num_rows(new_img) != img.nr() || num_columns(new_img) != img.nc()) { if (zoom_in_scale != 1) set_total_rect_size(num_columns(new_img)*zoom_in_scale, num_rows(new_img)*zoom_in_scale); else set_total_rect_size(num_columns(new_img)/zoom_out_scale, num_rows(new_img)/zoom_out_scale); } else { parent.invalidate_rectangle(rect); } highlighted_rect = std::numeric_limits::max(); rect_is_selected = false; parts_menu.disable(); assign_image_scaled(img,new_img); } virtual void set_pos ( long x, long y ) { auto_mutex lock(m); scrollable_region::set_pos(x,y); parts_menu.set_rect(rect); } virtual void set_size ( unsigned long width, unsigned long height ) { auto_mutex lock(m); scrollable_region::set_size(width,height); parts_menu.set_rect(rect); } struct overlay_rect { overlay_rect() :crossed_out(false) { assign_pixel(color, 0);} template overlay_rect(const rectangle& r, pixel_type p) : rect(r),crossed_out(false) { assign_pixel(color, p); } template overlay_rect(const rectangle& r, pixel_type p, const std::string& l) : rect(r),label(l),crossed_out(false) { assign_pixel(color, p); } template overlay_rect(const rectangle& r, pixel_type p, const std::string& l, const std::map& parts_) : rect(r),label(l),parts(parts_),crossed_out(false) { assign_pixel(color, p); } rectangle rect; rgb_alpha_pixel color; std::string label; std::map parts; bool crossed_out; }; struct overlay_line { overlay_line() { assign_pixel(color, 0);} template overlay_line(const dpoint& p1_, const dpoint& p2_, pixel_type p) : p1(p1_), p2(p2_) { assign_pixel(color, p); } dpoint p1; dpoint p2; rgb_alpha_pixel color; }; struct overlay_circle { overlay_circle():radius(0) { assign_pixel(color, 0);} template overlay_circle(const point& center_, const double radius_, pixel_type p) : center(center_), radius(radius_) { assign_pixel(color, p); } template overlay_circle(const point& center_, const double radius_, pixel_type p, const std::string& l) : center(center_), radius(radius_), label(l) { assign_pixel(color, p); } point center; double radius; rgb_alpha_pixel color; std::string label; }; void add_overlay ( const overlay_rect& overlay ); void add_overlay ( const overlay_line& overlay ); void add_overlay ( const overlay_circle& overlay ); void add_overlay ( const std::vector& overlay ); void add_overlay ( const std::vector& overlay ); void add_overlay ( const std::vector& overlay ); void clear_overlay ( ); rectangle get_image_display_rect ( ) const; std::vector get_overlay_rects ( ) const; void set_default_overlay_rect_label ( const std::string& label ); std::string get_default_overlay_rect_label ( ) const; void set_default_overlay_rect_color ( const rgb_alpha_pixel& color ); rgb_alpha_pixel get_default_overlay_rect_color ( ) const; template < typename T > void set_overlay_rects_changed_handler ( T& object, void (T::*event_handler_)() ) { auto_mutex M(m); event_handler = make_mfp(object,event_handler_); } void set_overlay_rects_changed_handler ( const any_function& event_handler_ ) { auto_mutex M(m); event_handler = event_handler_; } template < typename T > void set_overlay_rect_selected_handler ( T& object, void (T::*event_handler_)(const overlay_rect& orect) ) { auto_mutex M(m); orect_selected_event_handler = make_mfp(object,event_handler_); } void set_overlay_rect_selected_handler ( const any_function& event_handler_ ) { auto_mutex M(m); orect_selected_event_handler = event_handler_; } template < typename T > void set_image_clicked_handler ( T& object, void (T::*event_handler_)(const point& p, bool is_double_click, unsigned long btn) ) { auto_mutex M(m); image_clicked_handler = make_mfp(object,event_handler_); } void set_image_clicked_handler ( const any_function& event_handler_ ) { auto_mutex M(m); image_clicked_handler = event_handler_; } void add_labelable_part_name ( const std::string& name ); void clear_labelable_part_names ( ); void enable_overlay_editing ( ) { auto_mutex M(m); overlay_editing_enabled = true; } void disable_overlay_editing ( ) { auto_mutex M(m); overlay_editing_enabled = false; rect_is_selected = false; drawing_rect = false; parent.invalidate_rectangle(rect); } bool overlay_editing_is_enabled ( ) const { auto_mutex M(m); return overlay_editing_enabled; } void zoom_in ( ); void zoom_out ( ); private: void draw ( const canvas& c ) const; void on_wheel_up ( unsigned long state ); void on_wheel_down ( unsigned long state ); void on_mouse_down ( unsigned long btn, unsigned long state, long x, long y, bool is_double_click ); void on_mouse_up ( unsigned long btn, unsigned long state, long x, long y ); void on_mouse_move ( unsigned long state, long x, long y ); void on_keydown ( unsigned long key, bool is_printable, unsigned long state ); void on_part_add ( const std::string& part_name ); rectangle get_rect_on_screen ( unsigned long idx ) const; rectangle get_rect_on_screen ( rectangle orect ) const; rgb_alpha_pixel invert_pixel (const rgb_alpha_pixel& p) const { return rgb_alpha_pixel(255-p.red, 255-p.green, 255-p.blue, p.alpha); } virtual int next_free_user_event_number ( ) const { return scrollable_region::next_free_user_event_number()+1; } // The reason for using user actions here rather than just having the timer just call // what it needs directly is to avoid a potential deadlock during destruction of this widget. void timer_event_unhighlight_rect() { highlight_timer.stop(); parent.trigger_user_event(this,scrollable_region::next_free_user_event_number()); } void on_user_event (int num) { // ignore this user event if it isn't for us if (num != scrollable_region::next_free_user_event_number()) return; if (highlighted_rect < overlay_rects.size()) { highlighted_rect = std::numeric_limits::max(); parent.invalidate_rectangle(rect); } } array2d img; std::vector overlay_rects; std::vector overlay_lines; std::vector overlay_circles; long zoom_in_scale; long zoom_out_scale; bool drawing_rect; point rect_anchor; rectangle rect_to_draw; bool rect_is_selected; std::string selected_part_name; unsigned long selected_rect; rgb_alpha_pixel default_rect_color; std::string default_rect_label; any_function event_handler; any_function orect_selected_event_handler; any_function image_clicked_handler; popup_menu_region parts_menu; point last_right_click_pos; const double part_width; std::set part_names; bool overlay_editing_enabled; timer highlight_timer; unsigned long highlighted_rect; bool holding_shift_key; bool moving_overlay; unsigned long moving_rect; enum { MOVING_RECT_LEFT, MOVING_RECT_TOP, MOVING_RECT_RIGHT, MOVING_RECT_BOTTOM, MOVING_PART } moving_what; std::string moving_part_name; // restricted functions image_display(image_display&); // copy constructor image_display& operator=(image_display&); // assignment operator }; // ---------------------------------------------------------------------------------------- class perspective_display : public drawable, noncopyable { public: perspective_display( drawable_window& w ); ~perspective_display( ); virtual void set_size ( unsigned long width, unsigned long height ); struct overlay_line { overlay_line() { assign_pixel(color, 0);} overlay_line(const vector& p1_, const vector& p2_) : p1(p1_), p2(p2_) { assign_pixel(color, 255); } template overlay_line(const vector& p1_, const vector& p2_, pixel_type p) : p1(p1_), p2(p2_) { assign_pixel(color, p); } vector p1; vector p2; rgb_pixel color; }; struct overlay_dot { overlay_dot() { assign_pixel(color, 0);} overlay_dot(const vector& p_) : p(p_) { assign_pixel(color, 255); } template overlay_dot(const vector& p_, pixel_type color_) : p(p_) { assign_pixel(color, color_); } vector p; rgb_pixel color; }; void add_overlay ( const std::vector& overlay ); void add_overlay ( const std::vector& overlay ); void clear_overlay ( ); template < typename T > void set_dot_double_clicked_handler ( T& object, void (T::*event_handler_)(const vector&) ) { auto_mutex M(m); dot_clicked_event_handler = make_mfp(object,event_handler_); } void set_dot_double_clicked_handler ( const any_function&)>& event_handler_ ); private: void draw ( const canvas& c ) const; void on_wheel_up ( unsigned long state ); void on_wheel_down ( unsigned long state ); void on_mouse_down ( unsigned long btn, unsigned long state, long x, long y, bool is_double_click ); void on_mouse_move ( unsigned long state, long x, long y ); static bool compare_second ( const std::pair& a, const std::pair& b ) { return a.second < b.second; } point last; std::vector overlay_lines; std::vector overlay_dots; camera_transform tform; vector sum_pts; vector max_pts; any_function&)> dot_clicked_event_handler; mutable array2d depth; }; // ---------------------------------------------------------------------------------------- class perspective_window : public drawable_window, noncopyable { public: typedef perspective_display::overlay_line overlay_line; typedef perspective_display::overlay_dot overlay_dot; perspective_window( ) : disp(*this) { set_size(100,100); on_window_resized(); show(); } perspective_window( const std::vector >& point_cloud ) : disp(*this) { set_size(100,100); on_window_resized(); add_overlay(point_cloud); show(); } perspective_window( const std::vector >& point_cloud, const std::string& title ) : disp(*this) { set_size(100,100); on_window_resized(); add_overlay(point_cloud); set_title(title); show(); } ~perspective_window( ) { // You should always call close_window() in the destructor of window // objects to ensure that no events will be sent to this window while // it is being destructed. close_window(); } void add_overlay ( const std::vector& overlay ) { disp.add_overlay(overlay); } void add_overlay ( const std::vector& overlay ) { disp.add_overlay(overlay); } void clear_overlay ( ) { disp.clear_overlay(); } template void add_overlay(const vector& p1, const vector& p2, pixel_type p) { add_overlay(std::vector(1,overlay_line(p1,p2,p))); } void add_overlay(const std::vector >& d) { add_overlay(d, 255); } template void add_overlay(const std::vector >& d, pixel_type p) { std::vector temp; temp.resize(d.size()); for (unsigned long i = 0; i < temp.size(); ++i) temp[i] = overlay_dot(d[i], p); add_overlay(temp); } template < typename T > void set_dot_double_clicked_handler ( T& object, void (T::*event_handler_)(const vector&) ) { disp.set_dot_double_clicked_handler(object,event_handler_); } void set_dot_double_clicked_handler ( const any_function&)>& event_handler_ ) { disp.set_dot_double_clicked_handler(event_handler_); } private: void on_window_resized( ) { drawable_window::on_window_resized(); unsigned long width, height; get_size(width,height); disp.set_pos(0,0); disp.set_size(width, height); } perspective_display disp; }; // ---------------------------------------------------------------------------------------- class image_window : public drawable_window { public: typedef image_display::overlay_rect overlay_rect; typedef image_display::overlay_line overlay_line; typedef image_display::overlay_circle overlay_circle; image_window( ); template < typename image_type > image_window( const image_type& img ) : gui_img(*this), window_has_closed(false), have_last_click(false), mouse_btn(0), clicked_signaler(this->wm), have_last_keypress(false), tie_input_events(false) { gui_img.set_image_clicked_handler(*this, &image_window::on_image_clicked); gui_img.disable_overlay_editing(); set_image(img); show(); } template < typename image_type > image_window( const image_type& img, const std::string& title ) : gui_img(*this), window_has_closed(false), have_last_click(false), mouse_btn(0), clicked_signaler(this->wm), have_last_keypress(false), tie_input_events(false) { gui_img.set_image_clicked_handler(*this, &image_window::on_image_clicked); gui_img.disable_overlay_editing(); set_image(img); set_title(title); show(); } ~image_window( ); template < typename image_type > void set_image ( const image_type& img ) { const unsigned long padding = scrollable_region_style_default().get_border_size(); auto_mutex M(wm); gui_img.set_image(img); // Only ever mess with the size of the window if the user is giving us an image // that is a different size. Otherwise we assume that they will have already // sized the window to whatever they feel is reasonable for an image of the // current size. if (previous_image_size != get_rect(img)) { const rectangle r = gui_img.get_image_display_rect(); if (image_rect != r) { // set the size of this window to match the size of the input image set_size(r.width()+padding*2,r.height()+padding*2); // call this to make sure everything else is setup properly on_window_resized(); image_rect = r; } previous_image_size = get_rect(img); } } void add_overlay ( const overlay_rect& overlay ); template void add_overlay(const rectangle& r, pixel_type p) { add_overlay(image_display::overlay_rect(r,p)); } void add_overlay(const rectangle& r) { add_overlay(image_display::overlay_rect(r,rgb_pixel(255,0,0))); } template void add_overlay(const rectangle& r, pixel_type p, const std::string& l) { add_overlay(image_display::overlay_rect(r,p,l)); } template void add_overlay(const std::vector& r, pixel_type p) { std::vector temp; temp.resize(r.size()); for (unsigned long i = 0; i < temp.size(); ++i) temp[i] = overlay_rect(r[i], p); add_overlay(temp); } void add_overlay(const std::vector& r) { add_overlay(r, rgb_pixel(255,0,0)); } void add_overlay( const full_object_detection& object, const std::vector& part_names ) { add_overlay(overlay_rect(object.get_rect(), rgb_pixel(255,0,0))); std::vector temp; temp.reserve(object.num_parts()); for (unsigned long i = 0; i < object.num_parts(); ++i) { if (object.part(i) != OBJECT_PART_NOT_PRESENT) { if (i < part_names.size()) temp.push_back(overlay_circle(object.part(i), 7, rgb_pixel(0,255,0), part_names[i])); else temp.push_back(overlay_circle(object.part(i), 7, rgb_pixel(0,255,0))); } } add_overlay(temp); } void add_overlay( const full_object_detection& object ) { std::vector part_names; add_overlay(object, part_names); } void add_overlay( const std::vector& objects, const std::vector& part_names ) { std::vector rtemp; rtemp.reserve(objects.size()); for (unsigned long i = 0; i < objects.size(); ++i) { rtemp.push_back(overlay_rect(objects[i].get_rect(), rgb_pixel(255,0,0))); } add_overlay(rtemp); std::vector temp; for (unsigned long i = 0; i < objects.size(); ++i) { for (unsigned long j = 0; j < objects[i].num_parts(); ++j) { if (objects[i].part(j) != OBJECT_PART_NOT_PRESENT) { if (j < part_names.size()) temp.push_back(overlay_circle(objects[i].part(j), 7, rgb_pixel(0,255,0),part_names[j])); else temp.push_back(overlay_circle(objects[i].part(j), 7, rgb_pixel(0,255,0))); } } } add_overlay(temp); } void add_overlay( const std::vector& objects ) { std::vector part_names; add_overlay(objects, part_names); } void add_overlay ( const overlay_line& overlay ); template void add_overlay(const line& l, pixel_type p) { add_overlay(image_display::overlay_line(l.p1(),l.p2(),p)); } void add_overlay(const line& l) { add_overlay(l, rgb_pixel(255,0,0)); } void add_overlay ( const overlay_circle& overlay ); template void add_overlay(const point& p1, const point& p2, pixel_type p) { add_overlay(image_display::overlay_line(p1,p2,p)); } void add_overlay ( const std::vector& overlay ); void add_overlay ( const std::vector& overlay ); void add_overlay ( const std::vector& overlay ); void clear_overlay ( ); bool get_next_double_click ( point& p, unsigned long& mouse_button ); void tie_events ( ); void untie_events ( ); bool events_tied ( ) const; bool get_next_double_click ( point& p ) { unsigned long mouse_button; return get_next_double_click(p, mouse_button); } bool get_next_keypress ( unsigned long& key, bool& is_printable, unsigned long& state ); bool get_next_keypress ( unsigned long& key, bool& is_printable ) { unsigned long state; return get_next_keypress(key,is_printable,state); } private: virtual base_window::on_close_return_code on_window_close( ); void on_window_resized( ); void on_image_clicked ( const point& p, bool is_double_click, unsigned long btn ); virtual void on_keydown ( unsigned long key, bool is_printable, unsigned long state ); // restricted functions image_window(image_window&); image_window& operator= (image_window&); image_display gui_img; rectangle image_rect; rectangle previous_image_size; bool window_has_closed; bool have_last_click; point last_clicked_point; unsigned long mouse_btn; rsignaler clicked_signaler; bool have_last_keypress; unsigned long next_key; bool next_is_printable; unsigned long next_state; bool tie_input_events; }; // ---------------------------------------------------------------------------------------- } #ifdef NO_MAKEFILE #include "widgets.cpp" #endif #endif // DLIB_WIDGETs_