2014-02-10 02:10:30 +01:00
/*************************************************************************/
/* tabs.cpp */
/*************************************************************************/
/* This file is part of: */
/* GODOT ENGINE */
2017-08-27 14:16:55 +02:00
/* https://godotengine.org */
2014-02-10 02:10:30 +01:00
/*************************************************************************/
2021-01-01 20:13:46 +01:00
/* Copyright (c) 2007-2021 Juan Linietsky, Ariel Manzur. */
/* Copyright (c) 2014-2021 Godot Engine contributors (cf. AUTHORS.md). */
2014-02-10 02:10:30 +01:00
/* */
/* 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. */
/*************************************************************************/
2018-01-05 00:50:27 +01:00
2014-02-10 02:10:30 +01:00
# include "tabs.h"
2020-11-07 23:33:38 +01:00
# include "core/object/message_queue.h"
2020-09-03 13:22:16 +02:00
# include "core/string/translation.h"
2018-02-07 14:01:45 +01:00
# include "scene/gui/box_container.h"
# include "scene/gui/label.h"
# include "scene/gui/texture_rect.h"
2014-02-10 02:10:30 +01:00
Size2 Tabs : : get_minimum_size ( ) const {
2020-12-08 14:11:45 +01:00
Ref < StyleBox > tab_unselected = get_theme_stylebox ( " tab_unselected " ) ;
Ref < StyleBox > tab_selected = get_theme_stylebox ( " tab_selected " ) ;
2020-03-12 13:37:40 +01:00
Ref < StyleBox > tab_disabled = get_theme_stylebox ( " tab_disabled " ) ;
2014-02-10 02:10:30 +01:00
2020-12-08 14:11:45 +01:00
int y_margin = MAX ( MAX ( tab_unselected - > get_minimum_size ( ) . height , tab_selected - > get_minimum_size ( ) . height ) , tab_disabled - > get_minimum_size ( ) . height ) ;
2020-09-03 13:22:16 +02:00
Size2 ms ( 0 , 0 ) ;
2014-02-10 02:10:30 +01:00
2017-03-05 16:44:50 +01:00
for ( int i = 0 ; i < tabs . size ( ) ; i + + ) {
2019-06-11 20:43:37 +02:00
Ref < Texture2D > tex = tabs [ i ] . icon ;
2014-02-10 02:10:30 +01:00
if ( tex . is_valid ( ) ) {
2016-05-01 16:27:33 +02:00
ms . height = MAX ( ms . height , tex - > get_size ( ) . height ) ;
2020-05-14 16:41:43 +02:00
if ( tabs [ i ] . text ! = " " ) {
2020-03-12 13:37:40 +01:00
ms . width + = get_theme_constant ( " hseparation " ) ;
2020-05-14 16:41:43 +02:00
}
2014-02-10 02:10:30 +01:00
}
2016-05-01 16:27:33 +02:00
2020-09-03 13:22:16 +02:00
ms . width + = Math : : ceil ( tabs [ i ] . text_buf - > get_size ( ) . x ) ;
ms . height = MAX ( ms . height , tabs [ i ] . text_buf - > get_size ( ) . y + y_margin ) ;
2016-05-01 16:27:33 +02:00
2020-05-14 16:41:43 +02:00
if ( tabs [ i ] . disabled ) {
2017-02-27 19:07:50 +01:00
ms . width + = tab_disabled - > get_minimum_size ( ) . width ;
2020-05-14 16:41:43 +02:00
} else if ( current = = i ) {
2020-12-08 14:11:45 +01:00
ms . width + = tab_selected - > get_minimum_size ( ) . width ;
2020-05-14 16:41:43 +02:00
} else {
2020-12-08 14:11:45 +01:00
ms . width + = tab_unselected - > get_minimum_size ( ) . width ;
2020-05-14 16:41:43 +02:00
}
2014-02-10 02:10:30 +01:00
2015-07-26 15:44:10 +02:00
if ( tabs [ i ] . right_button . is_valid ( ) ) {
2019-06-11 20:43:37 +02:00
Ref < Texture2D > rb = tabs [ i ] . right_button ;
2016-05-01 16:27:33 +02:00
Size2 bms = rb - > get_size ( ) ;
2020-03-12 13:37:40 +01:00
bms . width + = get_theme_constant ( " hseparation " ) ;
2017-03-05 16:44:50 +01:00
ms . width + = bms . width ;
2020-12-08 14:11:45 +01:00
ms . height = MAX ( bms . height + tab_unselected - > get_minimum_size ( ) . height , ms . height ) ;
2015-07-26 15:44:10 +02:00
}
2015-08-18 20:27:01 +02:00
2017-03-05 16:44:50 +01:00
if ( cb_displaypolicy = = CLOSE_BUTTON_SHOW_ALWAYS | | ( cb_displaypolicy = = CLOSE_BUTTON_SHOW_ACTIVE_ONLY & & i = = current ) ) {
2020-03-12 13:37:40 +01:00
Ref < Texture2D > cb = get_theme_icon ( " close " ) ;
2016-05-01 16:27:33 +02:00
Size2 bms = cb - > get_size ( ) ;
2020-03-12 13:37:40 +01:00
bms . width + = get_theme_constant ( " hseparation " ) ;
2017-03-05 16:44:50 +01:00
ms . width + = bms . width ;
2020-12-08 14:11:45 +01:00
ms . height = MAX ( bms . height + tab_unselected - > get_minimum_size ( ) . height , ms . height ) ;
2015-08-18 20:27:01 +02:00
}
2014-02-10 02:10:30 +01:00
}
2017-03-05 16:44:50 +01:00
ms . width = 0 ; //TODO: should make this optional
2014-02-10 02:10:30 +01:00
return ms ;
}
2017-05-20 17:38:03 +02:00
void Tabs : : _gui_input ( const Ref < InputEvent > & p_event ) {
Ref < InputEventMouseMotion > mm = p_event ;
2014-02-10 02:10:30 +01:00
2017-05-20 17:38:03 +02:00
if ( mm . is_valid ( ) ) {
2017-06-03 10:54:24 +02:00
Point2 pos = mm - > get_position ( ) ;
2015-07-26 15:44:10 +02:00
2017-05-02 22:13:12 +02:00
highlight_arrow = - 1 ;
2015-12-14 14:24:28 +01:00
if ( buttons_visible ) {
2020-03-12 13:37:40 +01:00
Ref < Texture2D > incr = get_theme_icon ( " increment " ) ;
Ref < Texture2D > decr = get_theme_icon ( " decrement " ) ;
2015-12-14 14:24:28 +01:00
2020-09-03 13:22:16 +02:00
if ( is_layout_rtl ( ) ) {
if ( pos . x < decr - > get_width ( ) ) {
highlight_arrow = 1 ;
} else if ( pos . x < incr - > get_width ( ) + decr - > get_width ( ) ) {
highlight_arrow = 0 ;
}
} else {
int limit = get_size ( ) . width - incr - > get_width ( ) - decr - > get_width ( ) ;
if ( pos . x > limit + decr - > get_width ( ) ) {
highlight_arrow = 1 ;
} else if ( pos . x > limit ) {
highlight_arrow = 0 ;
}
2015-12-14 14:24:28 +01:00
}
}
2018-10-21 15:57:55 +02:00
_update_hover ( ) ;
2015-08-18 20:27:01 +02:00
update ( ) ;
2015-07-26 15:44:10 +02:00
return ;
}
2017-05-20 17:38:03 +02:00
Ref < InputEventMouseButton > mb = p_event ;
2017-11-16 23:57:57 +01:00
if ( mb . is_valid ( ) ) {
if ( mb - > is_pressed ( ) & & mb - > get_button_index ( ) = = BUTTON_WHEEL_UP & & ! mb - > get_command ( ) ) {
if ( scrolling_enabled & & buttons_visible ) {
if ( offset > 0 ) {
offset - - ;
update ( ) ;
}
}
2015-07-26 15:44:10 +02:00
}
2017-11-16 23:57:57 +01:00
if ( mb - > is_pressed ( ) & & mb - > get_button_index ( ) = = BUTTON_WHEEL_DOWN & & ! mb - > get_command ( ) ) {
if ( scrolling_enabled & & buttons_visible ) {
if ( missing_right ) {
offset + + ;
update ( ) ;
}
}
}
if ( rb_pressing & & ! mb - > is_pressed ( ) & & mb - > get_button_index ( ) = = BUTTON_LEFT ) {
if ( rb_hover ! = - 1 ) {
//pressed
emit_signal ( " right_button_pressed " , rb_hover ) ;
}
2016-05-01 16:27:33 +02:00
2017-11-16 23:57:57 +01:00
rb_pressing = false ;
update ( ) ;
2015-08-18 20:27:01 +02:00
}
2017-11-16 23:57:57 +01:00
if ( cb_pressing & & ! mb - > is_pressed ( ) & & mb - > get_button_index ( ) = = BUTTON_LEFT ) {
if ( cb_hover ! = - 1 ) {
//pressed
2020-12-08 10:51:06 +01:00
emit_signal ( " tab_closed " , cb_hover ) ;
2017-11-16 23:57:57 +01:00
}
2014-02-10 02:10:30 +01:00
2017-11-16 23:57:57 +01:00
cb_pressing = false ;
update ( ) ;
}
2014-02-10 02:10:30 +01:00
2018-01-02 08:10:49 +01:00
if ( mb - > is_pressed ( ) & & ( mb - > get_button_index ( ) = = BUTTON_LEFT | | ( select_with_rmb & & mb - > get_button_index ( ) = = BUTTON_RIGHT ) ) ) {
2017-11-16 23:57:57 +01:00
// clicks
Point2 pos ( mb - > get_position ( ) . x , mb - > get_position ( ) . y ) ;
2015-12-14 14:24:28 +01:00
2017-11-16 23:57:57 +01:00
if ( buttons_visible ) {
2020-03-12 13:37:40 +01:00
Ref < Texture2D > incr = get_theme_icon ( " increment " ) ;
Ref < Texture2D > decr = get_theme_icon ( " decrement " ) ;
2017-11-16 23:57:57 +01:00
2020-09-03 13:22:16 +02:00
if ( is_layout_rtl ( ) ) {
if ( pos . x < decr - > get_width ( ) ) {
if ( missing_right ) {
offset + + ;
update ( ) ;
}
return ;
} else if ( pos . x < incr - > get_width ( ) + decr - > get_width ( ) ) {
if ( offset > 0 ) {
offset - - ;
update ( ) ;
}
return ;
2017-11-16 23:57:57 +01:00
}
2020-09-03 13:22:16 +02:00
} else {
int limit = get_size ( ) . width - incr - > get_width ( ) - decr - > get_width ( ) ;
if ( pos . x > limit + decr - > get_width ( ) ) {
if ( missing_right ) {
offset + + ;
update ( ) ;
}
return ;
} else if ( pos . x > limit ) {
if ( offset > 0 ) {
offset - - ;
update ( ) ;
}
return ;
2017-11-16 23:57:57 +01:00
}
2015-12-14 14:24:28 +01:00
}
}
2017-11-16 23:57:57 +01:00
int found = - 1 ;
2020-04-08 11:14:46 +02:00
for ( int i = offset ; i < tabs . size ( ) ; i + + ) {
2017-11-16 23:57:57 +01:00
if ( tabs [ i ] . rb_rect . has_point ( pos ) ) {
rb_pressing = true ;
update ( ) ;
return ;
}
2015-07-26 15:44:10 +02:00
2017-11-16 23:57:57 +01:00
if ( tabs [ i ] . cb_rect . has_point ( pos ) ) {
cb_pressing = true ;
update ( ) ;
return ;
}
2015-08-18 20:27:01 +02:00
2020-09-03 13:22:16 +02:00
if ( pos . x > = get_tab_rect ( i ) . position . x & & pos . x < get_tab_rect ( i ) . position . x + tabs [ i ] . size_cache ) {
2017-11-16 23:57:57 +01:00
if ( ! tabs [ i ] . disabled ) {
found = i ;
}
break ;
2017-02-27 19:07:50 +01:00
}
2014-02-10 02:10:30 +01:00
}
2017-11-16 23:57:57 +01:00
if ( found ! = - 1 ) {
set_current_tab ( found ) ;
emit_signal ( " tab_clicked " , found ) ;
}
2014-02-10 02:10:30 +01:00
}
}
}
2020-09-03 13:22:16 +02:00
void Tabs : : _shape ( int p_tab ) {
Ref < Font > font = get_theme_font ( " font " ) ;
int font_size = get_theme_font_size ( " font_size " ) ;
tabs . write [ p_tab ] . xl_text = tr ( tabs [ p_tab ] . text ) ;
tabs . write [ p_tab ] . text_buf - > clear ( ) ;
if ( tabs [ p_tab ] . text_direction = = Control : : TEXT_DIRECTION_INHERITED ) {
tabs . write [ p_tab ] . text_buf - > set_direction ( is_layout_rtl ( ) ? TextServer : : DIRECTION_RTL : TextServer : : DIRECTION_LTR ) ;
} else {
tabs . write [ p_tab ] . text_buf - > set_direction ( ( TextServer : : Direction ) tabs [ p_tab ] . text_direction ) ;
}
tabs . write [ p_tab ] . text_buf - > add_string ( tabs . write [ p_tab ] . xl_text , font , font_size , tabs [ p_tab ] . opentype_features , ( tabs [ p_tab ] . language ! = " " ) ? tabs [ p_tab ] . language : TranslationServer : : get_singleton ( ) - > get_tool_locale ( ) ) ;
}
2016-05-01 16:27:33 +02:00
void Tabs : : _notification ( int p_what ) {
2017-03-05 16:44:50 +01:00
switch ( p_what ) {
2020-09-03 13:22:16 +02:00
case NOTIFICATION_LAYOUT_DIRECTION_CHANGED : {
_update_cache ( ) ;
update ( ) ;
} break ;
2019-05-05 18:50:37 +02:00
case NOTIFICATION_TRANSLATION_CHANGED : {
2019-09-30 11:56:20 +02:00
for ( int i = 0 ; i < tabs . size ( ) ; + + i ) {
2020-09-03 13:22:16 +02:00
_shape ( i ) ;
2019-09-30 11:56:20 +02:00
}
2020-09-03 13:22:16 +02:00
_update_cache ( ) ;
2019-05-05 18:50:37 +02:00
minimum_size_changed ( ) ;
update ( ) ;
} break ;
2016-01-23 00:19:57 +01:00
case NOTIFICATION_RESIZED : {
2017-06-15 17:30:03 +02:00
_update_cache ( ) ;
2016-01-23 00:19:57 +01:00
_ensure_no_over_offset ( ) ;
2017-06-15 17:30:03 +02:00
ensure_tab_visible ( current ) ;
2016-01-23 00:19:57 +01:00
} break ;
2014-02-10 02:10:30 +01:00
case NOTIFICATION_DRAW : {
2017-06-15 17:30:03 +02:00
_update_cache ( ) ;
2014-02-10 02:10:30 +01:00
RID ci = get_canvas_item ( ) ;
2020-12-08 14:11:45 +01:00
Ref < StyleBox > tab_unselected = get_theme_stylebox ( " tab_unselected " ) ;
Ref < StyleBox > tab_selected = get_theme_stylebox ( " tab_selected " ) ;
2020-03-12 13:37:40 +01:00
Ref < StyleBox > tab_disabled = get_theme_stylebox ( " tab_disabled " ) ;
2020-12-08 14:11:45 +01:00
Color font_selected_color = get_theme_color ( " font_selected_color " ) ;
Color font_unselected_color = get_theme_color ( " font_unselected_color " ) ;
Color font_disabled_color = get_theme_color ( " font_disabled_color " ) ;
2020-03-12 13:37:40 +01:00
Ref < Texture2D > close = get_theme_icon ( " close " ) ;
2020-12-25 22:45:28 +01:00
Color font_outline_color = get_theme_color ( " font_outline_color " ) ;
int outline_size = get_theme_constant ( " outline_size " ) ;
2020-09-03 13:22:16 +02:00
Vector2 size = get_size ( ) ;
bool rtl = is_layout_rtl ( ) ;
2014-02-10 02:10:30 +01:00
int h = get_size ( ) . height ;
2016-05-01 16:27:33 +02:00
int w = 0 ;
2015-12-14 14:24:28 +01:00
int mw = 0 ;
2017-03-05 16:44:50 +01:00
for ( int i = 0 ; i < tabs . size ( ) ; i + + ) {
2018-07-25 03:11:03 +02:00
tabs . write [ i ] . ofs_cache = mw ;
2016-05-01 16:27:33 +02:00
mw + = get_tab_width ( i ) ;
2015-12-14 14:24:28 +01:00
}
2017-03-05 16:44:50 +01:00
if ( tab_align = = ALIGN_CENTER ) {
w = ( get_size ( ) . width - mw ) / 2 ;
} else if ( tab_align = = ALIGN_RIGHT ) {
w = get_size ( ) . width - mw ;
2015-06-22 05:03:19 +02:00
}
2017-03-05 16:44:50 +01:00
if ( w < 0 ) {
w = 0 ;
2015-06-22 05:03:19 +02:00
}
2020-03-12 13:37:40 +01:00
Ref < Texture2D > incr = get_theme_icon ( " increment " ) ;
Ref < Texture2D > decr = get_theme_icon ( " decrement " ) ;
Ref < Texture2D > incr_hl = get_theme_icon ( " increment_highlight " ) ;
Ref < Texture2D > decr_hl = get_theme_icon ( " decrement_highlight " ) ;
2015-12-14 14:24:28 +01:00
2017-03-05 16:44:50 +01:00
int limit = get_size ( ) . width - incr - > get_size ( ) . width - decr - > get_size ( ) . width ;
2015-12-14 14:24:28 +01:00
2017-03-05 16:44:50 +01:00
missing_right = false ;
2015-12-14 14:24:28 +01:00
2020-04-08 11:14:46 +02:00
for ( int i = offset ; i < tabs . size ( ) ; i + + ) {
2018-07-25 03:11:03 +02:00
tabs . write [ i ] . ofs_cache = w ;
2014-02-10 02:10:30 +01:00
2017-06-15 17:30:03 +02:00
int lsize = tabs [ i ] . size_cache ;
2015-12-14 14:24:28 +01:00
2014-02-10 02:10:30 +01:00
Ref < StyleBox > sb ;
Color col ;
2017-02-27 19:07:50 +01:00
if ( tabs [ i ] . disabled ) {
sb = tab_disabled ;
2020-12-08 14:11:45 +01:00
col = font_disabled_color ;
2017-02-27 19:07:50 +01:00
} else if ( i = = current ) {
2020-12-08 14:11:45 +01:00
sb = tab_selected ;
col = font_selected_color ;
2014-02-10 02:10:30 +01:00
} else {
2020-12-08 14:11:45 +01:00
sb = tab_unselected ;
col = font_unselected_color ;
2014-02-10 02:10:30 +01:00
}
2017-06-15 17:30:03 +02:00
if ( w + lsize > limit ) {
max_drawn_tab = i - 1 ;
missing_right = true ;
break ;
} else {
max_drawn_tab = i ;
}
2020-09-03 13:22:16 +02:00
Rect2 sb_rect ;
if ( rtl ) {
sb_rect = Rect2 ( size . width - w - tabs [ i ] . size_cache , 0 , tabs [ i ] . size_cache , h ) ;
} else {
sb_rect = Rect2 ( w , 0 , tabs [ i ] . size_cache , h ) ;
}
2016-05-01 15:27:58 +02:00
sb - > draw ( ci , sb_rect ) ;
2014-02-10 02:10:30 +01:00
2020-12-22 17:24:29 +01:00
w + = sb - > get_margin ( SIDE_LEFT ) ;
2014-02-10 02:10:30 +01:00
2016-05-01 15:27:58 +02:00
Size2i sb_ms = sb - > get_minimum_size ( ) ;
2019-06-11 20:43:37 +02:00
Ref < Texture2D > icon = tabs [ i ] . icon ;
2014-02-10 02:10:30 +01:00
if ( icon . is_valid ( ) ) {
2020-09-03 13:22:16 +02:00
if ( rtl ) {
2020-12-22 17:24:29 +01:00
icon - > draw ( ci , Point2i ( size . width - w - icon - > get_width ( ) , sb - > get_margin ( SIDE_TOP ) + ( ( sb_rect . size . y - sb_ms . y ) - icon - > get_height ( ) ) / 2 ) ) ;
2020-09-03 13:22:16 +02:00
} else {
2020-12-22 17:24:29 +01:00
icon - > draw ( ci , Point2i ( w , sb - > get_margin ( SIDE_TOP ) + ( ( sb_rect . size . y - sb_ms . y ) - icon - > get_height ( ) ) / 2 ) ) ;
2020-09-03 13:22:16 +02:00
}
2020-05-14 16:41:43 +02:00
if ( tabs [ i ] . text ! = " " ) {
2020-03-12 13:37:40 +01:00
w + = icon - > get_width ( ) + get_theme_constant ( " hseparation " ) ;
2020-05-14 16:41:43 +02:00
}
2014-02-10 02:10:30 +01:00
}
2020-09-03 13:22:16 +02:00
if ( rtl ) {
2020-12-25 22:45:28 +01:00
Vector2 text_pos = Point2i ( size . width - w - tabs [ i ] . text_buf - > get_size ( ) . x , sb - > get_margin ( SIDE_TOP ) + ( ( sb_rect . size . y - sb_ms . y ) - tabs [ i ] . text_buf - > get_size ( ) . y ) / 2 ) ;
if ( outline_size > 0 & & font_outline_color . a > 0 ) {
tabs [ i ] . text_buf - > draw_outline ( ci , text_pos , outline_size , font_outline_color ) ;
}
tabs [ i ] . text_buf - > draw ( ci , text_pos , col ) ;
2020-09-03 13:22:16 +02:00
} else {
2020-12-25 22:45:28 +01:00
Vector2 text_pos = Point2i ( w , sb - > get_margin ( SIDE_TOP ) + ( ( sb_rect . size . y - sb_ms . y ) - tabs [ i ] . text_buf - > get_size ( ) . y ) / 2 ) ;
if ( outline_size > 0 & & font_outline_color . a > 0 ) {
tabs [ i ] . text_buf - > draw_outline ( ci , text_pos , outline_size , font_outline_color ) ;
}
tabs [ i ] . text_buf - > draw ( ci , text_pos , col ) ;
2020-09-03 13:22:16 +02:00
}
2014-02-10 02:10:30 +01:00
2017-06-15 17:30:03 +02:00
w + = tabs [ i ] . size_text ;
2015-07-26 15:44:10 +02:00
if ( tabs [ i ] . right_button . is_valid ( ) ) {
2020-03-12 13:37:40 +01:00
Ref < StyleBox > style = get_theme_stylebox ( " button " ) ;
2019-06-11 20:43:37 +02:00
Ref < Texture2D > rb = tabs [ i ] . right_button ;
2015-07-26 15:44:10 +02:00
2020-03-12 13:37:40 +01:00
w + = get_theme_constant ( " hseparation " ) ;
2015-07-26 15:44:10 +02:00
Rect2 rb_rect ;
2017-03-05 16:44:50 +01:00
rb_rect . size = style - > get_minimum_size ( ) + rb - > get_size ( ) ;
2020-09-03 13:22:16 +02:00
if ( rtl ) {
rb_rect . position . x = size . width - w - rb_rect . size . x ;
} else {
rb_rect . position . x = w ;
}
2020-12-22 17:24:29 +01:00
rb_rect . position . y = sb - > get_margin ( SIDE_TOP ) + ( ( sb_rect . size . y - sb_ms . y ) - ( rb_rect . size . y ) ) / 2 ;
2015-07-26 15:44:10 +02:00
2017-03-05 16:44:50 +01:00
if ( rb_hover = = i ) {
2020-05-14 16:41:43 +02:00
if ( rb_pressing ) {
2020-03-12 13:37:40 +01:00
get_theme_stylebox ( " button_pressed " ) - > draw ( ci , rb_rect ) ;
2020-05-14 16:41:43 +02:00
} else {
2017-03-05 16:44:50 +01:00
style - > draw ( ci , rb_rect ) ;
2020-05-14 16:41:43 +02:00
}
2015-07-26 15:44:10 +02:00
}
2020-09-03 13:22:16 +02:00
if ( rtl ) {
2020-12-22 17:24:29 +01:00
rb - > draw ( ci , Point2i ( size . width - w - rb_rect . size . x + style - > get_margin ( SIDE_LEFT ) , rb_rect . position . y + style - > get_margin ( SIDE_TOP ) ) ) ;
2020-09-03 13:22:16 +02:00
} else {
2020-12-22 17:24:29 +01:00
rb - > draw ( ci , Point2i ( w + style - > get_margin ( SIDE_LEFT ) , rb_rect . position . y + style - > get_margin ( SIDE_TOP ) ) ) ;
2020-09-03 13:22:16 +02:00
}
2017-03-05 16:44:50 +01:00
w + = rb - > get_width ( ) ;
2018-07-25 03:11:03 +02:00
tabs . write [ i ] . rb_rect = rb_rect ;
2015-07-26 15:44:10 +02:00
}
2017-03-05 16:44:50 +01:00
if ( cb_displaypolicy = = CLOSE_BUTTON_SHOW_ALWAYS | | ( cb_displaypolicy = = CLOSE_BUTTON_SHOW_ACTIVE_ONLY & & i = = current ) ) {
2020-03-12 13:37:40 +01:00
Ref < StyleBox > style = get_theme_stylebox ( " button " ) ;
2019-06-11 20:43:37 +02:00
Ref < Texture2D > cb = close ;
2015-08-18 20:27:01 +02:00
2020-03-12 13:37:40 +01:00
w + = get_theme_constant ( " hseparation " ) ;
2015-08-18 20:27:01 +02:00
2016-01-13 11:39:31 +01:00
Rect2 cb_rect ;
2017-03-05 16:44:50 +01:00
cb_rect . size = style - > get_minimum_size ( ) + cb - > get_size ( ) ;
2020-09-03 13:22:16 +02:00
if ( rtl ) {
cb_rect . position . x = size . width - w - cb_rect . size . x ;
} else {
cb_rect . position . x = w ;
}
2020-12-22 17:24:29 +01:00
cb_rect . position . y = sb - > get_margin ( SIDE_TOP ) + ( ( sb_rect . size . y - sb_ms . y ) - ( cb_rect . size . y ) ) / 2 ;
2015-08-18 20:27:01 +02:00
2017-02-27 19:07:50 +01:00
if ( ! tabs [ i ] . disabled & & cb_hover = = i ) {
2020-05-14 16:41:43 +02:00
if ( cb_pressing ) {
2020-03-12 13:37:40 +01:00
get_theme_stylebox ( " button_pressed " ) - > draw ( ci , cb_rect ) ;
2020-05-14 16:41:43 +02:00
} else {
2017-03-05 16:44:50 +01:00
style - > draw ( ci , cb_rect ) ;
2020-05-14 16:41:43 +02:00
}
2015-08-18 20:27:01 +02:00
}
2020-09-03 13:22:16 +02:00
if ( rtl ) {
2020-12-22 17:24:29 +01:00
cb - > draw ( ci , Point2i ( size . width - w - cb_rect . size . x + style - > get_margin ( SIDE_LEFT ) , cb_rect . position . y + style - > get_margin ( SIDE_TOP ) ) ) ;
2020-09-03 13:22:16 +02:00
} else {
2020-12-22 17:24:29 +01:00
cb - > draw ( ci , Point2i ( w + style - > get_margin ( SIDE_LEFT ) , cb_rect . position . y + style - > get_margin ( SIDE_TOP ) ) ) ;
2020-09-03 13:22:16 +02:00
}
2017-03-05 16:44:50 +01:00
w + = cb - > get_width ( ) ;
2018-07-25 03:11:03 +02:00
tabs . write [ i ] . cb_rect = cb_rect ;
2015-08-18 20:27:01 +02:00
}
2020-12-22 17:24:29 +01:00
w + = sb - > get_margin ( SIDE_RIGHT ) ;
2014-02-10 02:10:30 +01:00
}
2017-03-05 16:44:50 +01:00
if ( offset > 0 | | missing_right ) {
int vofs = ( get_size ( ) . height - incr - > get_size ( ) . height ) / 2 ;
2015-12-14 14:24:28 +01:00
2020-09-03 13:22:16 +02:00
if ( rtl ) {
if ( missing_right ) {
draw_texture ( highlight_arrow = = 1 ? decr_hl : decr , Point2 ( 0 , vofs ) ) ;
} else {
draw_texture ( decr , Point2 ( 0 , vofs ) , Color ( 1 , 1 , 1 , 0.5 ) ) ;
}
2015-12-14 14:24:28 +01:00
2020-09-03 13:22:16 +02:00
if ( offset > 0 ) {
draw_texture ( highlight_arrow = = 0 ? incr_hl : incr , Point2 ( incr - > get_size ( ) . width , vofs ) ) ;
} else {
draw_texture ( incr , Point2 ( incr - > get_size ( ) . width , vofs ) , Color ( 1 , 1 , 1 , 0.5 ) ) ;
}
2020-05-14 16:41:43 +02:00
} else {
2020-09-03 13:22:16 +02:00
if ( offset > 0 ) {
draw_texture ( highlight_arrow = = 0 ? decr_hl : decr , Point2 ( limit , vofs ) ) ;
} else {
draw_texture ( decr , Point2 ( limit , vofs ) , Color ( 1 , 1 , 1 , 0.5 ) ) ;
}
if ( missing_right ) {
draw_texture ( highlight_arrow = = 1 ? incr_hl : incr , Point2 ( limit + decr - > get_size ( ) . width , vofs ) ) ;
} else {
draw_texture ( incr , Point2 ( limit + decr - > get_size ( ) . width , vofs ) , Color ( 1 , 1 , 1 , 0.5 ) ) ;
}
2020-05-14 16:41:43 +02:00
}
2015-12-14 14:24:28 +01:00
2017-03-05 16:44:50 +01:00
buttons_visible = true ;
2015-12-14 14:24:28 +01:00
} else {
2017-03-05 16:44:50 +01:00
buttons_visible = false ;
2015-12-14 14:24:28 +01:00
}
2014-02-10 02:10:30 +01:00
} break ;
}
}
int Tabs : : get_tab_count ( ) const {
return tabs . size ( ) ;
}
void Tabs : : set_current_tab ( int p_current ) {
2020-05-14 16:41:43 +02:00
if ( current = = p_current ) {
2020-05-10 12:56:01 +02:00
return ;
2020-05-14 16:41:43 +02:00
}
2017-03-05 16:44:50 +01:00
ERR_FAIL_INDEX ( p_current , get_tab_count ( ) ) ;
2014-02-10 02:10:30 +01:00
2020-09-08 22:30:47 +02:00
previous = current ;
2017-03-05 16:44:50 +01:00
current = p_current ;
2014-02-10 02:10:30 +01:00
2017-06-15 17:30:03 +02:00
_update_cache ( ) ;
2014-02-10 02:10:30 +01:00
update ( ) ;
2017-08-05 17:56:00 +02:00
emit_signal ( " tab_changed " , p_current ) ;
2014-02-10 02:10:30 +01:00
}
int Tabs : : get_current_tab ( ) const {
return current ;
}
2020-09-08 22:30:47 +02:00
int Tabs : : get_previous_tab ( ) const {
return previous ;
}
2017-06-15 17:30:03 +02:00
int Tabs : : get_hovered_tab ( ) const {
return hover ;
}
2017-11-16 23:57:57 +01:00
int Tabs : : get_tab_offset ( ) const {
return offset ;
}
bool Tabs : : get_offset_buttons_visible ( ) const {
return buttons_visible ;
}
2017-03-05 16:44:50 +01:00
void Tabs : : set_tab_title ( int p_tab , const String & p_title ) {
ERR_FAIL_INDEX ( p_tab , tabs . size ( ) ) ;
2018-07-25 03:11:03 +02:00
tabs . write [ p_tab ] . text = p_title ;
2019-09-30 11:56:20 +02:00
tabs . write [ p_tab ] . xl_text = tr ( p_title ) ;
2020-09-03 13:22:16 +02:00
_shape ( p_tab ) ;
2014-02-10 02:10:30 +01:00
update ( ) ;
minimum_size_changed ( ) ;
}
2017-03-05 16:44:50 +01:00
String Tabs : : get_tab_title ( int p_tab ) const {
ERR_FAIL_INDEX_V ( p_tab , tabs . size ( ) , " " ) ;
2014-02-10 02:10:30 +01:00
return tabs [ p_tab ] . text ;
}
2020-09-03 13:22:16 +02:00
void Tabs : : set_tab_text_direction ( int p_tab , Control : : TextDirection p_text_direction ) {
ERR_FAIL_INDEX ( p_tab , tabs . size ( ) ) ;
ERR_FAIL_COND ( ( int ) p_text_direction < - 1 | | ( int ) p_text_direction > 3 ) ;
if ( tabs [ p_tab ] . text_direction ! = p_text_direction ) {
tabs . write [ p_tab ] . text_direction = p_text_direction ;
_shape ( p_tab ) ;
update ( ) ;
}
}
Control : : TextDirection Tabs : : get_tab_text_direction ( int p_tab ) const {
ERR_FAIL_INDEX_V ( p_tab , tabs . size ( ) , Control : : TEXT_DIRECTION_INHERITED ) ;
return tabs [ p_tab ] . text_direction ;
}
void Tabs : : clear_tab_opentype_features ( int p_tab ) {
ERR_FAIL_INDEX ( p_tab , tabs . size ( ) ) ;
tabs . write [ p_tab ] . opentype_features . clear ( ) ;
_shape ( p_tab ) ;
update ( ) ;
}
void Tabs : : set_tab_opentype_feature ( int p_tab , const String & p_name , int p_value ) {
ERR_FAIL_INDEX ( p_tab , tabs . size ( ) ) ;
int32_t tag = TS - > name_to_tag ( p_name ) ;
if ( ! tabs [ p_tab ] . opentype_features . has ( tag ) | | ( int ) tabs [ p_tab ] . opentype_features [ tag ] ! = p_value ) {
tabs . write [ p_tab ] . opentype_features [ tag ] = p_value ;
_shape ( p_tab ) ;
update ( ) ;
}
}
int Tabs : : get_tab_opentype_feature ( int p_tab , const String & p_name ) const {
ERR_FAIL_INDEX_V ( p_tab , tabs . size ( ) , - 1 ) ;
int32_t tag = TS - > name_to_tag ( p_name ) ;
if ( ! tabs [ p_tab ] . opentype_features . has ( tag ) ) {
return - 1 ;
}
return tabs [ p_tab ] . opentype_features [ tag ] ;
}
void Tabs : : set_tab_language ( int p_tab , const String & p_language ) {
ERR_FAIL_INDEX ( p_tab , tabs . size ( ) ) ;
if ( tabs [ p_tab ] . language ! = p_language ) {
tabs . write [ p_tab ] . language = p_language ;
_shape ( p_tab ) ;
update ( ) ;
}
}
String Tabs : : get_tab_language ( int p_tab ) const {
ERR_FAIL_INDEX_V ( p_tab , tabs . size ( ) , " " ) ;
return tabs [ p_tab ] . language ;
}
2019-06-11 20:43:37 +02:00
void Tabs : : set_tab_icon ( int p_tab , const Ref < Texture2D > & p_icon ) {
2017-03-05 16:44:50 +01:00
ERR_FAIL_INDEX ( p_tab , tabs . size ( ) ) ;
2018-07-25 03:11:03 +02:00
tabs . write [ p_tab ] . icon = p_icon ;
2014-02-10 02:10:30 +01:00
update ( ) ;
minimum_size_changed ( ) ;
}
2016-05-01 16:27:33 +02:00
2019-06-11 20:43:37 +02:00
Ref < Texture2D > Tabs : : get_tab_icon ( int p_tab ) const {
ERR_FAIL_INDEX_V ( p_tab , tabs . size ( ) , Ref < Texture2D > ( ) ) ;
2014-02-10 02:10:30 +01:00
return tabs [ p_tab ] . icon ;
}
2017-02-27 19:07:50 +01:00
void Tabs : : set_tab_disabled ( int p_tab , bool p_disabled ) {
ERR_FAIL_INDEX ( p_tab , tabs . size ( ) ) ;
2018-07-25 03:11:03 +02:00
tabs . write [ p_tab ] . disabled = p_disabled ;
2017-02-27 19:07:50 +01:00
update ( ) ;
}
2020-05-14 14:29:06 +02:00
2017-02-27 19:07:50 +01:00
bool Tabs : : get_tab_disabled ( int p_tab ) const {
ERR_FAIL_INDEX_V ( p_tab , tabs . size ( ) , false ) ;
return tabs [ p_tab ] . disabled ;
}
2019-06-11 20:43:37 +02:00
void Tabs : : set_tab_right_button ( int p_tab , const Ref < Texture2D > & p_right_button ) {
2017-03-05 16:44:50 +01:00
ERR_FAIL_INDEX ( p_tab , tabs . size ( ) ) ;
2018-07-25 03:11:03 +02:00
tabs . write [ p_tab ] . right_button = p_right_button ;
2017-11-16 23:57:57 +01:00
_update_cache ( ) ;
2015-07-26 15:44:10 +02:00
update ( ) ;
minimum_size_changed ( ) ;
}
2020-05-14 14:29:06 +02:00
2019-06-11 20:43:37 +02:00
Ref < Texture2D > Tabs : : get_tab_right_button ( int p_tab ) const {
ERR_FAIL_INDEX_V ( p_tab , tabs . size ( ) , Ref < Texture2D > ( ) ) ;
2015-07-26 15:44:10 +02:00
return tabs [ p_tab ] . right_button ;
}
2018-10-21 15:57:55 +02:00
void Tabs : : _update_hover ( ) {
if ( ! is_inside_tree ( ) ) {
return ;
}
const Point2 & pos = get_local_mouse_position ( ) ;
// test hovering to display right or close button
int hover_now = - 1 ;
int hover_buttons = - 1 ;
2020-04-08 11:14:46 +02:00
for ( int i = offset ; i < tabs . size ( ) ; i + + ) {
2018-10-21 15:57:55 +02:00
Rect2 rect = get_tab_rect ( i ) ;
if ( rect . has_point ( pos ) ) {
hover_now = i ;
}
if ( tabs [ i ] . rb_rect . has_point ( pos ) ) {
rb_hover = i ;
cb_hover = - 1 ;
hover_buttons = i ;
break ;
} else if ( ! tabs [ i ] . disabled & & tabs [ i ] . cb_rect . has_point ( pos ) ) {
cb_hover = i ;
rb_hover = - 1 ;
hover_buttons = i ;
break ;
}
}
if ( hover ! = hover_now ) {
hover = hover_now ;
2020-12-08 10:51:06 +01:00
emit_signal ( " tab_hovered " , hover ) ;
2018-10-21 15:57:55 +02:00
}
if ( hover_buttons = = - 1 ) { // no hover
rb_hover = hover_buttons ;
cb_hover = hover_buttons ;
}
}
2017-06-15 17:30:03 +02:00
void Tabs : : _update_cache ( ) {
2020-03-12 13:37:40 +01:00
Ref < StyleBox > tab_disabled = get_theme_stylebox ( " tab_disabled " ) ;
2020-12-08 14:11:45 +01:00
Ref < StyleBox > tab_unselected = get_theme_stylebox ( " tab_unselected " ) ;
Ref < StyleBox > tab_selected = get_theme_stylebox ( " tab_selected " ) ;
2020-03-12 13:37:40 +01:00
Ref < Texture2D > incr = get_theme_icon ( " increment " ) ;
Ref < Texture2D > decr = get_theme_icon ( " decrement " ) ;
2017-06-15 17:30:03 +02:00
int limit = get_size ( ) . width - incr - > get_width ( ) - decr - > get_width ( ) ;
int w = 0 ;
int mw = 0 ;
int size_fixed = 0 ;
int count_resize = 0 ;
for ( int i = 0 ; i < tabs . size ( ) ; i + + ) {
2018-07-25 03:11:03 +02:00
tabs . write [ i ] . ofs_cache = mw ;
tabs . write [ i ] . size_cache = get_tab_width ( i ) ;
2020-09-03 13:22:16 +02:00
tabs . write [ i ] . size_text = Math : : ceil ( tabs [ i ] . text_buf - > get_size ( ) . x ) ;
tabs . write [ i ] . text_buf - > set_width ( - 1 ) ;
2017-06-15 17:30:03 +02:00
mw + = tabs [ i ] . size_cache ;
if ( tabs [ i ] . size_cache < = min_width | | i = = current ) {
size_fixed + = tabs [ i ] . size_cache ;
} else {
count_resize + + ;
}
}
int m_width = min_width ;
if ( count_resize > 0 ) {
m_width = MAX ( ( limit - size_fixed ) / count_resize , min_width ) ;
}
2020-04-08 11:14:46 +02:00
for ( int i = offset ; i < tabs . size ( ) ; i + + ) {
2017-06-15 17:30:03 +02:00
Ref < StyleBox > sb ;
if ( tabs [ i ] . disabled ) {
sb = tab_disabled ;
} else if ( i = = current ) {
2020-12-08 14:11:45 +01:00
sb = tab_selected ;
2017-06-15 17:30:03 +02:00
} else {
2020-12-08 14:11:45 +01:00
sb = tab_unselected ;
2017-06-15 17:30:03 +02:00
}
int lsize = tabs [ i ] . size_cache ;
int slen = tabs [ i ] . size_text ;
if ( min_width > 0 & & mw > limit & & i ! = current ) {
if ( lsize > m_width ) {
2020-12-22 17:24:29 +01:00
slen = m_width - ( sb - > get_margin ( SIDE_LEFT ) + sb - > get_margin ( SIDE_RIGHT ) ) ;
2017-06-15 17:30:03 +02:00
if ( tabs [ i ] . icon . is_valid ( ) ) {
slen - = tabs [ i ] . icon - > get_width ( ) ;
2020-03-12 13:37:40 +01:00
slen - = get_theme_constant ( " hseparation " ) ;
2017-06-15 17:30:03 +02:00
}
if ( cb_displaypolicy = = CLOSE_BUTTON_SHOW_ALWAYS | | ( cb_displaypolicy = = CLOSE_BUTTON_SHOW_ACTIVE_ONLY & & i = = current ) ) {
2020-03-12 13:37:40 +01:00
Ref < Texture2D > cb = get_theme_icon ( " close " ) ;
2017-06-15 17:30:03 +02:00
slen - = cb - > get_width ( ) ;
2020-03-12 13:37:40 +01:00
slen - = get_theme_constant ( " hseparation " ) ;
2017-06-15 17:30:03 +02:00
}
slen = MAX ( slen , 1 ) ;
lsize = m_width ;
}
}
2018-07-25 03:11:03 +02:00
tabs . write [ i ] . ofs_cache = w ;
tabs . write [ i ] . size_cache = lsize ;
tabs . write [ i ] . size_text = slen ;
2020-09-03 13:22:16 +02:00
tabs . write [ i ] . text_buf - > set_width ( slen ) ;
2017-06-15 17:30:03 +02:00
w + = lsize ;
}
}
2019-09-23 04:07:00 +02:00
void Tabs : : _on_mouse_exited ( ) {
rb_hover = - 1 ;
cb_hover = - 1 ;
hover = - 1 ;
highlight_arrow = - 1 ;
update ( ) ;
}
2019-06-11 20:43:37 +02:00
void Tabs : : add_tab ( const String & p_str , const Ref < Texture2D > & p_icon ) {
2014-02-10 02:10:30 +01:00
Tab t ;
2017-03-05 16:44:50 +01:00
t . text = p_str ;
2019-09-30 11:56:20 +02:00
t . xl_text = tr ( p_str ) ;
2020-09-03 13:22:16 +02:00
t . text_buf . instance ( ) ;
t . text_buf - > set_direction ( is_layout_rtl ( ) ? TextServer : : DIRECTION_RTL : TextServer : : DIRECTION_LTR ) ;
t . text_buf - > add_string ( t . xl_text , get_theme_font ( " font " ) , get_theme_font_size ( " font_size " ) , Dictionary ( ) , TranslationServer : : get_singleton ( ) - > get_tool_locale ( ) ) ;
2017-03-05 16:44:50 +01:00
t . icon = p_icon ;
2017-02-27 19:07:50 +01:00
t . disabled = false ;
2017-06-15 17:30:03 +02:00
t . ofs_cache = 0 ;
t . size_cache = 0 ;
2015-08-18 20:27:01 +02:00
2014-02-10 02:10:30 +01:00
tabs . push_back ( t ) ;
2017-06-15 17:30:03 +02:00
_update_cache ( ) ;
2018-10-21 15:57:55 +02:00
call_deferred ( " _update_hover " ) ;
2014-02-10 02:10:30 +01:00
update ( ) ;
minimum_size_changed ( ) ;
}
2015-06-22 05:03:19 +02:00
void Tabs : : clear_tabs ( ) {
tabs . clear ( ) ;
2017-03-05 16:44:50 +01:00
current = 0 ;
2020-09-08 22:30:47 +02:00
previous = 0 ;
2018-10-21 15:57:55 +02:00
call_deferred ( " _update_hover " ) ;
2015-06-22 05:03:19 +02:00
update ( ) ;
}
2014-02-10 02:10:30 +01:00
void Tabs : : remove_tab ( int p_idx ) {
2017-03-05 16:44:50 +01:00
ERR_FAIL_INDEX ( p_idx , tabs . size ( ) ) ;
2014-02-10 02:10:30 +01:00
tabs . remove ( p_idx ) ;
2020-05-14 16:41:43 +02:00
if ( current > = p_idx ) {
2014-02-10 02:10:30 +01:00
current - - ;
2020-05-14 16:41:43 +02:00
}
2017-06-15 17:30:03 +02:00
_update_cache ( ) ;
2018-10-21 15:57:55 +02:00
call_deferred ( " _update_hover " ) ;
2014-02-10 02:10:30 +01:00
update ( ) ;
minimum_size_changed ( ) ;
2020-05-14 16:41:43 +02:00
if ( current < 0 ) {
2017-03-05 16:44:50 +01:00
current = 0 ;
2020-09-08 22:30:47 +02:00
previous = 0 ;
2020-05-14 16:41:43 +02:00
}
if ( current > = tabs . size ( ) ) {
2017-03-05 16:44:50 +01:00
current = tabs . size ( ) - 1 ;
2020-05-14 16:41:43 +02:00
}
2014-02-10 02:10:30 +01:00
2016-01-23 00:19:57 +01:00
_ensure_no_over_offset ( ) ;
2015-06-22 05:03:19 +02:00
}
2017-07-05 15:44:53 +02:00
Variant Tabs : : get_drag_data ( const Point2 & p_point ) {
2020-05-14 16:41:43 +02:00
if ( ! drag_to_rearrange_enabled ) {
2018-02-07 14:01:45 +01:00
return Variant ( ) ;
2020-05-14 16:41:43 +02:00
}
2018-02-07 14:01:45 +01:00
int tab_over = get_tab_idx_at_point ( p_point ) ;
2020-05-14 16:41:43 +02:00
if ( tab_over < 0 ) {
2018-02-07 14:01:45 +01:00
return Variant ( ) ;
2020-05-14 16:41:43 +02:00
}
2018-02-07 14:01:45 +01:00
HBoxContainer * drag_preview = memnew ( HBoxContainer ) ;
if ( ! tabs [ tab_over ] . icon . is_null ( ) ) {
TextureRect * tf = memnew ( TextureRect ) ;
tf - > set_texture ( tabs [ tab_over ] . icon ) ;
drag_preview - > add_child ( tf ) ;
}
2019-09-30 11:56:20 +02:00
Label * label = memnew ( Label ( tabs [ tab_over ] . xl_text ) ) ;
2018-02-07 14:01:45 +01:00
drag_preview - > add_child ( label ) ;
if ( ! tabs [ tab_over ] . right_button . is_null ( ) ) {
TextureRect * tf = memnew ( TextureRect ) ;
tf - > set_texture ( tabs [ tab_over ] . right_button ) ;
drag_preview - > add_child ( tf ) ;
}
set_drag_preview ( drag_preview ) ;
Dictionary drag_data ;
drag_data [ " type " ] = " tab_element " ;
drag_data [ " tab_element " ] = tab_over ;
drag_data [ " from_path " ] = get_path ( ) ;
return drag_data ;
2017-07-05 15:44:53 +02:00
}
bool Tabs : : can_drop_data ( const Point2 & p_point , const Variant & p_data ) const {
2020-05-14 16:41:43 +02:00
if ( ! drag_to_rearrange_enabled ) {
2018-02-07 14:01:45 +01:00
return false ;
2020-05-14 16:41:43 +02:00
}
2018-02-07 14:01:45 +01:00
Dictionary d = p_data ;
2020-05-14 16:41:43 +02:00
if ( ! d . has ( " type " ) ) {
2018-02-07 14:01:45 +01:00
return false ;
2020-05-14 16:41:43 +02:00
}
2018-02-07 14:01:45 +01:00
if ( String ( d [ " type " ] ) = = " tab_element " ) {
NodePath from_path = d [ " from_path " ] ;
NodePath to_path = get_path ( ) ;
if ( from_path = = to_path ) {
return true ;
} else if ( get_tabs_rearrange_group ( ) ! = - 1 ) {
// drag and drop between other Tabs
Node * from_node = get_node ( from_path ) ;
Tabs * from_tabs = Object : : cast_to < Tabs > ( from_node ) ;
if ( from_tabs & & from_tabs - > get_tabs_rearrange_group ( ) = = get_tabs_rearrange_group ( ) ) {
return true ;
}
}
}
return false ;
2017-07-05 15:44:53 +02:00
}
void Tabs : : drop_data ( const Point2 & p_point , const Variant & p_data ) {
2020-05-14 16:41:43 +02:00
if ( ! drag_to_rearrange_enabled ) {
2018-02-07 14:01:45 +01:00
return ;
2020-05-14 16:41:43 +02:00
}
2018-02-07 14:01:45 +01:00
2017-07-05 15:44:53 +02:00
int hover_now = get_tab_idx_at_point ( p_point ) ;
2018-02-07 14:01:45 +01:00
Dictionary d = p_data ;
2020-05-14 16:41:43 +02:00
if ( ! d . has ( " type " ) ) {
2018-02-07 14:01:45 +01:00
return ;
2020-05-14 16:41:43 +02:00
}
2018-02-07 14:01:45 +01:00
if ( String ( d [ " type " ] ) = = " tab_element " ) {
int tab_from_id = d [ " tab_element " ] ;
NodePath from_path = d [ " from_path " ] ;
NodePath to_path = get_path ( ) ;
if ( from_path = = to_path ) {
2020-05-14 16:41:43 +02:00
if ( hover_now < 0 ) {
2018-02-07 14:01:45 +01:00
hover_now = get_tab_count ( ) - 1 ;
2020-05-14 16:41:43 +02:00
}
2018-02-07 14:01:45 +01:00
move_tab ( tab_from_id , hover_now ) ;
emit_signal ( " reposition_active_tab_request " , hover_now ) ;
set_current_tab ( hover_now ) ;
} else if ( get_tabs_rearrange_group ( ) ! = - 1 ) {
// drag and drop between Tabs
Node * from_node = get_node ( from_path ) ;
Tabs * from_tabs = Object : : cast_to < Tabs > ( from_node ) ;
if ( from_tabs & & from_tabs - > get_tabs_rearrange_group ( ) = = get_tabs_rearrange_group ( ) ) {
2020-05-14 16:41:43 +02:00
if ( tab_from_id > = from_tabs - > get_tab_count ( ) ) {
2018-02-07 14:01:45 +01:00
return ;
2020-05-14 16:41:43 +02:00
}
2018-02-07 14:01:45 +01:00
Tab moving_tab = from_tabs - > tabs [ tab_from_id ] ;
2020-05-14 16:41:43 +02:00
if ( hover_now < 0 ) {
2018-02-07 14:01:45 +01:00
hover_now = get_tab_count ( ) ;
2020-05-14 16:41:43 +02:00
}
2018-02-07 14:01:45 +01:00
tabs . insert ( hover_now , moving_tab ) ;
from_tabs - > remove_tab ( tab_from_id ) ;
set_current_tab ( hover_now ) ;
emit_signal ( " tab_changed " , hover_now ) ;
_update_cache ( ) ;
}
}
}
update ( ) ;
2017-07-05 15:44:53 +02:00
}
int Tabs : : get_tab_idx_at_point ( const Point2 & p_point ) const {
int hover_now = - 1 ;
2020-04-08 11:14:46 +02:00
for ( int i = offset ; i < tabs . size ( ) ; i + + ) {
2017-07-05 15:44:53 +02:00
Rect2 rect = get_tab_rect ( i ) ;
if ( rect . has_point ( p_point ) ) {
hover_now = i ;
}
}
return hover_now ;
}
2015-06-22 05:03:19 +02:00
void Tabs : : set_tab_align ( TabAlign p_align ) {
2017-07-22 18:15:31 +02:00
ERR_FAIL_INDEX ( p_align , ALIGN_MAX ) ;
2017-03-05 16:44:50 +01:00
tab_align = p_align ;
2015-06-22 05:03:19 +02:00
update ( ) ;
}
Tabs : : TabAlign Tabs : : get_tab_align ( ) const {
return tab_align ;
2014-02-10 02:10:30 +01:00
}
2017-07-12 16:48:43 +02:00
void Tabs : : move_tab ( int from , int to ) {
2020-05-14 16:41:43 +02:00
if ( from = = to ) {
2017-07-12 16:48:43 +02:00
return ;
2020-05-14 16:41:43 +02:00
}
2017-07-12 16:48:43 +02:00
ERR_FAIL_INDEX ( from , tabs . size ( ) ) ;
ERR_FAIL_INDEX ( to , tabs . size ( ) ) ;
Tab tab_from = tabs [ from ] ;
tabs . remove ( from ) ;
tabs . insert ( to , tab_from ) ;
_update_cache ( ) ;
update ( ) ;
}
2016-01-13 11:39:31 +01:00
int Tabs : : get_tab_width ( int p_idx ) const {
2017-03-05 16:44:50 +01:00
ERR_FAIL_INDEX_V ( p_idx , tabs . size ( ) , 0 ) ;
2016-01-13 11:39:31 +01:00
2020-12-08 14:11:45 +01:00
Ref < StyleBox > tab_unselected = get_theme_stylebox ( " tab_unselected " ) ;
Ref < StyleBox > tab_selected = get_theme_stylebox ( " tab_selected " ) ;
2020-03-12 13:37:40 +01:00
Ref < StyleBox > tab_disabled = get_theme_stylebox ( " tab_disabled " ) ;
2016-05-01 16:27:33 +02:00
2017-03-05 16:44:50 +01:00
int x = 0 ;
2016-01-13 11:39:31 +01:00
2019-06-11 20:43:37 +02:00
Ref < Texture2D > tex = tabs [ p_idx ] . icon ;
2016-01-13 11:39:31 +01:00
if ( tex . is_valid ( ) ) {
2017-03-05 16:44:50 +01:00
x + = tex - > get_width ( ) ;
2020-05-14 16:41:43 +02:00
if ( tabs [ p_idx ] . text ! = " " ) {
2020-03-12 13:37:40 +01:00
x + = get_theme_constant ( " hseparation " ) ;
2020-05-14 16:41:43 +02:00
}
2016-01-13 11:39:31 +01:00
}
2020-09-03 13:22:16 +02:00
x + = Math : : ceil ( tabs [ p_idx ] . text_buf - > get_size ( ) . x ) ;
2016-05-01 16:27:33 +02:00
2020-05-14 16:41:43 +02:00
if ( tabs [ p_idx ] . disabled ) {
2017-02-27 19:07:50 +01:00
x + = tab_disabled - > get_minimum_size ( ) . width ;
2020-05-14 16:41:43 +02:00
} else if ( current = = p_idx ) {
2020-12-08 14:11:45 +01:00
x + = tab_selected - > get_minimum_size ( ) . width ;
2020-05-14 16:41:43 +02:00
} else {
2020-12-08 14:11:45 +01:00
x + = tab_unselected - > get_minimum_size ( ) . width ;
2020-05-14 16:41:43 +02:00
}
2016-01-13 11:39:31 +01:00
if ( tabs [ p_idx ] . right_button . is_valid ( ) ) {
2019-06-11 20:43:37 +02:00
Ref < Texture2D > rb = tabs [ p_idx ] . right_button ;
2017-03-05 16:44:50 +01:00
x + = rb - > get_width ( ) ;
2020-03-12 13:37:40 +01:00
x + = get_theme_constant ( " hseparation " ) ;
2016-01-13 11:39:31 +01:00
}
2017-03-05 16:44:50 +01:00
if ( cb_displaypolicy = = CLOSE_BUTTON_SHOW_ALWAYS | | ( cb_displaypolicy = = CLOSE_BUTTON_SHOW_ACTIVE_ONLY & & p_idx = = current ) ) {
2020-03-12 13:37:40 +01:00
Ref < Texture2D > cb = get_theme_icon ( " close " ) ;
2017-03-05 16:44:50 +01:00
x + = cb - > get_width ( ) ;
2020-03-12 13:37:40 +01:00
x + = get_theme_constant ( " hseparation " ) ;
2016-01-13 11:39:31 +01:00
}
return x ;
}
2014-02-10 02:10:30 +01:00
2016-01-23 00:19:57 +01:00
void Tabs : : _ensure_no_over_offset ( ) {
2020-05-14 16:41:43 +02:00
if ( ! is_inside_tree ( ) ) {
2016-01-23 00:19:57 +01:00
return ;
2020-05-14 16:41:43 +02:00
}
2016-01-23 00:19:57 +01:00
2020-03-12 13:37:40 +01:00
Ref < Texture2D > incr = get_theme_icon ( " increment " ) ;
Ref < Texture2D > decr = get_theme_icon ( " decrement " ) ;
2016-01-23 00:19:57 +01:00
2017-03-05 16:44:50 +01:00
int limit = get_size ( ) . width - incr - > get_width ( ) - decr - > get_width ( ) ;
2016-01-23 00:19:57 +01:00
2017-03-05 16:44:50 +01:00
while ( offset > 0 ) {
int total_w = 0 ;
2020-04-08 11:14:46 +02:00
for ( int i = offset - 1 ; i < tabs . size ( ) ; i + + ) {
2017-06-15 17:30:03 +02:00
total_w + = tabs [ i ] . size_cache ;
2016-01-23 00:19:57 +01:00
}
if ( total_w < limit ) {
offset - - ;
update ( ) ;
} else {
break ;
}
}
}
2016-01-11 01:45:11 +01:00
void Tabs : : ensure_tab_visible ( int p_idx ) {
2020-05-14 16:41:43 +02:00
if ( ! is_inside_tree ( ) ) {
2016-01-11 01:45:11 +01:00
return ;
2020-05-14 16:41:43 +02:00
}
2016-01-11 01:45:11 +01:00
2020-05-14 16:41:43 +02:00
if ( tabs . size ( ) = = 0 ) {
2020-05-10 12:56:01 +02:00
return ;
2020-05-14 16:41:43 +02:00
}
2017-03-05 16:44:50 +01:00
ERR_FAIL_INDEX ( p_idx , tabs . size ( ) ) ;
2016-01-11 01:45:11 +01:00
2017-06-15 17:30:03 +02:00
if ( p_idx = = offset ) {
return ;
}
if ( p_idx < offset ) {
2017-03-05 16:44:50 +01:00
offset = p_idx ;
2016-01-11 01:45:11 +01:00
update ( ) ;
return ;
}
2017-06-15 17:30:03 +02:00
int prev_offset = offset ;
2020-03-12 13:37:40 +01:00
Ref < Texture2D > incr = get_theme_icon ( " increment " ) ;
Ref < Texture2D > decr = get_theme_icon ( " decrement " ) ;
2017-03-05 16:44:50 +01:00
int limit = get_size ( ) . width - incr - > get_width ( ) - decr - > get_width ( ) ;
2017-06-15 17:30:03 +02:00
for ( int i = offset ; i < = p_idx ; i + + ) {
if ( tabs [ i ] . ofs_cache + tabs [ i ] . size_cache > limit ) {
offset + + ;
}
2016-01-11 01:45:11 +01:00
}
2017-06-15 17:30:03 +02:00
if ( prev_offset ! = offset ) {
update ( ) ;
2016-01-11 01:45:11 +01:00
}
2017-06-15 17:30:03 +02:00
}
2016-01-11 01:45:11 +01:00
2017-07-05 15:44:53 +02:00
Rect2 Tabs : : get_tab_rect ( int p_tab ) const {
2019-06-21 11:34:32 +02:00
ERR_FAIL_INDEX_V ( p_tab , tabs . size ( ) , Rect2 ( ) ) ;
2020-09-03 13:22:16 +02:00
if ( is_layout_rtl ( ) ) {
return Rect2 ( get_size ( ) . width - tabs [ p_tab ] . ofs_cache - tabs [ p_tab ] . size_cache , 0 , tabs [ p_tab ] . size_cache , get_size ( ) . height ) ;
} else {
return Rect2 ( tabs [ p_tab ] . ofs_cache , 0 , tabs [ p_tab ] . size_cache , get_size ( ) . height ) ;
}
2016-01-11 01:45:11 +01:00
}
2016-01-13 11:39:31 +01:00
void Tabs : : set_tab_close_display_policy ( CloseButtonDisplayPolicy p_policy ) {
2017-07-22 18:15:31 +02:00
ERR_FAIL_INDEX ( p_policy , CLOSE_BUTTON_MAX ) ;
2017-03-05 16:44:50 +01:00
cb_displaypolicy = p_policy ;
2016-01-13 11:39:31 +01:00
update ( ) ;
}
2017-07-22 18:15:31 +02:00
Tabs : : CloseButtonDisplayPolicy Tabs : : get_tab_close_display_policy ( ) const {
return cb_displaypolicy ;
}
2017-06-15 17:30:03 +02:00
void Tabs : : set_min_width ( int p_width ) {
min_width = p_width ;
}
2017-11-16 23:57:57 +01:00
void Tabs : : set_scrolling_enabled ( bool p_enabled ) {
scrolling_enabled = p_enabled ;
}
bool Tabs : : get_scrolling_enabled ( ) const {
return scrolling_enabled ;
}
2018-02-07 14:01:45 +01:00
void Tabs : : set_drag_to_rearrange_enabled ( bool p_enabled ) {
drag_to_rearrange_enabled = p_enabled ;
}
bool Tabs : : get_drag_to_rearrange_enabled ( ) const {
return drag_to_rearrange_enabled ;
}
2020-05-14 14:29:06 +02:00
2018-02-07 14:01:45 +01:00
void Tabs : : set_tabs_rearrange_group ( int p_group_id ) {
tabs_rearrange_group = p_group_id ;
}
int Tabs : : get_tabs_rearrange_group ( ) const {
return tabs_rearrange_group ;
}
2018-01-02 08:10:49 +01:00
void Tabs : : set_select_with_rmb ( bool p_enabled ) {
select_with_rmb = p_enabled ;
}
bool Tabs : : get_select_with_rmb ( ) const {
return select_with_rmb ;
}
2014-02-10 02:10:30 +01:00
void Tabs : : _bind_methods ( ) {
2017-03-05 16:44:50 +01:00
ClassDB : : bind_method ( D_METHOD ( " _gui_input " ) , & Tabs : : _gui_input ) ;
2018-10-21 15:57:55 +02:00
ClassDB : : bind_method ( D_METHOD ( " _update_hover " ) , & Tabs : : _update_hover ) ;
2017-03-05 16:44:50 +01:00
ClassDB : : bind_method ( D_METHOD ( " get_tab_count " ) , & Tabs : : get_tab_count ) ;
ClassDB : : bind_method ( D_METHOD ( " set_current_tab " , " tab_idx " ) , & Tabs : : set_current_tab ) ;
ClassDB : : bind_method ( D_METHOD ( " get_current_tab " ) , & Tabs : : get_current_tab ) ;
2020-09-08 22:30:47 +02:00
ClassDB : : bind_method ( D_METHOD ( " get_previous_tab " ) , & Tabs : : get_previous_tab ) ;
2017-03-05 16:44:50 +01:00
ClassDB : : bind_method ( D_METHOD ( " set_tab_title " , " tab_idx " , " title " ) , & Tabs : : set_tab_title ) ;
ClassDB : : bind_method ( D_METHOD ( " get_tab_title " , " tab_idx " ) , & Tabs : : get_tab_title ) ;
2020-09-03 13:22:16 +02:00
ClassDB : : bind_method ( D_METHOD ( " set_tab_text_direction " , " tab_idx " , " direction " ) , & Tabs : : set_tab_text_direction ) ;
ClassDB : : bind_method ( D_METHOD ( " get_tab_text_direction " , " tab_idx " ) , & Tabs : : get_tab_text_direction ) ;
ClassDB : : bind_method ( D_METHOD ( " set_tab_opentype_feature " , " tab_idx " , " tag " , " values " ) , & Tabs : : set_tab_opentype_feature ) ;
ClassDB : : bind_method ( D_METHOD ( " get_tab_opentype_feature " , " tab_idx " , " tag " ) , & Tabs : : get_tab_opentype_feature ) ;
ClassDB : : bind_method ( D_METHOD ( " clear_tab_opentype_features " , " tab_idx " ) , & Tabs : : clear_tab_opentype_features ) ;
ClassDB : : bind_method ( D_METHOD ( " set_tab_language " , " tab_idx " , " language " ) , & Tabs : : set_tab_language ) ;
ClassDB : : bind_method ( D_METHOD ( " get_tab_language " , " tab_idx " ) , & Tabs : : get_tab_language ) ;
2017-08-09 13:19:41 +02:00
ClassDB : : bind_method ( D_METHOD ( " set_tab_icon " , " tab_idx " , " icon " ) , & Tabs : : set_tab_icon ) ;
ClassDB : : bind_method ( D_METHOD ( " get_tab_icon " , " tab_idx " ) , & Tabs : : get_tab_icon ) ;
2017-02-27 19:07:50 +01:00
ClassDB : : bind_method ( D_METHOD ( " set_tab_disabled " , " tab_idx " , " disabled " ) , & Tabs : : set_tab_disabled ) ;
ClassDB : : bind_method ( D_METHOD ( " get_tab_disabled " , " tab_idx " ) , & Tabs : : get_tab_disabled ) ;
2017-03-05 16:44:50 +01:00
ClassDB : : bind_method ( D_METHOD ( " remove_tab " , " tab_idx " ) , & Tabs : : remove_tab ) ;
2019-06-11 20:43:37 +02:00
ClassDB : : bind_method ( D_METHOD ( " add_tab " , " title " , " icon " ) , & Tabs : : add_tab , DEFVAL ( " " ) , DEFVAL ( Ref < Texture2D > ( ) ) ) ;
2017-03-05 16:44:50 +01:00
ClassDB : : bind_method ( D_METHOD ( " set_tab_align " , " align " ) , & Tabs : : set_tab_align ) ;
ClassDB : : bind_method ( D_METHOD ( " get_tab_align " ) , & Tabs : : get_tab_align ) ;
2017-11-16 23:57:57 +01:00
ClassDB : : bind_method ( D_METHOD ( " get_tab_offset " ) , & Tabs : : get_tab_offset ) ;
ClassDB : : bind_method ( D_METHOD ( " get_offset_buttons_visible " ) , & Tabs : : get_offset_buttons_visible ) ;
2017-03-05 16:44:50 +01:00
ClassDB : : bind_method ( D_METHOD ( " ensure_tab_visible " , " idx " ) , & Tabs : : ensure_tab_visible ) ;
2017-07-12 16:48:43 +02:00
ClassDB : : bind_method ( D_METHOD ( " get_tab_rect " , " tab_idx " ) , & Tabs : : get_tab_rect ) ;
ClassDB : : bind_method ( D_METHOD ( " move_tab " , " from " , " to " ) , & Tabs : : move_tab ) ;
2017-07-22 18:15:31 +02:00
ClassDB : : bind_method ( D_METHOD ( " set_tab_close_display_policy " , " policy " ) , & Tabs : : set_tab_close_display_policy ) ;
ClassDB : : bind_method ( D_METHOD ( " get_tab_close_display_policy " ) , & Tabs : : get_tab_close_display_policy ) ;
2017-11-16 23:57:57 +01:00
ClassDB : : bind_method ( D_METHOD ( " set_scrolling_enabled " , " enabled " ) , & Tabs : : set_scrolling_enabled ) ;
ClassDB : : bind_method ( D_METHOD ( " get_scrolling_enabled " ) , & Tabs : : get_scrolling_enabled ) ;
2018-02-07 14:01:45 +01:00
ClassDB : : bind_method ( D_METHOD ( " set_drag_to_rearrange_enabled " , " enabled " ) , & Tabs : : set_drag_to_rearrange_enabled ) ;
ClassDB : : bind_method ( D_METHOD ( " get_drag_to_rearrange_enabled " ) , & Tabs : : get_drag_to_rearrange_enabled ) ;
ClassDB : : bind_method ( D_METHOD ( " set_tabs_rearrange_group " , " group_id " ) , & Tabs : : set_tabs_rearrange_group ) ;
ClassDB : : bind_method ( D_METHOD ( " get_tabs_rearrange_group " ) , & Tabs : : get_tabs_rearrange_group ) ;
2015-08-18 20:27:01 +02:00
2018-01-02 08:10:49 +01:00
ClassDB : : bind_method ( D_METHOD ( " set_select_with_rmb " , " enabled " ) , & Tabs : : set_select_with_rmb ) ;
ClassDB : : bind_method ( D_METHOD ( " get_select_with_rmb " ) , & Tabs : : get_select_with_rmb ) ;
2017-03-05 16:44:50 +01:00
ADD_SIGNAL ( MethodInfo ( " tab_changed " , PropertyInfo ( Variant : : INT , " tab " ) ) ) ;
ADD_SIGNAL ( MethodInfo ( " right_button_pressed " , PropertyInfo ( Variant : : INT , " tab " ) ) ) ;
2020-12-08 10:51:06 +01:00
ADD_SIGNAL ( MethodInfo ( " tab_closed " , PropertyInfo ( Variant : : INT , " tab " ) ) ) ;
ADD_SIGNAL ( MethodInfo ( " tab_hovered " , PropertyInfo ( Variant : : INT , " tab " ) ) ) ;
2017-07-05 15:44:53 +02:00
ADD_SIGNAL ( MethodInfo ( " reposition_active_tab_request " , PropertyInfo ( Variant : : INT , " idx_to " ) ) ) ;
2017-08-09 09:41:26 +02:00
ADD_SIGNAL ( MethodInfo ( " tab_clicked " , PropertyInfo ( Variant : : INT , " tab " ) ) ) ;
2014-02-10 02:10:30 +01:00
2017-03-05 16:44:50 +01:00
ADD_PROPERTY ( PropertyInfo ( Variant : : INT , " current_tab " , PROPERTY_HINT_RANGE , " -1,4096,1 " , PROPERTY_USAGE_EDITOR ) , " set_current_tab " , " get_current_tab " ) ;
2018-01-11 23:35:12 +01:00
ADD_PROPERTY ( PropertyInfo ( Variant : : INT , " tab_align " , PROPERTY_HINT_ENUM , " Left,Center,Right " ) , " set_tab_align " , " get_tab_align " ) ;
2018-11-08 15:30:02 +01:00
ADD_PROPERTY ( PropertyInfo ( Variant : : INT , " tab_close_display_policy " , PROPERTY_HINT_ENUM , " Show Never,Show Active Only,Show Always " ) , " set_tab_close_display_policy " , " get_tab_close_display_policy " ) ;
2017-11-16 23:57:57 +01:00
ADD_PROPERTY ( PropertyInfo ( Variant : : BOOL , " scrolling_enabled " ) , " set_scrolling_enabled " , " get_scrolling_enabled " ) ;
2018-02-07 14:01:45 +01:00
ADD_PROPERTY ( PropertyInfo ( Variant : : BOOL , " drag_to_rearrange_enabled " ) , " set_drag_to_rearrange_enabled " , " get_drag_to_rearrange_enabled " ) ;
2015-08-18 20:27:01 +02:00
2017-08-20 17:45:01 +02:00
BIND_ENUM_CONSTANT ( ALIGN_LEFT ) ;
BIND_ENUM_CONSTANT ( ALIGN_CENTER ) ;
BIND_ENUM_CONSTANT ( ALIGN_RIGHT ) ;
BIND_ENUM_CONSTANT ( ALIGN_MAX ) ;
2017-10-21 20:58:02 +02:00
BIND_ENUM_CONSTANT ( CLOSE_BUTTON_SHOW_NEVER ) ;
2017-08-20 17:45:01 +02:00
BIND_ENUM_CONSTANT ( CLOSE_BUTTON_SHOW_ACTIVE_ONLY ) ;
BIND_ENUM_CONSTANT ( CLOSE_BUTTON_SHOW_ALWAYS ) ;
BIND_ENUM_CONSTANT ( CLOSE_BUTTON_MAX ) ;
2014-02-10 02:10:30 +01:00
}
Tabs : : Tabs ( ) {
2020-02-21 18:28:45 +01:00
connect ( " mouse_exited " , callable_mp ( this , & Tabs : : _on_mouse_exited ) ) ;
2014-02-10 02:10:30 +01:00
}