AshanGimhana's picture
Upload folder using huggingface_hub
9375c9a verified
// Copyright (C) 2005 Davis E. King ([email protected]), Keita Mochizuki
// License: Boost Software License See LICENSE.txt for the full license.
#ifndef DLIB_BASE_WIDGETs_CPP_
#define DLIB_BASE_WIDGETs_CPP_
#include <iostream>
#include <memory>
#include "base_widgets.h"
#include "../assert.h"
namespace dlib
{
// ----------------------------------------------------------------------------------------
// ----------------------------------------------------------------------------------------
// button object methods
// ----------------------------------------------------------------------------------------
// ----------------------------------------------------------------------------------------
void button::
set_size (
unsigned long width,
unsigned long height
)
{
auto_mutex M(m);
rectangle min_rect = style->get_min_size(name_,*mfont);
// only change the size if it isn't going to be too small to fit the name
if (height >= min_rect.height() &&
width >= min_rect.width())
{
rectangle old(rect);
rect = resize_rect(rect,width,height);
parent.invalidate_rectangle(style->get_invalidation_rect(rect+old));
btn_tooltip.set_size(width,height);
}
}
// ----------------------------------------------------------------------------------------
void button::
show (
)
{
button_action::show();
btn_tooltip.show();
}
// ----------------------------------------------------------------------------------------
void button::
hide (
)
{
button_action::hide();
btn_tooltip.hide();
}
// ----------------------------------------------------------------------------------------
void button::
enable (
)
{
button_action::enable();
btn_tooltip.enable();
}
// ----------------------------------------------------------------------------------------
void button::
disable (
)
{
button_action::disable();
btn_tooltip.disable();
}
// ----------------------------------------------------------------------------------------
void button::
set_tooltip_text (
const std::string& text
)
{
btn_tooltip.set_text(text);
}
// ----------------------------------------------------------------------------------------
void button::
set_tooltip_text (
const std::wstring& text
)
{
btn_tooltip.set_text(text);
}
// ----------------------------------------------------------------------------------------
void button::
set_tooltip_text (
const ustring& text
)
{
btn_tooltip.set_text(text);
}
// ----------------------------------------------------------------------------------------
const std::string button::
tooltip_text (
) const
{
return btn_tooltip.text();
}
const std::wstring button::
tooltip_wtext (
) const
{
return btn_tooltip.wtext();
}
const dlib::ustring button::
tooltip_utext (
) const
{
return btn_tooltip.utext();
}
// ----------------------------------------------------------------------------------------
void button::
set_main_font (
const std::shared_ptr<font>& f
)
{
auto_mutex M(m);
mfont = f;
set_name(name_);
}
// ----------------------------------------------------------------------------------------
void button::
set_pos (
long x,
long y
)
{
auto_mutex M(m);
button_action::set_pos(x,y);
btn_tooltip.set_pos(x,y);
}
// ----------------------------------------------------------------------------------------
void button::
set_name (
const std::string& name
)
{
set_name(convert_mbstring_to_wstring(name));
}
void button::
set_name (
const std::wstring& name
)
{
set_name(convert_wstring_to_utf32(name));
}
void button::
set_name (
const ustring& name
)
{
auto_mutex M(m);
name_ = name;
// do this to get rid of any reference counting that may be present in
// the std::string implementation.
name_[0] = name_[0];
rectangle old(rect);
rect = move_rect(style->get_min_size(name,*mfont),rect.left(),rect.top());
btn_tooltip.set_size(rect.width(),rect.height());
parent.invalidate_rectangle(style->get_invalidation_rect(rect+old));
}
// ----------------------------------------------------------------------------------------
const std::string button::
name (
) const
{
auto_mutex M(m);
std::string temp = convert_wstring_to_mbstring(wname());
// do this to get rid of any reference counting that may be present in
// the std::string implementation.
char c = temp[0];
temp[0] = c;
return temp;
}
const std::wstring button::
wname (
) const
{
auto_mutex M(m);
std::wstring temp = convert_utf32_to_wstring(uname());
// do this to get rid of any reference counting that may be present in
// the std::wstring implementation.
wchar_t w = temp[0];
temp[0] = w;
return temp;
}
const dlib::ustring button::
uname (
) const
{
auto_mutex M(m);
dlib::ustring temp = name_;
// do this to get rid of any reference counting that may be present in
// the dlib::ustring implementation.
temp[0] = name_[0];
return temp;
}
// ----------------------------------------------------------------------------------------
void button::
on_button_up (
bool mouse_over
)
{
if (mouse_over)
{
// this is a valid button click
if (event_handler.is_set())
event_handler();
if (event_handler_self.is_set())
event_handler_self(*this);
}
if (button_up_handler.is_set())
button_up_handler(mouse_over);
if (button_up_handler_self.is_set())
button_up_handler_self(mouse_over,*this);
}
// ----------------------------------------------------------------------------------------
void button::
on_button_down (
)
{
if (button_down_handler.is_set())
button_down_handler();
if (button_down_handler_self.is_set())
button_down_handler_self(*this);
}
// ----------------------------------------------------------------------------------------
// ----------------------------------------------------------------------------------------
// draggable object methods
// ----------------------------------------------------------------------------------------
// ----------------------------------------------------------------------------------------
draggable::~draggable() {}
// ----------------------------------------------------------------------------------------
void draggable::
on_mouse_move (
unsigned long state,
long x,
long y
)
{
if (drag && (state & base_window::LEFT) && enabled && !hidden)
{
// the user is trying to drag this object. we should calculate the new
// x and y positions for the upper left corner of this object's rectangle
long new_x = x - this->x;
long new_y = y - this->y;
// make sure these points are inside the draggable area.
if (new_x < area.left())
new_x = area.left();
if (new_x + static_cast<long>(rect.width()) - 1 > area.right())
new_x = area.right() - rect.width() + 1;
if (new_y + static_cast<long>(rect.height()) - 1 > area.bottom())
new_y = area.bottom() - rect.height() + 1;
if (new_y < area.top())
new_y = area.top();
// now make the new rectangle for this object
rectangle new_rect(
new_x,
new_y,
new_x + rect.width() - 1,
new_y + rect.height() - 1
);
// only do anything if this is a new rectangle and it is inside area
if (new_rect != rect && area.intersect(new_rect) == new_rect)
{
parent.invalidate_rectangle(new_rect + rect);
rect = new_rect;
// call the on_drag() event handler
on_drag();
}
}
else
{
drag = false;
on_drag_stop();
}
}
// ----------------------------------------------------------------------------------------
void draggable::
on_mouse_up (
unsigned long ,
unsigned long state,
long ,
long
)
{
if (drag && (state & base_window::LEFT) == 0)
{
drag = false;
on_drag_stop();
}
}
// ----------------------------------------------------------------------------------------
void draggable::
on_mouse_down (
unsigned long btn,
unsigned long ,
long x,
long y,
bool
)
{
if (enabled && !hidden && rect.contains(x,y) && btn == base_window::LEFT)
{
drag = true;
this->x = x - rect.left();
this->y = y - rect.top();
}
}
// ----------------------------------------------------------------------------------------
// ----------------------------------------------------------------------------------------
// mouse_over_event object methods
// ----------------------------------------------------------------------------------------
// ----------------------------------------------------------------------------------------
mouse_over_event::~mouse_over_event() {}
// ----------------------------------------------------------------------------------------
void mouse_over_event::
on_mouse_leave (
)
{
if (is_mouse_over_)
{
is_mouse_over_ = false;
on_mouse_not_over();
}
}
// ----------------------------------------------------------------------------------------
void mouse_over_event::
on_mouse_move (
unsigned long ,
long x,
long y
)
{
if (rect.contains(x,y) == false)
{
if (is_mouse_over_)
{
is_mouse_over_ = false;
on_mouse_not_over();
}
}
else if (is_mouse_over_ == false)
{
is_mouse_over_ = true;
if (enabled && !hidden)
on_mouse_over();
}
}
// ----------------------------------------------------------------------------------------
bool mouse_over_event::
is_mouse_over (
) const
{
// check if the mouse is still really over this button
if (is_mouse_over_ && rect.contains(lastx,lasty) == false)
{
// trigger a user event to call on_mouse_not_over() and repaint this object.
// we must do this in another event because someone might call is_mouse_over()
// from draw() and you don't want this function to end up calling
// parent.invalidate_rectangle(). It would lead to draw() being called over
// and over.
parent.trigger_user_event((void*)this,drawable::next_free_user_event_number());
return false;
}
return is_mouse_over_;
}
// ----------------------------------------------------------------------------------------
void mouse_over_event::
on_user_event (
int num
)
{
if (is_mouse_over_ && num == drawable::next_free_user_event_number())
{
is_mouse_over_ = false;
on_mouse_not_over();
}
}
// ----------------------------------------------------------------------------------------
// ----------------------------------------------------------------------------------------
// button_action object methods
// ----------------------------------------------------------------------------------------
// ----------------------------------------------------------------------------------------
button_action::~button_action() {}
// ----------------------------------------------------------------------------------------
void button_action::
on_mouse_down (
unsigned long btn,
unsigned long ,
long x,
long y,
bool
)
{
if (enabled && !hidden && btn == base_window::LEFT && rect.contains(x,y))
{
is_depressed_ = true;
seen_click = true;
parent.invalidate_rectangle(rect);
on_button_down();
}
}
// ----------------------------------------------------------------------------------------
void button_action::
on_mouse_not_over (
)
{
if (is_depressed_)
{
is_depressed_ = false;
parent.invalidate_rectangle(rect);
on_button_up(false);
}
}
// ----------------------------------------------------------------------------------------
void button_action::
on_mouse_move (
unsigned long state,
long x,
long y
)
{
// forward event to the parent class so it can do it's thing as well as us
mouse_over_event::on_mouse_move(state,x,y);
if (enabled == false || hidden == true)
return;
if ((state & base_window::LEFT) == 0)
{
seen_click = false;
if (is_depressed_)
{
is_depressed_ = false;
parent.invalidate_rectangle(rect);
on_button_up(false);
}
// the left button isn't down so we don't care about anything else
return;
}
if (rect.contains(x,y) == false)
{
if (is_depressed_)
{
is_depressed_ = false;
parent.invalidate_rectangle(rect);
on_button_up(false);
}
}
else if (is_depressed_ == false && seen_click)
{
is_depressed_ = true;
parent.invalidate_rectangle(rect);
on_button_down();
}
}
// ----------------------------------------------------------------------------------------
void button_action::
on_mouse_up (
unsigned long btn,
unsigned long,
long x,
long y
)
{
if (enabled && !hidden && btn == base_window::LEFT)
{
if (is_depressed_)
{
is_depressed_ = false;
parent.invalidate_rectangle(rect);
if (rect.contains(x,y))
{
on_button_up(true);
}
else
{
on_button_up(false);
}
}
else if (seen_click && rect.contains(x,y))
{
// this case here covers the unlikly event that you click on a button,
// move the mouse off the button and then move it back very quickly and
// release the mouse button. It is possible that this mouse up event
// will occur before any mouse move event so you might not have set
// that the button is depressed yet.
// So we should say that this triggers an on_button_down() event and
// then an on_button_up(true) event.
parent.invalidate_rectangle(rect);
on_button_down();
on_button_up(true);
}
seen_click = false;
}
}
// ----------------------------------------------------------------------------------------
bool button_action::
is_depressed (
) const
{
// check if the mouse is still really over this button
if (enabled && !hidden && is_depressed_ && rect.contains(lastx,lasty) == false)
{
// trigger a user event to call on_button_up() and repaint this object.
// we must do this in another event because someone might call is_depressed()
// from draw() and you don't want this function to end up calling
// parent.invalidate_rectangle(). It would lead to draw() being called over
// and over.
parent.trigger_user_event((void*)this,mouse_over_event::next_free_user_event_number());
return false;
}
return is_depressed_;
}
// ----------------------------------------------------------------------------------------
void button_action::
on_user_event (
int num
)
{
// forward event to the parent class so it can do it's thing as well as us
mouse_over_event::on_user_event(num);
if (is_depressed_ && num == mouse_over_event::next_free_user_event_number())
{
is_depressed_ = false;
parent.invalidate_rectangle(rect);
on_button_up(false);
}
}
// ----------------------------------------------------------------------------------------
// ----------------------------------------------------------------------------------------
// scroll_bar object methods
// ----------------------------------------------------------------------------------------
// ----------------------------------------------------------------------------------------
scroll_bar::
scroll_bar(
drawable_window& w,
bar_orientation orientation
) :
drawable(w),
b1(w),
b2(w),
slider(w,*this,&scroll_bar::on_slider_drag),
ori(orientation),
top_filler(w,*this,&scroll_bar::top_filler_down,&scroll_bar::top_filler_up),
bottom_filler(w,*this,&scroll_bar::bottom_filler_down,&scroll_bar::bottom_filler_up),
pos(0),
max_pos(0),
js(10),
b1_timer(*this,&scroll_bar::b1_down_t),
b2_timer(*this,&scroll_bar::b2_down_t),
top_filler_timer(*this,&scroll_bar::top_filler_down_t),
bottom_filler_timer(*this,&scroll_bar::bottom_filler_down_t)
{
set_style(scroll_bar_style_default());
// don't show the slider when there is no place it can move.
slider.hide();
set_length(100);
b1.set_button_down_handler(*this,&scroll_bar::b1_down);
b2.set_button_down_handler(*this,&scroll_bar::b2_down);
b1.set_button_up_handler(*this,&scroll_bar::b1_up);
b2.set_button_up_handler(*this,&scroll_bar::b2_up);
b1.disable();
b2.disable();
enable_events();
}
// ----------------------------------------------------------------------------------------
scroll_bar::
~scroll_bar(
)
{
disable_events();
parent.invalidate_rectangle(rect);
// wait for all the timers to be stopped
b1_timer.stop_and_wait();
b2_timer.stop_and_wait();
top_filler_timer.stop_and_wait();
bottom_filler_timer.stop_and_wait();
}
// ----------------------------------------------------------------------------------------
scroll_bar::bar_orientation scroll_bar::
orientation (
) const
{
auto_mutex M(m);
return ori;
}
// ----------------------------------------------------------------------------------------
void scroll_bar::
set_length (
unsigned long length
)
{
auto_mutex M(m);
// make the min length be at least 1
if (length == 0)
{
length = 1;
}
parent.invalidate_rectangle(rect);
if (ori == HORIZONTAL)
{
rect.set_right(rect.left() + length - 1);
rect.set_bottom(rect.top() + style->get_width() - 1);
const long btn_size = style->get_button_length(rect.width(), max_pos);
b1.set_size(btn_size,style->get_width());
b2.set_size(btn_size,style->get_width());
slider.set_size(get_slider_size(),style->get_width());
}
else
{
rect.set_right(rect.left() + style->get_width() - 1);
rect.set_bottom(rect.top() + length - 1);
const long btn_size = style->get_button_length(rect.height(), max_pos);
b1.set_size(style->get_width(),btn_size);
b2.set_size(style->get_width(),btn_size);
slider.set_size(style->get_width(),get_slider_size());
}
// call this to put everything is in the right spot.
set_pos (rect.left(),rect.top());
if ((b2.get_rect().top() - b1.get_rect().bottom() - 1 <= 8 && ori == VERTICAL) ||
(b2.get_rect().left() - b1.get_rect().right() - 1 <= 8 && ori == HORIZONTAL) ||
max_pos == 0)
{
hide_slider();
}
else if (enabled && !hidden)
{
show_slider();
}
}
// ----------------------------------------------------------------------------------------
void scroll_bar::
set_pos (
long x,
long y
)
{
auto_mutex M(m);
drawable::set_pos(x,y);
b1.set_pos(rect.left(),rect.top());
if (ori == HORIZONTAL)
{
// make the b2 button appear at the end of the scroll_bar
b2.set_pos(rect.right()-b2.get_rect().width() + 1,rect.top());
if (max_pos != 0)
{
double range = b2.get_rect().left() - b1.get_rect().right() - slider.get_rect().width() - 1;
double slider_pos = pos;
slider_pos /= max_pos;
slider_pos *= range;
slider.set_pos(
static_cast<long>(slider_pos)+rect.left() + b1.get_rect().width(),
rect.top()
);
// move the draggable area for the slider to the new location
rectangle area = rect;
area.set_left(area.left() + style->get_width());
area.set_right(area.right() - style->get_width());
slider.set_draggable_area(area);
}
}
else
{
// make the b2 button appear at the end of the scroll_bar
b2.set_pos(rect.left(), rect.bottom() - b2.get_rect().height() + 1);
if (max_pos != 0)
{
double range = b2.get_rect().top() - b1.get_rect().bottom() - slider.get_rect().height() - 1;
double slider_pos = pos;
slider_pos /= max_pos;
slider_pos *= range;
slider.set_pos(
rect.left(),
static_cast<long>(slider_pos) + rect.top() + b1.get_rect().height()
);
// move the draggable area for the slider to the new location
rectangle area = rect;
area.set_top(area.top() + style->get_width());
area.set_bottom(area.bottom() - style->get_width());
slider.set_draggable_area(area);
}
}
adjust_fillers();
}
// ----------------------------------------------------------------------------------------
unsigned long scroll_bar::
get_slider_size (
) const
{
if (ori == HORIZONTAL)
return style->get_slider_length(rect.width(),max_pos);
else
return style->get_slider_length(rect.height(),max_pos);
}
// ----------------------------------------------------------------------------------------
void scroll_bar::
adjust_fillers (
)
{
rectangle top(rect), bottom(rect);
if (ori == HORIZONTAL)
{
if (slider.is_hidden())
{
top.set_left(b1.get_rect().right()+1);
top.set_right(b2.get_rect().left()-1);
bottom.set_left(1);
bottom.set_right(-1);
}
else
{
top.set_left(b1.get_rect().right()+1);
top.set_right(slider.get_rect().left()-1);
bottom.set_left(slider.get_rect().right()+1);
bottom.set_right(b2.get_rect().left()-1);
}
}
else
{
if (slider.is_hidden())
{
top.set_top(b1.get_rect().bottom()+1);
top.set_bottom(b2.get_rect().top()-1);
bottom.set_top(1);
bottom.set_bottom(-1);
}
else
{
top.set_top(b1.get_rect().bottom()+1);
top.set_bottom(slider.get_rect().top()-1);
bottom.set_top(slider.get_rect().bottom()+1);
bottom.set_bottom(b2.get_rect().top()-1);
}
}
top_filler.rect = top;
bottom_filler.rect = bottom;
}
// ----------------------------------------------------------------------------------------
void scroll_bar::
hide_slider (
)
{
rectangle top(rect), bottom(rect);
slider.hide();
top_filler.disable();
bottom_filler.disable();
bottom_filler.hide();
if (ori == HORIZONTAL)
{
top.set_left(b1.get_rect().right()+1);
top.set_right(b2.get_rect().left()-1);
}
else
{
top.set_top(b1.get_rect().bottom()+1);
top.set_bottom(b2.get_rect().top()-1);
}
top_filler.rect = top;
bottom_filler.rect = bottom;
}
// ----------------------------------------------------------------------------------------
void scroll_bar::
show_slider (
)
{
if ((b2.get_rect().top() - b1.get_rect().bottom() - 1 <= 8 && ori == VERTICAL) ||
(b2.get_rect().left() - b1.get_rect().right() - 1 <= 8 && ori == HORIZONTAL) ||
max_pos == 0)
return;
rectangle top(rect), bottom(rect);
slider.show();
top_filler.enable();
bottom_filler.enable();
bottom_filler.show();
if (ori == HORIZONTAL)
{
top.set_left(b1.get_rect().right()+1);
top.set_right(slider.get_rect().left()-1);
bottom.set_left(slider.get_rect().right()+1);
bottom.set_right(b2.get_rect().left()-1);
}
else
{
top.set_top(b1.get_rect().bottom()+1);
top.set_bottom(slider.get_rect().top()-1);
bottom.set_top(slider.get_rect().bottom()+1);
bottom.set_bottom(b2.get_rect().top()-1);
}
top_filler.rect = top;
bottom_filler.rect = bottom;
}
// ----------------------------------------------------------------------------------------
long scroll_bar::
max_slider_pos (
) const
{
auto_mutex M(m);
return max_pos;
}
// ----------------------------------------------------------------------------------------
void scroll_bar::
set_max_slider_pos (
long mpos
)
{
auto_mutex M(m);
max_pos = mpos;
if (pos > mpos)
pos = mpos;
if (ori == HORIZONTAL)
set_length(rect.width());
else
set_length(rect.height());
if (mpos != 0 && enabled)
{
b1.enable();
b2.enable();
}
else
{
b1.disable();
b2.disable();
}
}
// ----------------------------------------------------------------------------------------
void scroll_bar::
set_slider_pos (
long pos
)
{
auto_mutex M(m);
if (pos < 0)
pos = 0;
if (pos > max_pos)
pos = max_pos;
this->pos = pos;
// move the slider object to its new position
set_pos(rect.left(),rect.top());
}
// ----------------------------------------------------------------------------------------
long scroll_bar::
slider_pos (
) const
{
auto_mutex M(m);
return pos;
}
// ----------------------------------------------------------------------------------------
void scroll_bar::
on_slider_drag (
)
{
if (ori == HORIZONTAL)
{
double slider_pos = slider.get_rect().left() - b1.get_rect().right() - 1;
double range = b2.get_rect().left() - b1.get_rect().right() - slider.get_rect().width() - 1;
slider_pos /= range;
slider_pos *= max_pos;
pos = static_cast<unsigned long>(slider_pos);
}
else
{
double slider_pos = slider.get_rect().top() - b1.get_rect().bottom() - 1;
double range = b2.get_rect().top() - b1.get_rect().bottom() - slider.get_rect().height() - 1;
slider_pos /= range;
slider_pos *= max_pos;
pos = static_cast<unsigned long>(slider_pos);
}
adjust_fillers();
if (scroll_handler.is_set())
scroll_handler();
}
// ----------------------------------------------------------------------------------------
void scroll_bar::
draw (
const canvas&
) const
{
}
// ----------------------------------------------------------------------------------------
void scroll_bar::
b1_down (
)
{
if (pos != 0)
{
set_slider_pos(pos-1);
if (scroll_handler.is_set())
scroll_handler();
if (b1_timer.delay_time() == 1000)
b1_timer.set_delay_time(500);
else
b1_timer.set_delay_time(50);
b1_timer.start();
}
}
// ----------------------------------------------------------------------------------------
void scroll_bar::
b1_up (
bool
)
{
b1_timer.stop();
b1_timer.set_delay_time(1000);
}
// ----------------------------------------------------------------------------------------
void scroll_bar::
b2_down (
)
{
if (pos != max_pos)
{
set_slider_pos(pos+1);
if (scroll_handler.is_set())
scroll_handler();
if (b2_timer.delay_time() == 1000)
b2_timer.set_delay_time(500);
else
b2_timer.set_delay_time(50);
b2_timer.start();
}
}
// ----------------------------------------------------------------------------------------
void scroll_bar::
b2_up (
bool
)
{
b2_timer.stop();
b2_timer.set_delay_time(1000);
}
// ----------------------------------------------------------------------------------------
void scroll_bar::
top_filler_down (
)
{
// ignore this if the mouse is now outside this object. This could happen
// since the timers are also calling this function.
if (top_filler.rect.contains(lastx,lasty) == false)
{
top_filler_up(false);
return;
}
if (pos != 0)
{
if (pos < js)
{
// if there is less than jump_size() space left then jump the remaining
// amount.
delayed_set_slider_pos(0);
}
else
{
delayed_set_slider_pos(pos-js);
}
if (top_filler_timer.delay_time() == 1000)
top_filler_timer.set_delay_time(500);
else
top_filler_timer.set_delay_time(50);
top_filler_timer.start();
}
}
// ----------------------------------------------------------------------------------------
void scroll_bar::
top_filler_up (
bool
)
{
top_filler_timer.stop();
top_filler_timer.set_delay_time(1000);
}
// ----------------------------------------------------------------------------------------
void scroll_bar::
bottom_filler_down (
)
{
// ignore this if the mouse is now outside this object. This could happen
// since the timers are also calling this function.
if (bottom_filler.rect.contains(lastx,lasty) == false)
{
bottom_filler_up(false);
return;
}
if (pos != max_pos)
{
if (max_pos - pos < js)
{
// if there is less than jump_size() space left then jump the remaining
// amount.
delayed_set_slider_pos(max_pos);
}
else
{
delayed_set_slider_pos(pos+js);
}
if (bottom_filler_timer.delay_time() == 1000)
bottom_filler_timer.set_delay_time(500);
else
bottom_filler_timer.set_delay_time(50);
bottom_filler_timer.start();
}
}
// ----------------------------------------------------------------------------------------
void scroll_bar::
bottom_filler_up (
bool
)
{
bottom_filler_timer.stop();
bottom_filler_timer.set_delay_time(1000);
}
// ----------------------------------------------------------------------------------------
void scroll_bar::
set_jump_size (
long js_
)
{
auto_mutex M(m);
if (js_ < 1)
js = 1;
else
js = js_;
}
// ----------------------------------------------------------------------------------------
long scroll_bar::
jump_size (
) const
{
auto_mutex M(m);
return js;
}
// ----------------------------------------------------------------------------------------
void scroll_bar::
on_user_event (
int i
)
{
switch (i)
{
case 0:
b1_down();
break;
case 1:
b2_down();
break;
case 2:
top_filler_down();
break;
case 3:
bottom_filler_down();
break;
case 4:
// if the position we are supposed to switch the slider too isn't
// already set
if (delayed_pos != pos)
{
set_slider_pos(delayed_pos);
if (scroll_handler.is_set())
scroll_handler();
}
break;
default:
break;
}
}
// ----------------------------------------------------------------------------------------
void scroll_bar::
delayed_set_slider_pos (
unsigned long dpos
)
{
delayed_pos = dpos;
parent.trigger_user_event(this,4);
}
// ----------------------------------------------------------------------------------------
void scroll_bar::
b1_down_t (
)
{
parent.trigger_user_event(this,0);
}
// ----------------------------------------------------------------------------------------
void scroll_bar::
b2_down_t (
)
{
parent.trigger_user_event(this,1);
}
// ----------------------------------------------------------------------------------------
void scroll_bar::
top_filler_down_t (
)
{
parent.trigger_user_event(this,2);
}
// ----------------------------------------------------------------------------------------
void scroll_bar::
bottom_filler_down_t (
)
{
parent.trigger_user_event(this,3);
}
// ----------------------------------------------------------------------------------------
// ----------------------------------------------------------------------------------------
// widget_group object methods
// ----------------------------------------------------------------------------------------
// ----------------------------------------------------------------------------------------
void widget_group::
empty (
)
{
auto_mutex M(m);
widgets.clear();
wg_widgets.clear();
}
// ----------------------------------------------------------------------------------------
void widget_group::
add (
drawable& widget,
unsigned long x,
unsigned long y
)
{
auto_mutex M(m);
drawable* w = &widget;
relpos rp;
rp.x = x;
rp.y = y;
if (widgets.is_in_domain(w))
{
widgets[w].x = x;
widgets[w].y = y;
}
else
{
widgets.add(w,rp);
}
if (is_hidden())
widget.hide();
else
widget.show();
if (is_enabled())
widget.enable();
else
widget.disable();
widget.set_z_order(z_order());
widget.set_pos(x+rect.left(),y+rect.top());
}
// ----------------------------------------------------------------------------------------
void widget_group::
add (
widget_group& widget,
unsigned long x,
unsigned long y
)
{
auto_mutex M(m);
drawable& w = widget;
add(w, x, y);
widget_group* wg = &widget;
wg_widgets.add(wg);
}
// ----------------------------------------------------------------------------------------
bool widget_group::
is_member (
const drawable& widget
) const
{
auto_mutex M(m);
drawable* w = const_cast<drawable*>(&widget);
return widgets.is_in_domain(w);
}
// ----------------------------------------------------------------------------------------
void widget_group::
remove (
const drawable& widget
)
{
auto_mutex M(m);
drawable* w = const_cast<drawable*>(&widget);
if (widgets.is_in_domain(w))
{
widgets.destroy(w);
// check if we also have an entry in the wg_widgets set and if
// so then remove that too
widget_group* wg = reinterpret_cast<widget_group*>(w);
if (wg_widgets.is_member(wg))
{
wg_widgets.destroy(wg);
}
}
}
// ----------------------------------------------------------------------------------------
size_t widget_group::
size (
) const
{
auto_mutex M(m);
return widgets.size();
}
// ----------------------------------------------------------------------------------------
void widget_group::
set_pos (
long x,
long y
)
{
auto_mutex M(m);
widgets.reset();
while (widgets.move_next())
{
const unsigned long rx = widgets.element().value().x;
const unsigned long ry = widgets.element().value().y;
widgets.element().key()->set_pos(x+rx,y+ry);
}
drawable::set_pos(x,y);
}
// ----------------------------------------------------------------------------------------
void widget_group::
set_z_order (
long order
)
{
auto_mutex M(m);
widgets.reset();
while (widgets.move_next())
widgets.element().key()->set_z_order(order);
drawable::set_z_order(order);
}
// ----------------------------------------------------------------------------------------
void widget_group::
show (
)
{
auto_mutex M(m);
widgets.reset();
while (widgets.move_next())
widgets.element().key()->show();
drawable::show();
}
// ----------------------------------------------------------------------------------------
void widget_group::
hide (
)
{
auto_mutex M(m);
widgets.reset();
while (widgets.move_next())
widgets.element().key()->hide();
drawable::hide();
}
// ----------------------------------------------------------------------------------------
void widget_group::
enable (
)
{
auto_mutex M(m);
widgets.reset();
while (widgets.move_next())
widgets.element().key()->enable();
drawable::enable();
}
// ----------------------------------------------------------------------------------------
void widget_group::
disable ()
{
auto_mutex M(m);
widgets.reset();
while (widgets.move_next())
widgets.element().key()->disable();
drawable::disable();
}
// ----------------------------------------------------------------------------------------
void widget_group::
fit_to_contents (
)
{
auto_mutex M(m);
// call fit_to_contents on all the widget_groups we contain
wg_widgets.reset();
while (wg_widgets.move_next())
wg_widgets.element()->fit_to_contents();
// now accumulate a rectangle that contains everything in this widget_group
rectangle r;
widgets.reset();
while (widgets.move_next())
r = r + widgets.element().key()->get_rect();
if (r.is_empty())
{
// make sure it is still empty after we set it at the correct position
r.set_right(rect.left()-1);
r.set_bottom(rect.top()-1);
}
r.set_left(rect.left());
r.set_top(rect.top());
rect = r;
}
// ----------------------------------------------------------------------------------------
// ----------------------------------------------------------------------------------------
// class popup_menu
// ----------------------------------------------------------------------------------------
// ----------------------------------------------------------------------------------------
popup_menu::
popup_menu (
) :
base_window(false,true),
pad(2),
item_pad(3),
cur_rect(pad,pad,pad-1,pad-1),
left_width(0),
middle_width(0),
selected_item(0),
submenu_open(false)
{
}
// ----------------------------------------------------------------------------------------
void popup_menu::
enable_menu_item (
unsigned long idx
)
{
DLIB_ASSERT ( idx < size() ,
"\tvoid popup_menu::enable_menu_item()"
<< "\n\tidx: " << idx
<< "\n\tsize(): " << size()
);
auto_mutex M(wm);
item_enabled[idx] = true;
invalidate_rectangle(cur_rect);
}
// ----------------------------------------------------------------------------------------
void popup_menu::
disable_menu_item (
unsigned long idx
)
{
DLIB_ASSERT ( idx < size() ,
"\tvoid popup_menu::enable_menu_item()"
<< "\n\tidx: " << idx
<< "\n\tsize(): " << size()
);
auto_mutex M(wm);
item_enabled[idx] = false;
invalidate_rectangle(cur_rect);
}
// ----------------------------------------------------------------------------------------
size_t popup_menu::
size (
) const
{
auto_mutex M(wm);
return items.size();
}
// ----------------------------------------------------------------------------------------
void popup_menu::
clear (
)
{
auto_mutex M(wm);
hide();
cur_rect = rectangle(pad,pad,pad-1,pad-1);
win_rect = rectangle();
left_width = 0;
middle_width = 0;
items.clear();
item_enabled.clear();
left_rects.clear();
middle_rects.clear();
right_rects.clear();
line_rects.clear();
submenus.clear();
selected_item = 0;
submenu_open = false;
}
// ----------------------------------------------------------------------------------------
void popup_menu::
show (
)
{
auto_mutex M(wm);
selected_item = submenus.size();
base_window::show();
}
// ----------------------------------------------------------------------------------------
void popup_menu::
hide (
)
{
auto_mutex M(wm);
// hide ourselves
close_submenu();
selected_item = submenus.size();
base_window::hide();
}
// ----------------------------------------------------------------------------------------
void popup_menu::
select_first_item (
)
{
auto_mutex M(wm);
close_submenu();
selected_item = items.size();
for (unsigned long i = 0; i < items.size(); ++i)
{
if ((items[i]->has_click_event() || submenus[i]) && item_enabled[i])
{
selected_item = i;
break;
}
}
invalidate_rectangle(cur_rect);
}
// ----------------------------------------------------------------------------------------
bool popup_menu::
forwarded_on_keydown (
unsigned long key,
bool is_printable,
unsigned long state
)
{
auto_mutex M(wm);
// do nothing if this popup menu is empty
if (items.size() == 0)
return false;
// check if the selected item is a submenu
if (selected_item != submenus.size() && submenus[selected_item] != 0 && submenu_open)
{
// send the key to the submenu and return if that menu used the key
if (submenus[selected_item]->forwarded_on_keydown(key,is_printable,state) == true)
return true;
}
if (key == KEY_UP)
{
for (unsigned long i = 0; i < items.size(); ++i)
{
selected_item = (selected_item + items.size() - 1)%items.size();
// only stop looking if this one is enabled and has a click event or is a submenu
if (item_enabled[selected_item] && (items[selected_item]->has_click_event() || submenus[selected_item]) )
break;
}
invalidate_rectangle(cur_rect);
return true;
}
else if (key == KEY_DOWN)
{
for (unsigned long i = 0; i < items.size(); ++i)
{
selected_item = (selected_item + 1)%items.size();
// only stop looking if this one is enabled and has a click event or is a submenu
if (item_enabled[selected_item] && (items[selected_item]->has_click_event() || submenus[selected_item]))
break;
}
invalidate_rectangle(cur_rect);
return true;
}
else if (key == KEY_RIGHT && submenu_open == false && display_selected_submenu())
{
submenus[selected_item]->select_first_item();
return true;
}
else if (key == KEY_LEFT && selected_item != submenus.size() &&
submenus[selected_item] != 0 && submenu_open)
{
close_submenu();
return true;
}
else if (key == '\n')
{
if (selected_item != submenus.size() && (items[selected_item]->has_click_event() || submenus[selected_item]))
{
const long idx = selected_item;
// only hide this popup window if this isn't a submenu
if (submenus[idx] == 0)
{
hide();
hide_handlers.reset();
while (hide_handlers.move_next())
hide_handlers.element()();
}
else
{
display_selected_submenu();
submenus[idx]->select_first_item();
}
items[idx]->on_click();
return true;
}
}
else if (is_printable)
{
// check if there is a hotkey for this key
for (unsigned long i = 0; i < items.size(); ++i)
{
if (std::tolower(key) == std::tolower(items[i]->get_hot_key()) &&
(items[i]->has_click_event() || submenus[i]) && item_enabled[i] )
{
// only hide this popup window if this isn't a submenu
if (submenus[i] == 0)
{
hide();
hide_handlers.reset();
while (hide_handlers.move_next())
hide_handlers.element()();
}
else
{
if (selected_item != items.size())
invalidate_rectangle(line_rects[selected_item]);
selected_item = i;
display_selected_submenu();
invalidate_rectangle(line_rects[i]);
submenus[i]->select_first_item();
}
items[i]->on_click();
}
}
// always say we use a printable key for hotkeys
return true;
}
return false;
}
// ----------------------------------------------------------------------------------------
void popup_menu::
on_submenu_hide (
)
{
hide();
hide_handlers.reset();
while (hide_handlers.move_next())
hide_handlers.element()();
}
// ----------------------------------------------------------------------------------------
void popup_menu::
on_window_resized(
)
{
invalidate_rectangle(win_rect);
}
// ----------------------------------------------------------------------------------------
void popup_menu::
on_mouse_up (
unsigned long btn,
unsigned long,
long x,
long y
)
{
if (cur_rect.contains(x,y) && btn == LEFT)
{
// figure out which item this was on
for (unsigned long i = 0; i < items.size(); ++i)
{
if (line_rects[i].contains(x,y) && item_enabled[i] && items[i]->has_click_event())
{
// only hide this popup window if this isn't a submenu
if (submenus[i] == 0)
{
hide();
hide_handlers.reset();
while (hide_handlers.move_next())
hide_handlers.element()();
}
items[i]->on_click();
break;
}
}
}
}
// ----------------------------------------------------------------------------------------
void popup_menu::
on_mouse_move (
unsigned long ,
long x,
long y
)
{
if (cur_rect.contains(x,y))
{
// check if the mouse is still in the same rect it was in last time
rectangle last_rect;
if (selected_item != submenus.size())
{
last_rect = line_rects[selected_item];
}
// if the mouse isn't in the same rectangle any more
if (last_rect.contains(x,y) == false)
{
if (selected_item != submenus.size())
{
invalidate_rectangle(last_rect);
close_submenu();
selected_item = submenus.size();
}
// figure out if we should redraw any menu items
for (unsigned long i = 0; i < items.size(); ++i)
{
if (items[i]->has_click_event() || submenus[i])
{
if (line_rects[i].contains(x,y))
{
selected_item = i;
break;
}
}
}
// if we found a rectangle that contains the mouse then
// tell it to redraw itself
if (selected_item != submenus.size())
{
display_selected_submenu();
invalidate_rectangle(line_rects[selected_item]);
}
}
}
}
// ----------------------------------------------------------------------------------------
void popup_menu::
close_submenu (
)
{
if (selected_item != submenus.size() && submenus[selected_item] && submenu_open)
{
submenus[selected_item]->hide();
submenu_open = false;
}
}
// ----------------------------------------------------------------------------------------
bool popup_menu::
display_selected_submenu (
)
{
// show the submenu if one exists
if (selected_item != submenus.size() && submenus[selected_item])
{
long wx, wy;
get_pos(wx,wy);
wx += line_rects[selected_item].right();
wy += line_rects[selected_item].top();
submenus[selected_item]->set_pos(wx+1,wy-2);
submenus[selected_item]->show();
submenu_open = true;
return true;
}
return false;
}
// ----------------------------------------------------------------------------------------
void popup_menu::
on_mouse_leave (
)
{
if (selected_item != submenus.size())
{
// only unhighlight a menu item if it isn't a submenu item
if (submenus[selected_item] == 0)
{
invalidate_rectangle(line_rects[selected_item]);
selected_item = submenus.size();
}
}
}
// ----------------------------------------------------------------------------------------
void popup_menu::
paint (
const canvas& c
)
{
c.fill(200,200,200);
draw_rectangle(c, win_rect);
for (unsigned long i = 0; i < items.size(); ++i)
{
bool is_selected = false;
if (selected_item != submenus.size() && i == selected_item &&
item_enabled[i])
is_selected = true;
items[i]->draw_background(c,line_rects[i], item_enabled[i], is_selected);
items[i]->draw_left(c,left_rects[i], item_enabled[i], is_selected);
items[i]->draw_middle(c,middle_rects[i], item_enabled[i], is_selected);
items[i]->draw_right(c,right_rects[i], item_enabled[i], is_selected);
}
}
// ----------------------------------------------------------------------------------------
// ----------------------------------------------------------------------------------------
// class zoomable_region
// ----------------------------------------------------------------------------------------
// ----------------------------------------------------------------------------------------
zoomable_region::
zoomable_region (
drawable_window& w,
unsigned long events
) :
drawable(w,MOUSE_CLICK | MOUSE_WHEEL | MOUSE_MOVE | events),
min_scale(0.15),
max_scale(1.0),
zoom_increment_(0.90),
vsb(w, scroll_bar::VERTICAL),
hsb(w, scroll_bar::HORIZONTAL)
{
scale = 1;
mouse_drag_screen = false;
style.reset(new scrollable_region_style_default());
hsb.set_scroll_handler(*this,&zoomable_region::on_h_scroll);
vsb.set_scroll_handler(*this,&zoomable_region::on_v_scroll);
}
// ----------------------------------------------------------------------------------------
zoomable_region::
~zoomable_region()
{
}
// ----------------------------------------------------------------------------------------
void zoomable_region::
set_pos (
long x,
long y
)
{
auto_mutex M(m);
drawable::set_pos(x,y);
const long border_size = style->get_border_size();
vsb.set_pos(rect.right()-border_size+1-vsb.width(),rect.top()+border_size);
hsb.set_pos(rect.left()+border_size,rect.bottom()-border_size+1-hsb.height());
display_rect_ = rectangle(rect.left()+border_size,
rect.top()+border_size,
rect.right()-border_size-vsb.width(),
rect.bottom()-border_size-hsb.height());
}
// ----------------------------------------------------------------------------------------
void zoomable_region::
set_zoom_increment (
double zi
)
{
DLIB_ASSERT(0.0 < zi && zi < 1.0,
"\tvoid zoomable_region::set_zoom_increment(zi)"
<< "\n\t the zoom increment must be between 0 and 1"
<< "\n\t zi: " << zi
<< "\n\t this: " << this
);
auto_mutex M(m);
zoom_increment_ = zi;
}
// ----------------------------------------------------------------------------------------
double zoomable_region::
zoom_increment (
) const
{
auto_mutex M(m);
return zoom_increment_;
}
// ----------------------------------------------------------------------------------------
void zoomable_region::
set_max_zoom_scale (
double ms
)
{
DLIB_ASSERT(ms > 0,
"\tvoid zoomable_region::set_max_zoom_scale(ms)"
<< "\n\t the max zoom scale must be greater than 0"
<< "\n\t ms: " << ms
<< "\n\t this: " << this
);
auto_mutex M(m);
max_scale = ms;
if (scale > ms)
{
scale = max_scale;
lr_point = gui_to_graph_space(point(display_rect_.right(),display_rect_.bottom()));
redraw_graph();
}
}
// ----------------------------------------------------------------------------------------
void zoomable_region::
set_min_zoom_scale (
double ms
)
{
DLIB_ASSERT(ms > 0,
"\tvoid zoomable_region::set_min_zoom_scale(ms)"
<< "\n\t the min zoom scale must be greater than 0"
<< "\n\t ms: " << ms
<< "\n\t this: " << this
);
auto_mutex M(m);
min_scale = ms;
if (scale < ms)
{
scale = min_scale;
}
// just call set_size so that everything gets redrawn right
set_size(rect.width(), rect.height());
}
// ----------------------------------------------------------------------------------------
double zoomable_region::
min_zoom_scale (
) const
{
auto_mutex M(m);
return min_scale;
}
// ----------------------------------------------------------------------------------------
double zoomable_region::
max_zoom_scale (
) const
{
auto_mutex M(m);
return max_scale;
}
// ----------------------------------------------------------------------------------------
void zoomable_region::
set_size (
unsigned long width,
unsigned long height
)
{
auto_mutex M(m);
rectangle old(rect);
const long border_size = style->get_border_size();
rect = resize_rect(rect,width,height);
vsb.set_pos(rect.right()-border_size+1-vsb.width(), rect.top()+border_size);
hsb.set_pos(rect.left()+border_size, rect.bottom()-border_size+1-hsb.height());
display_rect_ = rectangle(rect.left()+border_size,
rect.top()+border_size,
rect.right()-border_size-vsb.width(),
rect.bottom()-border_size-hsb.height());
vsb.set_length(display_rect_.height());
hsb.set_length(display_rect_.width());
parent.invalidate_rectangle(rect+old);
const double old_scale = scale;
const vector<double,2> old_gr_orig(gr_orig);
scale = min_scale;
gr_orig = vector<double,2>(0,0);
lr_point = gui_to_graph_space(point(display_rect_.right(),display_rect_.bottom()));
scale = old_scale;
// call adjust_origin() so that the scroll bars get their max slider positions
// setup right
const point rect_corner(display_rect_.left(), display_rect_.top());
adjust_origin(rect_corner, old_gr_orig);
}
// ----------------------------------------------------------------------------------------
void zoomable_region::
show (
)
{
auto_mutex M(m);
drawable::show();
hsb.show();
vsb.show();
}
// ----------------------------------------------------------------------------------------
void zoomable_region::
hide (
)
{
auto_mutex M(m);
drawable::hide();
hsb.hide();
vsb.hide();
}
// ----------------------------------------------------------------------------------------
void zoomable_region::
enable (
)
{
auto_mutex M(m);
drawable::enable();
hsb.enable();
vsb.enable();
}
// ----------------------------------------------------------------------------------------
void zoomable_region::
disable (
)
{
auto_mutex M(m);
drawable::disable();
hsb.disable();
vsb.disable();
}
// ----------------------------------------------------------------------------------------
void zoomable_region::
set_z_order (
long order
)
{
auto_mutex M(m);
drawable::set_z_order(order);
hsb.set_z_order(order);
vsb.set_z_order(order);
}
// ----------------------------------------------------------------------------------------
point zoomable_region::
graph_to_gui_space (
const vector<double,2>& p
) const
{
const point rect_corner(display_rect_.left(), display_rect_.top());
return (p - gr_orig)*scale + rect_corner;
}
// ----------------------------------------------------------------------------------------
vector<double,2> zoomable_region::
gui_to_graph_space (
const point& p
) const
{
const point rect_corner(display_rect_.left(), display_rect_.top());
return (p - rect_corner)/scale + gr_orig;
}
// ----------------------------------------------------------------------------------------
point zoomable_region::
max_graph_point (
) const
{
return lr_point;
}
// ----------------------------------------------------------------------------------------
rectangle zoomable_region::
display_rect (
) const
{
return display_rect_;
}
// ----------------------------------------------------------------------------------------
double zoomable_region::
zoom_scale (
) const
{
return scale;
}
// ----------------------------------------------------------------------------------------
void zoomable_region::
set_zoom_scale (
double new_scale
)
{
// if new_scale isn't in the right range then put it back in range before we do the
// rest of this function
if (!(min_scale <= new_scale && new_scale <= max_scale))
{
if (new_scale > max_scale)
new_scale = max_scale;
else
new_scale = min_scale;
}
// find the point in the center of the graph area
point center((display_rect_.left()+display_rect_.right())/2, (display_rect_.top()+display_rect_.bottom())/2);
point graph_p(gui_to_graph_space(center));
scale = new_scale;
adjust_origin(center, graph_p);
redraw_graph();
}
// ----------------------------------------------------------------------------------------
void zoomable_region::
center_display_at_graph_point (
const vector<double,2>& p
)
{
// find the point in the center of the graph area
point center((display_rect_.left()+display_rect_.right())/2, (display_rect_.top()+display_rect_.bottom())/2);
adjust_origin(center, p);
redraw_graph();
}
// ----------------------------------------------------------------------------------------
void zoomable_region::
on_wheel_down (
unsigned long
)
{
// zoom out
if (enabled && !hidden && scale > min_scale && display_rect_.contains(lastx,lasty))
{
point gui_p(lastx,lasty);
point graph_p(gui_to_graph_space(gui_p));
const double old_scale = scale;
scale *= zoom_increment_;
if (scale < min_scale)
scale = min_scale;
redraw_graph();
adjust_origin(gui_p, graph_p);
if (scale != old_scale)
on_view_changed();
}
}
// ----------------------------------------------------------------------------------------
void zoomable_region::
on_wheel_up (
unsigned long
)
{
// zoom in
if (enabled && !hidden && scale < max_scale && display_rect_.contains(lastx,lasty))
{
point gui_p(lastx,lasty);
point graph_p(gui_to_graph_space(gui_p));
const double old_scale = scale;
scale /= zoom_increment_;
if (scale > max_scale)
scale = max_scale;
redraw_graph();
adjust_origin(gui_p, graph_p);
if (scale != old_scale)
on_view_changed();
}
}
// ----------------------------------------------------------------------------------------
void zoomable_region::
on_mouse_move (
unsigned long state,
long x,
long y
)
{
if (enabled && !hidden && mouse_drag_screen)
{
adjust_origin(point(x,y), drag_screen_point);
redraw_graph();
on_view_changed();
}
// check if the mouse isn't being dragged anymore
if ((state & base_window::LEFT) == 0)
{
mouse_drag_screen = false;
}
}
// ----------------------------------------------------------------------------------------
void zoomable_region::
on_mouse_up (
unsigned long ,
unsigned long ,
long ,
long
)
{
mouse_drag_screen = false;
}
// ----------------------------------------------------------------------------------------
void zoomable_region::
on_mouse_down (
unsigned long btn,
unsigned long ,
long x,
long y,
bool
)
{
if (enabled && !hidden && display_rect_.contains(x,y) && btn == base_window::LEFT)
{
mouse_drag_screen = true;
drag_screen_point = gui_to_graph_space(point(x,y));
}
}
// ----------------------------------------------------------------------------------------
void zoomable_region::
draw (
const canvas& c
) const
{
style->draw_scrollable_region_border(c, rect, enabled);
}
// ----------------------------------------------------------------------------------------
void zoomable_region::
on_h_scroll (
)
{
gr_orig.x() = hsb.slider_pos();
redraw_graph();
on_view_changed();
}
// ----------------------------------------------------------------------------------------
void zoomable_region::
on_v_scroll (
)
{
gr_orig.y() = vsb.slider_pos();
redraw_graph();
on_view_changed();
}
// ----------------------------------------------------------------------------------------
void zoomable_region::
redraw_graph (
)
{
parent.invalidate_rectangle(display_rect_);
}
// ----------------------------------------------------------------------------------------
void zoomable_region::
adjust_origin (
const point& gui_p,
const vector<double,2>& graph_p
)
{
const point rect_corner(display_rect_.left(), display_rect_.top());
const dlib::vector<double,2> v(gui_p - rect_corner);
gr_orig = graph_p - v/scale;
// make sure the origin isn't outside the point (0,0)
if (gr_orig.x() < 0)
gr_orig.x() = 0;
if (gr_orig.y() < 0)
gr_orig.y() = 0;
// make sure the lower right corner of the display_rect_ doesn't map to a point beyond lr_point
point lr_rect_corner(display_rect_.right(), display_rect_.bottom());
point p = graph_to_gui_space(lr_point);
vector<double,2> lr_rect_corner_graph_space(gui_to_graph_space(lr_rect_corner));
vector<double,2> delta(lr_point - lr_rect_corner_graph_space);
if (lr_rect_corner.x() > p.x())
{
gr_orig.x() += delta.x();
}
if (lr_rect_corner.y() > p.y())
{
gr_orig.y() += delta.y();
}
const vector<double,2> ul_rect_corner_graph_space(gui_to_graph_space(rect_corner));
lr_rect_corner_graph_space = gui_to_graph_space(lr_rect_corner);
// now adjust the scroll bars
hsb.set_max_slider_pos((unsigned long)std::max(lr_point.x()-(lr_rect_corner_graph_space.x()-ul_rect_corner_graph_space.x()),0.0));
vsb.set_max_slider_pos((unsigned long)std::max(lr_point.y()-(lr_rect_corner_graph_space.y()-ul_rect_corner_graph_space.y()),0.0));
// adjust slider position now.
hsb.set_slider_pos(static_cast<long>(ul_rect_corner_graph_space.x()));
vsb.set_slider_pos(static_cast<long>(ul_rect_corner_graph_space.y()));
}
// ----------------------------------------------------------------------------------------
// ----------------------------------------------------------------------------------------
// class scrollable_region
// ----------------------------------------------------------------------------------------
// ----------------------------------------------------------------------------------------
scrollable_region::
scrollable_region (
drawable_window& w,
unsigned long events
) :
drawable(w, MOUSE_WHEEL|events|MOUSE_CLICK|MOUSE_MOVE),
hsb(w,scroll_bar::HORIZONTAL),
vsb(w,scroll_bar::VERTICAL),
hscroll_bar_inc(1),
vscroll_bar_inc(1),
h_wheel_scroll_bar_inc(1),
v_wheel_scroll_bar_inc(1),
mouse_drag_enabled_(false),
user_is_dragging_mouse(false)
{
style.reset(new scrollable_region_style_default());
hsb.set_scroll_handler(*this,&scrollable_region::on_h_scroll);
vsb.set_scroll_handler(*this,&scrollable_region::on_v_scroll);
}
// ----------------------------------------------------------------------------------------
scrollable_region::
~scrollable_region (
)
{
}
// ----------------------------------------------------------------------------------------
void scrollable_region::
show (
)
{
auto_mutex M(m);
drawable::show();
if (need_h_scroll())
hsb.show();
if (need_v_scroll())
vsb.show();
}
// ----------------------------------------------------------------------------------------
void scrollable_region::
hide (
)
{
auto_mutex M(m);
drawable::hide();
hsb.hide();
vsb.hide();
}
// ----------------------------------------------------------------------------------------
void scrollable_region::
enable (
)
{
auto_mutex M(m);
drawable::enable();
hsb.enable();
vsb.enable();
}
// ----------------------------------------------------------------------------------------
void scrollable_region::
disable (
)
{
auto_mutex M(m);
drawable::disable();
hsb.disable();
vsb.disable();
}
// ----------------------------------------------------------------------------------------
void scrollable_region::
set_z_order (
long order
)
{
auto_mutex M(m);
drawable::set_z_order(order);
hsb.set_z_order(order);
vsb.set_z_order(order);
}
// ----------------------------------------------------------------------------------------
void scrollable_region::
set_size (
unsigned long width,
unsigned long height
)
{
auto_mutex M(m);
rectangle old(rect);
rect = resize_rect(rect,width,height);
vsb.set_pos(rect.right()-style->get_border_size()-vsb.width()+1, rect.top()+style->get_border_size());
hsb.set_pos(rect.left()+style->get_border_size(), rect.bottom()-style->get_border_size()-hsb.height()+1);
// adjust the display_rect_
if (need_h_scroll() && need_v_scroll())
{
// both scroll bars aren't hidden
if (!hidden)
{
vsb.show();
hsb.show();
}
display_rect_ = rectangle( rect.left()+style->get_border_size(),
rect.top()+style->get_border_size(),
rect.right()-style->get_border_size()-vsb.width(),
rect.bottom()-style->get_border_size()-hsb.height());
// figure out how many scroll bar positions there should be
unsigned long hdelta = total_rect_.width()-display_rect_.width();
unsigned long vdelta = total_rect_.height()-display_rect_.height();
hdelta = (hdelta+hscroll_bar_inc-1)/hscroll_bar_inc;
vdelta = (vdelta+vscroll_bar_inc-1)/vscroll_bar_inc;
hsb.set_max_slider_pos(hdelta);
vsb.set_max_slider_pos(vdelta);
vsb.set_jump_size((display_rect_.height()+vscroll_bar_inc-1)/vscroll_bar_inc/2+1);
hsb.set_jump_size((display_rect_.width()+hscroll_bar_inc-1)/hscroll_bar_inc/2+1);
}
else if (need_h_scroll())
{
// only hsb is hidden
if (!hidden)
{
hsb.show();
vsb.hide();
}
display_rect_ = rectangle( rect.left()+style->get_border_size(),
rect.top()+style->get_border_size(),
rect.right()-style->get_border_size(),
rect.bottom()-style->get_border_size()-hsb.height());
// figure out how many scroll bar positions there should be
unsigned long hdelta = total_rect_.width()-display_rect_.width();
hdelta = (hdelta+hscroll_bar_inc-1)/hscroll_bar_inc;
hsb.set_max_slider_pos(hdelta);
vsb.set_max_slider_pos(0);
hsb.set_jump_size((display_rect_.width()+hscroll_bar_inc-1)/hscroll_bar_inc/2+1);
}
else if (need_v_scroll())
{
// only vsb is hidden
if (!hidden)
{
hsb.hide();
vsb.show();
}
display_rect_ = rectangle( rect.left()+style->get_border_size(),
rect.top()+style->get_border_size(),
rect.right()-style->get_border_size()-vsb.width(),
rect.bottom()-style->get_border_size());
unsigned long vdelta = total_rect_.height()-display_rect_.height();
vdelta = (vdelta+vscroll_bar_inc-1)/vscroll_bar_inc;
hsb.set_max_slider_pos(0);
vsb.set_max_slider_pos(vdelta);
vsb.set_jump_size((display_rect_.height()+vscroll_bar_inc-1)/vscroll_bar_inc/2+1);
}
else
{
// both are hidden
if (!hidden)
{
hsb.hide();
vsb.hide();
}
display_rect_ = rectangle( rect.left()+style->get_border_size(),
rect.top()+style->get_border_size(),
rect.right()-style->get_border_size(),
rect.bottom()-style->get_border_size());
hsb.set_max_slider_pos(0);
vsb.set_max_slider_pos(0);
}
vsb.set_length(display_rect_.height());
hsb.set_length(display_rect_.width());
// adjust the total_rect_ position by trigging the scroll events
on_h_scroll();
on_v_scroll();
parent.invalidate_rectangle(rect+old);
}
// ----------------------------------------------------------------------------------------
unsigned long scrollable_region::
horizontal_mouse_wheel_scroll_increment (
) const
{
auto_mutex M(m);
return h_wheel_scroll_bar_inc;
}
// ----------------------------------------------------------------------------------------
unsigned long scrollable_region::
vertical_mouse_wheel_scroll_increment (
) const
{
auto_mutex M(m);
return v_wheel_scroll_bar_inc;
}
// ----------------------------------------------------------------------------------------
void scrollable_region::
set_horizontal_mouse_wheel_scroll_increment (
unsigned long inc
)
{
auto_mutex M(m);
h_wheel_scroll_bar_inc = inc;
}
// ----------------------------------------------------------------------------------------
void scrollable_region::
set_vertical_mouse_wheel_scroll_increment (
unsigned long inc
)
{
auto_mutex M(m);
v_wheel_scroll_bar_inc = inc;
}
// ----------------------------------------------------------------------------------------
unsigned long scrollable_region::
horizontal_scroll_increment (
) const
{
auto_mutex M(m);
return hscroll_bar_inc;
}
// ----------------------------------------------------------------------------------------
unsigned long scrollable_region::
vertical_scroll_increment (
) const
{
auto_mutex M(m);
return vscroll_bar_inc;
}
// ----------------------------------------------------------------------------------------
void scrollable_region::
set_horizontal_scroll_increment (
unsigned long inc
)
{
auto_mutex M(m);
hscroll_bar_inc = inc;
// call set_size to reset the scroll bars
set_size(rect.width(),rect.height());
}
// ----------------------------------------------------------------------------------------
void scrollable_region::
set_vertical_scroll_increment (
unsigned long inc
)
{
auto_mutex M(m);
vscroll_bar_inc = inc;
// call set_size to reset the scroll bars
set_size(rect.width(),rect.height());
}
// ----------------------------------------------------------------------------------------
long scrollable_region::
horizontal_scroll_pos (
) const
{
auto_mutex M(m);
return hsb.slider_pos();
}
// ----------------------------------------------------------------------------------------
long scrollable_region::
vertical_scroll_pos (
) const
{
auto_mutex M(m);
return vsb.slider_pos();
}
// ----------------------------------------------------------------------------------------
void scrollable_region::
set_horizontal_scroll_pos (
long pos
)
{
auto_mutex M(m);
hsb.set_slider_pos(pos);
on_h_scroll();
}
// ----------------------------------------------------------------------------------------
void scrollable_region::
set_vertical_scroll_pos (
long pos
)
{
auto_mutex M(m);
vsb.set_slider_pos(pos);
on_v_scroll();
}
// ----------------------------------------------------------------------------------------
void scrollable_region::
set_pos (
long x,
long y
)
{
auto_mutex M(m);
drawable::set_pos(x,y);
vsb.set_pos(rect.right()-style->get_border_size()-vsb.width()+1, rect.top()+style->get_border_size());
hsb.set_pos(rect.left()+style->get_border_size(), rect.bottom()-style->get_border_size()-hsb.height()+1);
const long delta_x = total_rect_.left() - display_rect_.left();
const long delta_y = total_rect_.top() - display_rect_.top();
display_rect_ = move_rect(display_rect_, rect.left()+style->get_border_size(), rect.top()+style->get_border_size());
total_rect_ = move_rect(total_rect_, display_rect_.left()+delta_x, display_rect_.top()+delta_y);
}
// ----------------------------------------------------------------------------------------
bool scrollable_region::
mouse_drag_enabled (
) const
{
auto_mutex M(m);
return mouse_drag_enabled_;
}
// ----------------------------------------------------------------------------------------
void scrollable_region::
enable_mouse_drag (
)
{
auto_mutex M(m);
mouse_drag_enabled_ = true;
}
// ----------------------------------------------------------------------------------------
void scrollable_region::
disable_mouse_drag (
)
{
auto_mutex M(m);
mouse_drag_enabled_ = false;
}
// ----------------------------------------------------------------------------------------
const rectangle& scrollable_region::
display_rect (
) const
{
return display_rect_;
}
// ----------------------------------------------------------------------------------------
void scrollable_region::
set_total_rect_size (
unsigned long width,
unsigned long height
)
{
DLIB_ASSERT((width > 0 && height > 0) || (width == 0 && height == 0),
"\tvoid scrollable_region::set_total_rect_size(width,height)"
<< "\n\twidth and height must be > 0 or both == 0"
<< "\n\twidth: " << width
<< "\n\theight: " << height
<< "\n\tthis: " << this
);
total_rect_ = move_rect(rectangle(width,height),
display_rect_.left()-static_cast<long>(hsb.slider_pos()),
display_rect_.top()-static_cast<long>(vsb.slider_pos()));
// call this just to reconfigure the scroll bars
set_size(rect.width(),rect.height());
}
// ----------------------------------------------------------------------------------------
const rectangle& scrollable_region::
total_rect (
) const
{
return total_rect_;
}
// ----------------------------------------------------------------------------------------
void scrollable_region::
scroll_to_rect (
const rectangle& r_
)
{
const rectangle r(total_rect_.intersect(r_));
const rectangle old(total_rect_);
// adjust the horizontal scroll bar so that r fits as best as possible
if (r.left() < display_rect_.left())
{
long distance = (r.left()-total_rect_.left())/hscroll_bar_inc;
hsb.set_slider_pos(distance);
}
else if (r.right() > display_rect_.right())
{
long distance = (r.right()-total_rect_.left()-display_rect_.width()+hscroll_bar_inc)/hscroll_bar_inc;
hsb.set_slider_pos(distance);
}
// adjust the vertical scroll bar so that r fits as best as possible
if (r.top() < display_rect_.top())
{
long distance = (r.top()-total_rect_.top())/vscroll_bar_inc;
vsb.set_slider_pos(distance);
}
else if (r.bottom() > display_rect_.bottom())
{
long distance = (r.bottom()-total_rect_.top()-display_rect_.height()+vscroll_bar_inc)/vscroll_bar_inc;
vsb.set_slider_pos(distance);
}
// adjust total_rect_ so that it matches where the scroll bars are now
total_rect_ = move_rect(total_rect_,
display_rect_.left()-hscroll_bar_inc*hsb.slider_pos(),
display_rect_.top()-vscroll_bar_inc*vsb.slider_pos());
// only redraw if we actually changed something
if (total_rect_ != old)
{
parent.invalidate_rectangle(display_rect_);
}
}
// ----------------------------------------------------------------------------------------
void scrollable_region::
on_wheel_down (
unsigned long
)
{
if (rect.contains(lastx,lasty) && enabled && !hidden)
{
if (need_v_scroll())
{
long pos = vsb.slider_pos();
vsb.set_slider_pos(pos+(long)v_wheel_scroll_bar_inc);
on_v_scroll();
}
else if (need_h_scroll())
{
long pos = hsb.slider_pos();
hsb.set_slider_pos(pos+(long)h_wheel_scroll_bar_inc);
on_h_scroll();
}
}
}
// ----------------------------------------------------------------------------------------
void scrollable_region::
on_mouse_move (
unsigned long state,
long x,
long y
)
{
if (enabled && !hidden && user_is_dragging_mouse && state==base_window::LEFT)
{
point current_delta = point(x,y) - point(total_rect().left(), total_rect().top());
rectangle new_rect(translate_rect(display_rect(), drag_origin - current_delta));
new_rect = centered_rect(new_rect, new_rect.width()-hscroll_bar_inc, new_rect.height()-vscroll_bar_inc);
scroll_to_rect(new_rect);
on_view_changed();
}
else
{
user_is_dragging_mouse = false;
}
}
// ----------------------------------------------------------------------------------------
void scrollable_region::
on_mouse_down (
unsigned long btn,
unsigned long ,
long x,
long y,
bool
)
{
if (mouse_drag_enabled_ && enabled && !hidden && display_rect().contains(x,y) && (btn==base_window::LEFT))
{
drag_origin = point(x,y) - point(total_rect().left(), total_rect().top());
user_is_dragging_mouse = true;
}
else
{
user_is_dragging_mouse = false;
}
}
// ----------------------------------------------------------------------------------------
void scrollable_region::
on_mouse_up (
unsigned long ,
unsigned long ,
long ,
long
)
{
user_is_dragging_mouse = false;
}
// ----------------------------------------------------------------------------------------
void scrollable_region::
on_wheel_up (
unsigned long
)
{
if (rect.contains(lastx,lasty) && enabled && !hidden)
{
if (need_v_scroll())
{
long pos = vsb.slider_pos();
vsb.set_slider_pos(pos-(long)v_wheel_scroll_bar_inc);
on_v_scroll();
}
else if (need_h_scroll())
{
long pos = hsb.slider_pos();
hsb.set_slider_pos(pos-(long)h_wheel_scroll_bar_inc);
on_h_scroll();
}
}
}
// ----------------------------------------------------------------------------------------
void scrollable_region::
draw (
const canvas& c
) const
{
style->draw_scrollable_region_border(c, rect, enabled);
}
// ----------------------------------------------------------------------------------------
bool scrollable_region::
need_h_scroll (
) const
{
if (total_rect_.width() > rect.width()-style->get_border_size()*2)
{
return true;
}
else
{
// check if we would need a vertical scroll bar and if adding one would make us need
// a horizontal one
if (total_rect_.height() > rect.height()-style->get_border_size()*2 &&
total_rect_.width() > rect.width()-style->get_border_size()*2-vsb.width())
return true;
else
return false;
}
}
// ----------------------------------------------------------------------------------------
bool scrollable_region::
need_v_scroll (
) const
{
if (total_rect_.height() > rect.height()-style->get_border_size()*2)
{
return true;
}
else
{
// check if we would need a horizontal scroll bar and if adding one would make us need
// a vertical_scroll_pos one
if (total_rect_.width() > rect.width()-style->get_border_size()*2 &&
total_rect_.height() > rect.height()-style->get_border_size()*2-hsb.height())
return true;
else
return false;
}
}
// ----------------------------------------------------------------------------------------
void scrollable_region::
on_h_scroll (
)
{
total_rect_ = move_rect(total_rect_, display_rect_.left()-hscroll_bar_inc*hsb.slider_pos(), total_rect_.top());
parent.invalidate_rectangle(display_rect_);
if (events_are_enabled())
on_view_changed();
}
// ----------------------------------------------------------------------------------------
void scrollable_region::
on_v_scroll (
)
{
total_rect_ = move_rect(total_rect_, total_rect_.left(), display_rect_.top()-vscroll_bar_inc*vsb.slider_pos());
parent.invalidate_rectangle(display_rect_);
if (events_are_enabled())
on_view_changed();
}
// ----------------------------------------------------------------------------------------
// ----------------------------------------------------------------------------------------
// class popup_menu_region
// ----------------------------------------------------------------------------------------
// ----------------------------------------------------------------------------------------
popup_menu_region::
popup_menu_region(
drawable_window& w
) :
drawable(w,MOUSE_CLICK | KEYBOARD_EVENTS | FOCUS_EVENTS | WINDOW_MOVED),
popup_menu_shown(false)
{
menu_.set_on_hide_handler(*this,&popup_menu_region::on_menu_becomes_hidden);
enable_events();
}
// ----------------------------------------------------------------------------------------
popup_menu_region::
~popup_menu_region(
)
{
disable_events();
}
// ----------------------------------------------------------------------------------------
void popup_menu_region::
set_size (
unsigned long width,
unsigned long height
)
{
auto_mutex M(m);
rect = resize_rect(rect,width,height);
}
// ----------------------------------------------------------------------------------------
void popup_menu_region::
set_rect (
const rectangle& new_rect
)
{
auto_mutex M(m);
rect = new_rect;
}
// ----------------------------------------------------------------------------------------
popup_menu& popup_menu_region::
menu (
)
{
return menu_;
}
// ----------------------------------------------------------------------------------------
void popup_menu_region::
hide (
)
{
auto_mutex M(m);
drawable::hide();
menu_.hide();
popup_menu_shown = false;
}
// ----------------------------------------------------------------------------------------
void popup_menu_region::
disable (
)
{
auto_mutex M(m);
drawable::disable();
menu_.hide();
popup_menu_shown = false;
}
// ----------------------------------------------------------------------------------------
void popup_menu_region::
on_keydown (
unsigned long key,
bool is_printable,
unsigned long state
)
{
if (enabled && !hidden && popup_menu_shown)
{
menu_.forwarded_on_keydown(key, is_printable, state);
}
else if (popup_menu_shown)
{
menu_.hide();
popup_menu_shown = false;
}
if (key == (unsigned long)base_window::KEY_ESC)
{
menu_.hide();
popup_menu_shown = false;
}
}
// ----------------------------------------------------------------------------------------
void popup_menu_region::
on_menu_becomes_hidden (
)
{
popup_menu_shown = false;
}
// ----------------------------------------------------------------------------------------
void popup_menu_region::
on_focus_lost (
)
{
if (popup_menu_shown)
{
menu_.hide();
popup_menu_shown = false;
}
}
// ----------------------------------------------------------------------------------------
void popup_menu_region::
on_focus_gained (
)
{
if (popup_menu_shown)
{
menu_.hide();
popup_menu_shown = false;
}
}
// ----------------------------------------------------------------------------------------
void popup_menu_region::
on_window_moved(
)
{
if (popup_menu_shown)
{
menu_.hide();
popup_menu_shown = false;
}
}
// ----------------------------------------------------------------------------------------
void popup_menu_region::
on_mouse_down (
unsigned long btn,
unsigned long ,
long x,
long y,
bool
)
{
if (enabled && !hidden && rect.contains(x,y) && btn == base_window::RIGHT)
{
long orig_x, orig_y;
parent.get_pos(orig_x, orig_y);
menu_.set_pos(orig_x+x, orig_y+y);
menu_.show();
popup_menu_shown = true;
}
else if (popup_menu_shown)
{
menu_.hide();
popup_menu_shown = false;
}
}
// ----------------------------------------------------------------------------------------
void popup_menu_region::
draw (
const canvas&
) const
{
}
// ----------------------------------------------------------------------------------------
}
#endif // DLIB_BASE_WIDGETs_CPP_