/*************************************************************************/ /* scroll_bar.cpp */ /*************************************************************************/ /* This file is part of: */ /* GODOT ENGINE */ /* http://www.godotengine.org */ /*************************************************************************/ /* Copyright (c) 2007-2016 Juan Linietsky, Ariel Manzur. */ /* */ /* Permission is hereby granted, free of charge, to any person obtaining */ /* a copy of this software and associated documentation files (the */ /* "Software"), to deal in the Software without restriction, including */ /* without limitation the rights to use, copy, modify, merge, publish, */ /* distribute, sublicense, and/or sell copies of the Software, and to */ /* permit persons to whom the Software is furnished to do so, subject to */ /* the following conditions: */ /* */ /* The above copyright notice and this permission notice shall be */ /* included in all copies or substantial portions of the Software. */ /* */ /* THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, */ /* EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF */ /* MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT.*/ /* IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY */ /* CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, */ /* TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE */ /* SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE. */ /*************************************************************************/ #include "scroll_bar.h" #include "os/keyboard.h" #include "print_string.h" #include "os/os.h" bool ScrollBar::focus_by_default=false; void ScrollBar::set_can_focus_by_default(bool p_can_focus) { focus_by_default=p_can_focus; } void ScrollBar::_input_event(InputEvent p_event) { switch(p_event.type) { case InputEvent::MOUSE_BUTTON: { const InputEventMouseButton &b=p_event.mouse_button; accept_event(); if (b.button_index==5 && b.pressed) { //if (orientation==VERTICAL) // set_val( get_val() + get_page() / 4.0 ); //else set_val( get_val() + get_page() / 4.0 ); accept_event(); } if (b.button_index==4 && b.pressed) { //if (orientation==HORIZONTAL) // set_val( get_val() - get_page() / 4.0 ); //else set_val( get_val() - get_page() / 4.0 ); accept_event(); } if (b.button_index!=1) return; if (b.pressed) { double ofs = orientation==VERTICAL ? b.y : b.x ; Ref<Texture> decr = get_icon("decrement"); Ref<Texture> incr = get_icon("increment"); double decr_size = orientation==VERTICAL ? decr->get_height() : decr->get_width(); double incr_size = orientation==VERTICAL ? incr->get_height() : incr->get_width(); double grabber_ofs = get_grabber_offset(); double grabber_size = get_grabber_size(); double total = orientation==VERTICAL ? get_size().height : get_size().width; if (ofs < decr_size ) { set_val( get_val() - (custom_step>=0?custom_step:get_step()) ); break; } if (ofs > total-incr_size ) { set_val( get_val() + (custom_step>=0?custom_step:get_step()) ); break; } ofs-=decr_size; if ( ofs < grabber_ofs ) { set_val( get_val() - get_page() ); break; } ofs-=grabber_ofs; if (ofs < grabber_size ) { drag.active=true; drag.pos_at_click=grabber_ofs+ofs; drag.value_at_click=get_unit_value(); update(); } else { set_val( get_val() + get_page() ); } } else { drag.active=false; update(); } } break; case InputEvent::MOUSE_MOTION: { const InputEventMouseMotion &m=p_event.mouse_motion; accept_event(); if (drag.active) { double ofs = orientation==VERTICAL ? m.y : m.x ; Ref<Texture> decr = get_icon("decrement"); double decr_size = orientation==VERTICAL ? decr->get_height() : decr->get_width(); ofs-=decr_size; double diff = (ofs-drag.pos_at_click) / get_area_size(); set_unit_value( drag.value_at_click + diff ); } else { double ofs = orientation==VERTICAL ? m.y : m.x ; Ref<Texture> decr = get_icon("decrement"); Ref<Texture> incr = get_icon("increment"); double decr_size = orientation==VERTICAL ? decr->get_height() : decr->get_width(); double incr_size = orientation==VERTICAL ? incr->get_height() : incr->get_width(); double total = orientation==VERTICAL ? get_size().height : get_size().width; HiliteStatus new_hilite; if (ofs < decr_size ) { new_hilite=HILITE_DECR; } else if (ofs > total-incr_size ) { new_hilite=HILITE_INCR; } else { new_hilite=HILITE_RANGE; } if (new_hilite!=hilite) { hilite=new_hilite; update(); } } } break; case InputEvent::KEY: { const InputEventKey &k=p_event.key; if (!k.pressed) return; switch (k.scancode) { case KEY_LEFT: { if (orientation!=HORIZONTAL) return; set_val( get_val() - (custom_step>=0?custom_step:get_step()) ); } break; case KEY_RIGHT: { if (orientation!=HORIZONTAL) return; set_val( get_val() + (custom_step>=0?custom_step:get_step()) ); } break; case KEY_UP: { if (orientation!=VERTICAL) return; set_val( get_val() - (custom_step>=0?custom_step:get_step()) ); } break; case KEY_DOWN: { if (orientation!=VERTICAL) return; set_val( get_val() + (custom_step>=0?custom_step:get_step()) ); } break; case KEY_HOME: { set_val( get_min() ); } break; case KEY_END: { set_val( get_max() ); } break; } break; } } } void ScrollBar::_notification(int p_what) { if (p_what==NOTIFICATION_DRAW) { RID ci = get_canvas_item(); Ref<Texture> decr = hilite==HILITE_DECR ? get_icon("decrement_hilite") : get_icon("decrement"); Ref<Texture> incr = hilite==HILITE_INCR ? get_icon("increment_hilite") : get_icon("increment"); Ref<StyleBox> bg = has_focus() ? get_stylebox("scroll_focus") : get_stylebox("scroll"); Ref<StyleBox> grabber = (drag.active || hilite==HILITE_RANGE) ? get_stylebox("grabber_hilite") : get_stylebox("grabber"); Point2 ofs; VisualServer *vs = VisualServer::get_singleton(); vs->canvas_item_add_texture_rect( ci, Rect2( Point2(), decr->get_size()),decr->get_rid() ); if (orientation==HORIZONTAL) ofs.x+=decr->get_width(); else ofs.y+=decr->get_height(); Size2 area=get_size(); if (orientation==HORIZONTAL) area.width-=incr->get_width()+decr->get_width(); else area.height-=incr->get_height()+decr->get_height(); bg->draw(ci,Rect2(ofs,area)); if (orientation==HORIZONTAL) ofs.width+=area.width; else ofs.height+=area.height; vs->canvas_item_add_texture_rect( ci, Rect2( ofs, decr->get_size()),incr->get_rid() ); Rect2 grabber_rect; if (orientation==HORIZONTAL) { grabber_rect.size.width=get_grabber_size(); grabber_rect.size.height=get_size().height; grabber_rect.pos.y=0; grabber_rect.pos.x=get_grabber_offset()+decr->get_width()+bg->get_margin( MARGIN_LEFT ); } else { grabber_rect.size.width=get_size().width; grabber_rect.size.height=get_grabber_size(); grabber_rect.pos.y=get_grabber_offset()+decr->get_height()+bg->get_margin( MARGIN_TOP ); grabber_rect.pos.x=0; } grabber->draw(ci,grabber_rect); } if (p_what==NOTIFICATION_ENTER_TREE) { if (has_node(drag_slave_path)) { Node *n = get_node(drag_slave_path); drag_slave=n->cast_to<Control>(); } if (drag_slave) { drag_slave->connect("input_event",this,"_drag_slave_input"); drag_slave->connect("exit_tree",this,"_drag_slave_exit",varray(),CONNECT_ONESHOT); } } if (p_what==NOTIFICATION_EXIT_TREE) { if (drag_slave) { drag_slave->disconnect("input_event",this,"_drag_slave_input"); drag_slave->disconnect("exit_tree",this,"_drag_slave_exit"); } drag_slave=NULL; } if (p_what==NOTIFICATION_FIXED_PROCESS) { if (drag_slave_touching) { if (drag_slave_touching_deaccel) { Vector2 pos = Vector2(orientation==HORIZONTAL?get_val():0,orientation==VERTICAL?get_val():0); pos+=drag_slave_speed*get_fixed_process_delta_time(); bool turnoff=false; if (orientation==HORIZONTAL) { if (pos.x<0) { pos.x=0; turnoff=true; } if (pos.x > (get_max()-get_page())) { pos.x=get_max()-get_page(); turnoff=true; } set_val(pos.x); float sgn_x = drag_slave_speed.x<0? -1 : 1; float val_x = Math::abs(drag_slave_speed.x); val_x-=1000*get_fixed_process_delta_time(); if (val_x<0) { turnoff=true; } drag_slave_speed.x=sgn_x*val_x; } else { if (pos.y<0) { pos.y=0; turnoff=true; } if (pos.y > (get_max()-get_page())) { pos.y=get_max()-get_page(); turnoff=true; } set_val(pos.y); float sgn_y = drag_slave_speed.y<0? -1 : 1; float val_y = Math::abs(drag_slave_speed.y); val_y-=1000*get_fixed_process_delta_time(); if (val_y<0) { turnoff=true; } drag_slave_speed.y=sgn_y*val_y; } if (turnoff) { set_fixed_process(false); drag_slave_touching=false; drag_slave_touching_deaccel=false; } } else { if (time_since_motion==0 || time_since_motion>0.1) { Vector2 diff = drag_slave_accum - last_drag_slave_accum; last_drag_slave_accum=drag_slave_accum; drag_slave_speed=diff/get_fixed_process_delta_time(); } time_since_motion+=get_fixed_process_delta_time(); } } } if (p_what==NOTIFICATION_MOUSE_EXIT) { hilite=HILITE_NONE; update(); } } double ScrollBar::get_grabber_min_size() const { Ref<StyleBox> grabber=get_stylebox("grabber"); Size2 gminsize=grabber->get_minimum_size()+grabber->get_center_size(); return (orientation==VERTICAL)?gminsize.height:gminsize.width; } double ScrollBar::get_grabber_size() const { float range = get_max()-get_min(); if (range<=0) return 0; float page = (get_page()>0)? get_page() : 0; // if (grabber_range < get_step()) // grabber_range=get_step(); double area_size=get_area_size(); double grabber_size = page / range * area_size; return grabber_size+get_grabber_min_size(); } double ScrollBar::get_area_size() const { if (orientation==VERTICAL) { double area=get_size().height; area-=get_stylebox("scroll")->get_minimum_size().height; area-=get_icon("increment")->get_height(); area-=get_icon("decrement")->get_height(); area-=get_grabber_min_size(); return area; } else if (orientation==HORIZONTAL) { double area=get_size().width; area-=get_stylebox("scroll")->get_minimum_size().width; area-=get_icon("increment")->get_width(); area-=get_icon("decrement")->get_width(); area-=get_grabber_min_size(); return area; } else { return 0; } } double ScrollBar::get_area_offset() const { double ofs=0; if (orientation==VERTICAL) { ofs+=get_stylebox("hscroll")->get_margin( MARGIN_TOP ); ofs+=get_icon("decrement")->get_height(); } if (orientation==HORIZONTAL) { ofs+=get_stylebox("hscroll")->get_margin( MARGIN_LEFT ); ofs+=get_icon("decrement")->get_width(); } return ofs; } double ScrollBar::get_click_pos(const Point2& p_pos) const { float pos=(orientation==VERTICAL)?p_pos.y:p_pos.x; pos-=get_area_offset(); float area=get_area_size(); if (area==0) return 0; else return pos/area; } double ScrollBar::get_grabber_offset() const { return (get_area_size()) * get_unit_value(); } Size2 ScrollBar::get_minimum_size() const { Ref<Texture> incr = get_icon("increment"); Ref<Texture> decr = get_icon("decrement"); Ref<StyleBox> bg = get_stylebox("scroll"); Size2 minsize; if (orientation==VERTICAL) { minsize.width=MAX(incr->get_size().width,(bg->get_minimum_size()+bg->get_center_size()).width); minsize.height+=incr->get_size().height; minsize.height+=decr->get_size().height; minsize.height+=bg->get_minimum_size().height; minsize.height+=get_grabber_min_size(); } if (orientation==HORIZONTAL) { minsize.height=MAX(incr->get_size().height,(bg->get_center_size()+bg->get_minimum_size()).height); minsize.width+=incr->get_size().width; minsize.width+=decr->get_size().width; minsize.width+=bg->get_minimum_size().width; minsize.width+=get_grabber_min_size(); } return minsize; } void ScrollBar::set_custom_step(float p_custom_step) { custom_step=p_custom_step; } float ScrollBar::get_custom_step() const { return custom_step; } void ScrollBar::_drag_slave_exit() { if (drag_slave) { drag_slave->disconnect("input_event",this,"_drag_slave_input"); } drag_slave=NULL; } void ScrollBar::_drag_slave_input(const InputEvent& p_input) { switch(p_input.type) { case InputEvent::MOUSE_BUTTON: { const InputEventMouseButton &mb=p_input.mouse_button; if (mb.button_index!=1) break; if (mb.pressed) { if (drag_slave_touching) { set_fixed_process(false); drag_slave_touching_deaccel=false; drag_slave_touching=false; drag_slave_speed=Vector2(); drag_slave_accum=Vector2(); last_drag_slave_accum=Vector2(); drag_slave_from=Vector2(); } if (true) { drag_slave_speed=Vector2(); drag_slave_accum=Vector2(); last_drag_slave_accum=Vector2(); //drag_slave_from=Vector2(h_scroll->get_val(),v_scroll->get_val()); drag_slave_from= Vector2(orientation==HORIZONTAL?get_val():0,orientation==VERTICAL?get_val():0); drag_slave_touching=OS::get_singleton()->has_touchscreen_ui_hint(); drag_slave_touching_deaccel=false; time_since_motion=0; if (drag_slave_touching) { set_fixed_process(true); time_since_motion=0; } } } else { if (drag_slave_touching) { if (drag_slave_speed==Vector2()) { drag_slave_touching_deaccel=false; drag_slave_touching=false; set_fixed_process(false); } else { drag_slave_touching_deaccel=true; } } } } break; case InputEvent::MOUSE_MOTION: { const InputEventMouseMotion &mm=p_input.mouse_motion; if (drag_slave_touching && ! drag_slave_touching_deaccel) { Vector2 motion = Vector2(mm.relative_x,mm.relative_y); drag_slave_accum-=motion; Vector2 diff = drag_slave_from+drag_slave_accum; if (orientation==HORIZONTAL) set_val(diff.x); //else // drag_slave_accum.x=0; if (orientation==VERTICAL) set_val(diff.y); //else // drag_slave_accum.y=0; time_since_motion=0; } } break; } } void ScrollBar::set_drag_slave(const NodePath& p_path) { if (is_inside_tree()) { if (drag_slave) { drag_slave->disconnect("input_event",this,"_drag_slave_input"); drag_slave->disconnect("exit_tree",this,"_drag_slave_exit"); } } drag_slave=NULL; drag_slave_path=p_path; if (is_inside_tree()) { if (has_node(p_path)) { Node *n = get_node(p_path); drag_slave=n->cast_to<Control>(); } if (drag_slave) { drag_slave->connect("input_event",this,"_drag_slave_input"); drag_slave->connect("exit_tree",this,"_drag_slave_exit",varray(),CONNECT_ONESHOT); } } } NodePath ScrollBar::get_drag_slave() const{ return drag_slave_path; } #if 0 void ScrollBar::mouse_button(const Point2& p_pos, int b.button_index,bool b.pressed,int p_modifier_mask) { // wheel! if (b.button_index==BUTTON_WHEEL_UP && b.pressed) { if (orientation==VERTICAL) set_val( get_val() - get_page() / 4.0 ); else set_val( get_val() + get_page() / 4.0 ); } if (b.button_index==BUTTON_WHEEL_DOWN && b.pressed) { if (orientation==HORIZONTAL) set_val( get_val() - get_page() / 4.0 ); else set_val( get_val() + get_page() / 4.0 ); } if (b.button_index!=BUTTON_LEFT) return; if (b.pressed) { int ofs = orientation==VERTICAL ? p_pos.y : p_pos.x ; int grabber_ofs = get_grabber_offset(); int grabber_size = get_grabber_size(); if ( ofs < grabber_ofs ) { set_val( get_val() - get_page() ); } else if (ofs > grabber_ofs + grabber_size ) { set_val( get_val() + get_page() ); } else { drag.active=true; drag.pos_at_click=get_click_pos(p_pos); drag.value_at_click=get_unit_value(); } } else { drag.active=false; } } void ScrollBar::mouse_motion(const Point2& p_pos, const Point2& p_rel, int b.button_index_mask) { if (!drag.active) return; double value_ofs=drag.value_at_click+(get_click_pos(p_pos)-drag.pos_at_click); value_ofs=value_ofs*( get_max() - get_min() ); if (value_ofs<get_min()) value_ofs=get_min(); if (value_ofs>(get_max()-get_page())) value_ofs=get_max()-get_page(); if (get_val()==value_ofs) return; //dont bother if the value is the same set_val( value_ofs ); } bool ScrollBar::key(unsigned long p_unicode, unsigned long p_scan_code,bool b.pressed,bool p_repeat,int p_modifier_mask) { if (!b.pressed) return false; switch (p_scan_code) { case KEY_LEFT: { if (orientation!=HORIZONTAL) return false; set_val( get_val() - get_step() ); } break; case KEY_RIGHT: { if (orientation!=HORIZONTAL) return false; set_val( get_val() + get_step() ); } break; case KEY_UP: { if (orientation!=VERTICAL) return false; set_val( get_val() - get_step() ); } break; case KEY_DOWN: { if (orientation!=VERTICAL) return false; set_val( get_val() + get_step() ); } break; case KEY_HOME: { set_val( get_min() ); } break; case KEY_END: { set_val( get_max() ); } break; default: return false; } return true; } #endif void ScrollBar::_bind_methods() { ObjectTypeDB::bind_method(_MD("_input_event"),&ScrollBar::_input_event); ObjectTypeDB::bind_method(_MD("set_custom_step","step"),&ScrollBar::set_custom_step); ObjectTypeDB::bind_method(_MD("get_custom_step"),&ScrollBar::get_custom_step); ObjectTypeDB::bind_method(_MD("_drag_slave_input"),&ScrollBar::_drag_slave_input); ObjectTypeDB::bind_method(_MD("_drag_slave_exit"),&ScrollBar::_drag_slave_exit); ADD_PROPERTY( PropertyInfo(Variant::REAL,"custom_step",PROPERTY_HINT_RANGE,"-1,4096"), _SCS("set_custom_step"),_SCS("get_custom_step")); } ScrollBar::ScrollBar(Orientation p_orientation) { orientation=p_orientation; hilite=HILITE_NONE; custom_step=-1; drag_slave=NULL; drag.active=false; drag_slave_speed=Vector2(); drag_slave_touching=false; drag_slave_touching_deaccel=false; if (focus_by_default) set_focus_mode( FOCUS_ALL ); } ScrollBar::~ScrollBar() { }