3344 lines
94 KiB
C++
3344 lines
94 KiB
C++
// Copyright (C) 2005 Davis E. King (davis@dlib.net), 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_
|
|
|