2014-02-10 01:10:30 +00:00
/*************************************************************************/
2021-10-17 16:55:44 +00:00
/* tab_bar.cpp */
2014-02-10 01:10:30 +00:00
/*************************************************************************/
/* This file is part of: */
/* GODOT ENGINE */
2017-08-27 12:16:55 +00:00
/* https://godotengine.org */
2014-02-10 01:10:30 +00:00
/*************************************************************************/
2022-01-03 20:27:34 +00:00
/* Copyright (c) 2007-2022 Juan Linietsky, Ariel Manzur. */
/* Copyright (c) 2014-2022 Godot Engine contributors (cf. AUTHORS.md). */
2014-02-10 01:10:30 +00: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-04 23:50:27 +00:00
2021-10-17 16:55:44 +00:00
# include "tab_bar.h"
2014-02-10 01:10:30 +00:00
2020-11-07 22:33:38 +00:00
# include "core/object/message_queue.h"
2020-09-03 11:22:16 +00:00
# include "core/string/translation.h"
2018-02-07 13:01:45 +00:00
# include "scene/gui/box_container.h"
# include "scene/gui/label.h"
# include "scene/gui/texture_rect.h"
2014-02-10 01:10:30 +00:00
2021-10-17 16:55:44 +00:00
Size2 TabBar : : get_minimum_size ( ) const {
2022-01-19 16:11:44 +00:00
Size2 ms ;
if ( tabs . is_empty ( ) ) {
return ms ;
}
2021-07-17 21:22:52 +00:00
Ref < StyleBox > tab_unselected = get_theme_stylebox ( SNAME ( " tab_unselected " ) ) ;
Ref < StyleBox > tab_selected = get_theme_stylebox ( SNAME ( " tab_selected " ) ) ;
Ref < StyleBox > tab_disabled = get_theme_stylebox ( SNAME ( " tab_disabled " ) ) ;
2022-01-19 16:11:44 +00:00
Ref < StyleBox > button_highlight = get_theme_stylebox ( SNAME ( " button_highlight " ) ) ;
Ref < Texture2D > close = get_theme_icon ( SNAME ( " close " ) ) ;
int hseparation = get_theme_constant ( SNAME ( " hseparation " ) ) ;
2014-02-10 01:10:30 +00:00
2020-12-08 13:11:45 +00: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 11:22:16 +00:00
2017-03-05 15:44:50 +00:00
for ( int i = 0 ; i < tabs . size ( ) ; i + + ) {
2022-01-19 16:11:44 +00:00
if ( tabs [ i ] . hidden ) {
continue ;
2014-02-10 01:10:30 +00:00
}
2016-05-01 14:27:33 +00:00
2022-01-19 16:11:44 +00:00
int ofs = ms . width ;
2016-05-01 14:27:33 +00:00
2022-01-19 16:11:44 +00:00
Ref < StyleBox > style ;
2020-05-14 14:41:43 +00:00
if ( tabs [ i ] . disabled ) {
2022-01-19 16:11:44 +00:00
style = tab_disabled ;
2020-05-14 14:41:43 +00:00
} else if ( current = = i ) {
2022-01-19 16:11:44 +00:00
style = tab_selected ;
2020-05-14 14:41:43 +00:00
} else {
2022-01-19 16:11:44 +00:00
style = tab_unselected ;
}
ms . width + = style - > get_minimum_size ( ) . width ;
Ref < Texture2D > tex = tabs [ i ] . icon ;
if ( tex . is_valid ( ) ) {
ms . height = MAX ( ms . height , tex - > get_size ( ) . height ) ;
ms . width + = tex - > get_size ( ) . width + hseparation ;
2020-05-14 14:41:43 +00:00
}
2014-02-10 01:10:30 +00:00
2022-01-19 16:11:44 +00:00
if ( ! tabs [ i ] . text . is_empty ( ) ) {
ms . width + = tabs [ i ] . size_text + hseparation ;
}
ms . height = MAX ( ms . height , tabs [ i ] . text_buf - > get_size ( ) . y + y_margin ) ;
bool close_visible = cb_displaypolicy = = CLOSE_BUTTON_SHOW_ALWAYS | | ( cb_displaypolicy = = CLOSE_BUTTON_SHOW_ACTIVE_ONLY & & i = = current ) ;
2015-07-26 13:44:10 +00:00
if ( tabs [ i ] . right_button . is_valid ( ) ) {
2019-06-11 18:43:37 +00:00
Ref < Texture2D > rb = tabs [ i ] . right_button ;
2022-01-19 16:11:44 +00:00
if ( close_visible ) {
ms . width + = button_highlight - > get_minimum_size ( ) . width + rb - > get_width ( ) ;
} else {
ms . width + = button_highlight - > get_margin ( SIDE_LEFT ) + rb - > get_width ( ) + hseparation ;
}
ms . height = MAX ( rb - > get_height ( ) + style - > get_minimum_size ( ) . height , ms . height ) ;
}
if ( close_visible ) {
ms . width + = button_highlight - > get_margin ( SIDE_LEFT ) + close - > get_width ( ) + hseparation ;
ms . height = MAX ( close - > get_height ( ) + style - > get_minimum_size ( ) . height , ms . height ) ;
2015-07-26 13:44:10 +00:00
}
2015-08-18 18:27:01 +00:00
2022-01-19 16:11:44 +00:00
if ( ms . width - ofs > style - > get_minimum_size ( ) . width ) {
ms . width - = hseparation ;
2015-08-18 18:27:01 +00:00
}
2014-02-10 01:10:30 +00:00
}
2021-03-30 10:20:00 +00:00
if ( clip_tabs ) {
ms . width = 0 ;
}
2014-02-10 01:10:30 +00:00
return ms ;
}
2021-10-17 16:55:44 +00:00
void TabBar : : gui_input ( const Ref < InputEvent > & p_event ) {
2021-04-05 06:52:21 +00:00
ERR_FAIL_COND ( p_event . is_null ( ) ) ;
2017-05-20 15:38:03 +00:00
Ref < InputEventMouseMotion > mm = p_event ;
2014-02-10 01:10:30 +00:00
2017-05-20 15:38:03 +00:00
if ( mm . is_valid ( ) ) {
2017-06-03 08:54:24 +00:00
Point2 pos = mm - > get_position ( ) ;
2015-07-26 13:44:10 +00:00
2015-12-14 13:24:28 +00:00
if ( buttons_visible ) {
2021-07-17 21:22:52 +00:00
Ref < Texture2D > incr = get_theme_icon ( SNAME ( " increment " ) ) ;
Ref < Texture2D > decr = get_theme_icon ( SNAME ( " decrement " ) ) ;
2015-12-14 13:24:28 +00:00
2020-09-03 11:22:16 +00:00
if ( is_layout_rtl ( ) ) {
if ( pos . x < decr - > get_width ( ) ) {
2021-09-18 14:24:36 +00:00
if ( highlight_arrow ! = 1 ) {
highlight_arrow = 1 ;
update ( ) ;
}
2020-09-03 11:22:16 +00:00
} else if ( pos . x < incr - > get_width ( ) + decr - > get_width ( ) ) {
2021-09-18 14:24:36 +00:00
if ( highlight_arrow ! = 0 ) {
highlight_arrow = 0 ;
update ( ) ;
}
} else if ( highlight_arrow ! = - 1 ) {
highlight_arrow = - 1 ;
update ( ) ;
2020-09-03 11:22:16 +00:00
}
} else {
2021-03-30 10:20:00 +00:00
int limit_minus_buttons = get_size ( ) . width - incr - > get_width ( ) - decr - > get_width ( ) ;
if ( pos . x > limit_minus_buttons + decr - > get_width ( ) ) {
2021-09-18 14:24:36 +00:00
if ( highlight_arrow ! = 1 ) {
highlight_arrow = 1 ;
update ( ) ;
}
2021-03-30 10:20:00 +00:00
} else if ( pos . x > limit_minus_buttons ) {
2021-09-18 14:24:36 +00:00
if ( highlight_arrow ! = 0 ) {
highlight_arrow = 0 ;
update ( ) ;
}
} else if ( highlight_arrow ! = - 1 ) {
highlight_arrow = - 1 ;
update ( ) ;
2020-09-03 11:22:16 +00:00
}
2015-12-14 13:24:28 +00:00
}
}
2018-10-21 13:57:55 +00:00
_update_hover ( ) ;
2015-07-26 13:44:10 +00:00
return ;
}
2017-05-20 15:38:03 +00:00
Ref < InputEventMouseButton > mb = p_event ;
2017-11-16 22:57:57 +00:00
if ( mb . is_valid ( ) ) {
2021-08-13 21:31:57 +00:00
if ( mb - > is_pressed ( ) & & mb - > get_button_index ( ) = = MouseButton : : WHEEL_UP & & ! mb - > is_command_pressed ( ) ) {
2017-11-16 22:57:57 +00:00
if ( scrolling_enabled & & buttons_visible ) {
if ( offset > 0 ) {
offset - - ;
2022-01-12 00:33:43 +00:00
_update_cache ( ) ;
2017-11-16 22:57:57 +00:00
update ( ) ;
}
}
2015-07-26 13:44:10 +00:00
}
2021-08-13 21:31:57 +00:00
if ( mb - > is_pressed ( ) & & mb - > get_button_index ( ) = = MouseButton : : WHEEL_DOWN & & ! mb - > is_command_pressed ( ) ) {
2017-11-16 22:57:57 +00:00
if ( scrolling_enabled & & buttons_visible ) {
2022-01-12 00:33:43 +00:00
if ( missing_right & & offset < tabs . size ( ) ) {
2017-11-16 22:57:57 +00:00
offset + + ;
2022-01-12 00:33:43 +00:00
_update_cache ( ) ;
2017-11-16 22:57:57 +00:00
update ( ) ;
}
}
}
2021-08-13 21:31:57 +00:00
if ( rb_pressing & & ! mb - > is_pressed ( ) & & mb - > get_button_index ( ) = = MouseButton : : LEFT ) {
2017-11-16 22:57:57 +00:00
if ( rb_hover ! = - 1 ) {
2022-01-19 16:11:44 +00:00
emit_signal ( SNAME ( " tab_button_pressed " ) , rb_hover ) ;
2017-11-16 22:57:57 +00:00
}
2016-05-01 14:27:33 +00:00
2017-11-16 22:57:57 +00:00
rb_pressing = false ;
update ( ) ;
2015-08-18 18:27:01 +00:00
}
2021-08-13 21:31:57 +00:00
if ( cb_pressing & & ! mb - > is_pressed ( ) & & mb - > get_button_index ( ) = = MouseButton : : LEFT ) {
2017-11-16 22:57:57 +00:00
if ( cb_hover ! = - 1 ) {
2021-10-27 22:39:13 +00:00
emit_signal ( SNAME ( " tab_close_pressed " ) , cb_hover ) ;
2017-11-16 22:57:57 +00:00
}
2014-02-10 01:10:30 +00:00
2017-11-16 22:57:57 +00:00
cb_pressing = false ;
update ( ) ;
}
2014-02-10 01:10:30 +00:00
2021-08-13 21:31:57 +00:00
if ( mb - > is_pressed ( ) & & ( mb - > get_button_index ( ) = = MouseButton : : LEFT | | ( select_with_rmb & & mb - > get_button_index ( ) = = MouseButton : : RIGHT ) ) ) {
2021-09-25 09:01:45 +00:00
Point2 pos = mb - > get_position ( ) ;
2015-12-14 13:24:28 +00:00
2017-11-16 22:57:57 +00:00
if ( buttons_visible ) {
2021-07-17 21:22:52 +00:00
Ref < Texture2D > incr = get_theme_icon ( SNAME ( " increment " ) ) ;
Ref < Texture2D > decr = get_theme_icon ( SNAME ( " decrement " ) ) ;
2017-11-16 22:57:57 +00:00
2020-09-03 11:22:16 +00:00
if ( is_layout_rtl ( ) ) {
if ( pos . x < decr - > get_width ( ) ) {
if ( missing_right ) {
offset + + ;
2022-01-12 00:33:43 +00:00
_update_cache ( ) ;
2020-09-03 11:22:16 +00:00
update ( ) ;
}
return ;
} else if ( pos . x < incr - > get_width ( ) + decr - > get_width ( ) ) {
if ( offset > 0 ) {
offset - - ;
2022-01-12 00:33:43 +00:00
_update_cache ( ) ;
2020-09-03 11:22:16 +00:00
update ( ) ;
}
return ;
2017-11-16 22:57:57 +00:00
}
2020-09-03 11:22:16 +00:00
} else {
int limit = get_size ( ) . width - incr - > get_width ( ) - decr - > get_width ( ) ;
if ( pos . x > limit + decr - > get_width ( ) ) {
if ( missing_right ) {
offset + + ;
2022-01-12 00:33:43 +00:00
_update_cache ( ) ;
2020-09-03 11:22:16 +00:00
update ( ) ;
}
return ;
} else if ( pos . x > limit ) {
if ( offset > 0 ) {
offset - - ;
2022-01-12 00:33:43 +00:00
_update_cache ( ) ;
2020-09-03 11:22:16 +00:00
update ( ) ;
}
return ;
2017-11-16 22:57:57 +00:00
}
2015-12-14 13:24:28 +00:00
}
}
2021-10-27 19:13:04 +00:00
if ( tabs . is_empty ( ) ) {
2021-10-20 12:37:43 +00:00
// Return early if there are no actual tabs to handle input for.
return ;
}
2017-11-16 22:57:57 +00:00
int found = - 1 ;
2021-10-08 01:38:04 +00:00
for ( int i = offset ; i < = max_drawn_tab ; i + + ) {
2022-01-19 16:11:44 +00:00
if ( tabs [ i ] . hidden ) {
continue ;
}
2017-11-16 22:57:57 +00:00
if ( tabs [ i ] . rb_rect . has_point ( pos ) ) {
rb_pressing = true ;
update ( ) ;
return ;
}
2015-07-26 13:44:10 +00:00
2021-11-29 20:12:05 +00:00
if ( tabs [ i ] . cb_rect . has_point ( pos ) & & ( cb_displaypolicy = = CLOSE_BUTTON_SHOW_ALWAYS | | ( cb_displaypolicy = = CLOSE_BUTTON_SHOW_ACTIVE_ONLY & & i = = current ) ) ) {
2017-11-16 22:57:57 +00:00
cb_pressing = true ;
update ( ) ;
return ;
}
2015-08-18 18:27:01 +00:00
2020-09-03 11:22:16 +00: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 22:57:57 +00:00
if ( ! tabs [ i ] . disabled ) {
found = i ;
}
break ;
2017-02-27 18:07:50 +00:00
}
2014-02-10 01:10:30 +00:00
}
2017-11-16 22:57:57 +00:00
if ( found ! = - 1 ) {
set_current_tab ( found ) ;
2022-01-19 16:11:44 +00:00
if ( mb - > get_button_index ( ) = = MouseButton : : RIGHT ) {
// Right mouse button clicked.
emit_signal ( SNAME ( " tab_rmb_clicked " ) , found ) ;
}
2021-07-17 21:22:52 +00:00
emit_signal ( SNAME ( " tab_clicked " ) , found ) ;
2017-11-16 22:57:57 +00:00
}
2014-02-10 01:10:30 +00:00
}
}
}
2021-10-17 16:55:44 +00:00
void TabBar : : _shape ( int p_tab ) {
2021-07-17 21:22:52 +00:00
Ref < Font > font = get_theme_font ( SNAME ( " font " ) ) ;
int font_size = get_theme_font_size ( SNAME ( " font_size " ) ) ;
2020-09-03 11:22:16 +00:00
2021-05-27 17:31:33 +00:00
tabs . write [ p_tab ] . xl_text = atr ( tabs [ p_tab ] . text ) ;
2020-09-03 11:22:16 +00:00
tabs . write [ p_tab ] . text_buf - > clear ( ) ;
2021-09-24 00:20:15 +00:00
tabs . write [ p_tab ] . text_buf - > set_width ( - 1 ) ;
2020-09-03 11:22:16 +00:00
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 ) ;
}
2022-01-19 16:11:44 +00:00
tabs . write [ p_tab ] . text_buf - > add_string ( tabs [ p_tab ] . xl_text , font , font_size , tabs [ p_tab ] . opentype_features , ! tabs [ p_tab ] . language . is_empty ( ) ? tabs [ p_tab ] . language : TranslationServer : : get_singleton ( ) - > get_tool_locale ( ) ) ;
2020-09-03 11:22:16 +00:00
}
2021-10-17 16:55:44 +00:00
void TabBar : : _notification ( int p_what ) {
2017-03-05 15:44:50 +00:00
switch ( p_what ) {
2020-09-03 11:22:16 +00:00
case NOTIFICATION_LAYOUT_DIRECTION_CHANGED : {
update ( ) ;
} break ;
2022-02-15 17:06:48 +00:00
2021-06-04 13:21:52 +00:00
case NOTIFICATION_THEME_CHANGED :
2019-05-05 16:50:37 +00:00
case NOTIFICATION_TRANSLATION_CHANGED : {
2019-09-30 09:56:20 +00:00
for ( int i = 0 ; i < tabs . size ( ) ; + + i ) {
2020-09-03 11:22:16 +00:00
_shape ( i ) ;
2019-09-30 09:56:20 +00:00
}
2022-01-19 16:11:44 +00:00
[[fallthrough]] ;
}
2016-01-22 23:19:57 +00:00
case NOTIFICATION_RESIZED : {
2022-01-19 16:11:44 +00:00
int ofs_old = offset ;
int max_old = max_drawn_tab ;
2017-06-15 15:30:03 +00:00
_update_cache ( ) ;
2016-01-22 23:19:57 +00:00
_ensure_no_over_offset ( ) ;
2022-01-19 16:11:44 +00:00
if ( scroll_to_selected & & ( offset ! = ofs_old | | max_drawn_tab ! = max_old ) ) {
ensure_tab_visible ( current ) ;
}
2016-01-22 23:19:57 +00:00
} break ;
2022-02-15 17:06:48 +00:00
2014-02-10 01:10:30 +00:00
case NOTIFICATION_DRAW : {
2022-01-12 00:33:43 +00:00
if ( tabs . is_empty ( ) ) {
return ;
}
2021-07-17 21:22:52 +00:00
Ref < StyleBox > tab_unselected = get_theme_stylebox ( SNAME ( " tab_unselected " ) ) ;
Ref < StyleBox > tab_selected = get_theme_stylebox ( SNAME ( " tab_selected " ) ) ;
Ref < StyleBox > tab_disabled = get_theme_stylebox ( SNAME ( " tab_disabled " ) ) ;
Color font_selected_color = get_theme_color ( SNAME ( " font_selected_color " ) ) ;
Color font_unselected_color = get_theme_color ( SNAME ( " font_unselected_color " ) ) ;
Color font_disabled_color = get_theme_color ( SNAME ( " font_disabled_color " ) ) ;
Ref < Texture2D > incr = get_theme_icon ( SNAME ( " increment " ) ) ;
Ref < Texture2D > decr = get_theme_icon ( SNAME ( " decrement " ) ) ;
Ref < Texture2D > incr_hl = get_theme_icon ( SNAME ( " increment_highlight " ) ) ;
Ref < Texture2D > decr_hl = get_theme_icon ( SNAME ( " decrement_highlight " ) ) ;
2015-12-14 13:24:28 +00:00
2022-01-12 00:33:43 +00:00
bool rtl = is_layout_rtl ( ) ;
2022-01-12 23:43:31 +00:00
Vector2 size = get_size ( ) ;
int limit_minus_buttons = size . width - incr - > get_width ( ) - decr - > get_width ( ) ;
2014-02-10 01:10:30 +00:00
2022-01-12 23:43:31 +00:00
int ofs = tabs [ offset ] . ofs_cache ;
2015-12-14 13:24:28 +00:00
2022-01-12 23:43:31 +00:00
// Draw unselected tabs in the back.
2022-01-12 00:33:43 +00:00
for ( int i = offset ; i < = max_drawn_tab ; i + + ) {
2022-01-19 16:11:44 +00:00
if ( tabs [ i ] . hidden ) {
continue ;
}
2022-01-12 23:43:31 +00:00
if ( i ! = current ) {
Ref < StyleBox > sb ;
Color col ;
if ( tabs [ i ] . disabled ) {
sb = tab_disabled ;
col = font_disabled_color ;
} else if ( i = = current ) {
sb = tab_selected ;
col = font_selected_color ;
2020-09-03 11:22:16 +00:00
} else {
2022-01-12 23:43:31 +00:00
sb = tab_unselected ;
col = font_unselected_color ;
2020-05-14 14:41:43 +00:00
}
2014-02-10 01:10:30 +00:00
2022-01-12 23:43:31 +00:00
_draw_tab ( sb , col , i , rtl ? size . width - ofs - tabs [ i ] . size_cache : ofs ) ;
2020-09-03 11:22:16 +00:00
}
2014-02-10 01:10:30 +00:00
2022-01-12 23:43:31 +00:00
ofs + = tabs [ i ] . size_cache ;
}
2015-08-18 18:27:01 +00:00
2022-01-12 23:43:31 +00:00
// Draw selected tab in the front, but only if it's visible.
2022-01-19 16:11:44 +00:00
if ( current > = offset & & current < = max_drawn_tab & & ! tabs [ current ] . hidden ) {
2022-01-12 23:43:31 +00:00
Ref < StyleBox > sb = tabs [ current ] . disabled ? tab_disabled : tab_selected ;
float x = rtl ? size . width - tabs [ current ] . ofs_cache - tabs [ current ] . size_cache : tabs [ current ] . ofs_cache ;
2015-08-18 18:27:01 +00:00
2022-01-12 23:43:31 +00:00
_draw_tab ( sb , font_selected_color , current , x ) ;
2014-02-10 01:10:30 +00:00
}
2022-01-19 16:11:44 +00:00
if ( buttons_visible ) {
2017-03-05 15:44:50 +00:00
int vofs = ( get_size ( ) . height - incr - > get_size ( ) . height ) / 2 ;
2015-12-14 13:24:28 +00:00
2020-09-03 11:22:16 +00: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 13:24:28 +00:00
2020-09-03 11:22:16 +00: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 14:41:43 +00:00
} else {
2020-09-03 11:22:16 +00:00
if ( offset > 0 ) {
2021-03-30 10:20:00 +00:00
draw_texture ( highlight_arrow = = 0 ? decr_hl : decr , Point2 ( limit_minus_buttons , vofs ) ) ;
2020-09-03 11:22:16 +00:00
} else {
2021-03-30 10:20:00 +00:00
draw_texture ( decr , Point2 ( limit_minus_buttons , vofs ) , Color ( 1 , 1 , 1 , 0.5 ) ) ;
2020-09-03 11:22:16 +00:00
}
if ( missing_right ) {
2021-03-30 10:20:00 +00:00
draw_texture ( highlight_arrow = = 1 ? incr_hl : incr , Point2 ( limit_minus_buttons + decr - > get_size ( ) . width , vofs ) ) ;
2020-09-03 11:22:16 +00:00
} else {
2021-03-30 10:20:00 +00:00
draw_texture ( incr , Point2 ( limit_minus_buttons + decr - > get_size ( ) . width , vofs ) , Color ( 1 , 1 , 1 , 0.5 ) ) ;
2020-09-03 11:22:16 +00:00
}
2020-05-14 14:41:43 +00:00
}
2015-12-14 13:24:28 +00:00
}
2014-02-10 01:10:30 +00:00
} break ;
}
}
2022-01-12 23:43:31 +00:00
void TabBar : : _draw_tab ( Ref < StyleBox > & p_tab_style , Color & p_font_color , int p_index , float p_x ) {
RID ci = get_canvas_item ( ) ;
2022-01-19 16:11:44 +00:00
bool rtl = is_layout_rtl ( ) ;
2022-01-12 23:43:31 +00:00
Color font_outline_color = get_theme_color ( SNAME ( " font_outline_color " ) ) ;
int outline_size = get_theme_constant ( SNAME ( " outline_size " ) ) ;
2022-01-19 16:11:44 +00:00
int hseparation = get_theme_constant ( SNAME ( " hseparation " ) ) ;
2022-01-12 23:43:31 +00:00
2022-01-19 16:11:44 +00:00
Rect2 sb_rect = Rect2 ( p_x , 0 , tabs [ p_index ] . size_cache , get_size ( ) . height ) ;
2022-01-12 23:43:31 +00:00
p_tab_style - > draw ( ci , sb_rect ) ;
2022-01-19 16:11:44 +00:00
p_x + = rtl ? tabs [ p_index ] . size_cache - p_tab_style - > get_margin ( SIDE_LEFT ) : p_tab_style - > get_margin ( SIDE_LEFT ) ;
2022-01-12 23:43:31 +00:00
Size2i sb_ms = p_tab_style - > get_minimum_size ( ) ;
2022-01-19 16:11:44 +00:00
// Draw the icon.
Ref < Texture2D > icon = tabs [ p_index ] . icon ;
2022-01-12 23:43:31 +00:00
if ( icon . is_valid ( ) ) {
2022-01-19 16:11:44 +00:00
icon - > draw ( ci , Point2i ( rtl ? p_x - icon - > get_width ( ) : p_x , p_tab_style - > get_margin ( SIDE_TOP ) + ( ( sb_rect . size . y - sb_ms . y ) - icon - > get_height ( ) ) / 2 ) ) ;
2022-01-12 23:43:31 +00:00
2022-01-19 16:11:44 +00:00
p_x = rtl ? p_x - icon - > get_width ( ) - hseparation : p_x + icon - > get_width ( ) + hseparation ;
2022-01-12 23:43:31 +00:00
}
2022-01-19 16:11:44 +00:00
// Draw the text.
if ( ! tabs [ p_index ] . text . is_empty ( ) ) {
Point2i text_pos = Point2i ( rtl ? p_x - tabs [ p_index ] . size_text : p_x ,
p_tab_style - > get_margin ( SIDE_TOP ) + ( ( sb_rect . size . y - sb_ms . y ) - tabs [ p_index ] . text_buf - > get_size ( ) . y ) / 2 ) ;
2022-01-12 23:43:31 +00:00
2022-01-19 16:11:44 +00:00
if ( outline_size > 0 & & font_outline_color . a > 0 ) {
tabs [ p_index ] . text_buf - > draw_outline ( ci , text_pos , outline_size , font_outline_color ) ;
}
tabs [ p_index ] . text_buf - > draw ( ci , text_pos , p_font_color ) ;
2022-01-12 23:43:31 +00:00
2022-01-19 16:11:44 +00:00
p_x = rtl ? p_x - tabs [ p_index ] . size_text - hseparation : p_x + tabs [ p_index ] . size_text + hseparation ;
}
2022-01-12 23:43:31 +00:00
2022-01-19 16:11:44 +00:00
// Draw and calculate rect of the right button.
if ( tabs [ p_index ] . right_button . is_valid ( ) ) {
Ref < StyleBox > style = get_theme_stylebox ( SNAME ( " button_highlight " ) ) ;
Ref < Texture2D > rb = tabs [ p_index ] . right_button ;
2022-01-12 23:43:31 +00:00
Rect2 rb_rect ;
rb_rect . size = style - > get_minimum_size ( ) + rb - > get_size ( ) ;
2022-01-19 16:11:44 +00:00
rb_rect . position . x = rtl ? p_x - rb_rect . size . width : p_x ;
2022-01-12 23:43:31 +00:00
rb_rect . position . y = p_tab_style - > get_margin ( SIDE_TOP ) + ( ( sb_rect . size . y - sb_ms . y ) - ( rb_rect . size . y ) ) / 2 ;
2022-01-19 16:11:44 +00:00
tabs . write [ p_index ] . rb_rect = rb_rect ;
2022-01-12 23:43:31 +00:00
if ( rb_hover = = p_index ) {
if ( rb_pressing ) {
get_theme_stylebox ( SNAME ( " button_pressed " ) ) - > draw ( ci , rb_rect ) ;
} else {
style - > draw ( ci , rb_rect ) ;
}
}
2022-01-19 16:11:44 +00:00
rb - > draw ( ci , Point2i ( rb_rect . position . x + style - > get_margin ( SIDE_LEFT ) , rb_rect . position . y + style - > get_margin ( SIDE_TOP ) ) ) ;
p_x = rtl ? rb_rect . position . x : rb_rect . position . x + rb_rect . size . width ;
2022-01-12 23:43:31 +00:00
}
2022-01-19 16:11:44 +00:00
// Draw and calculate rect of the close button.
2022-01-12 23:43:31 +00:00
if ( cb_displaypolicy = = CLOSE_BUTTON_SHOW_ALWAYS | | ( cb_displaypolicy = = CLOSE_BUTTON_SHOW_ACTIVE_ONLY & & p_index = = current ) ) {
2022-01-19 16:11:44 +00:00
Ref < StyleBox > style = get_theme_stylebox ( SNAME ( " button_highlight " ) ) ;
2022-01-12 23:43:31 +00:00
Ref < Texture2D > cb = get_theme_icon ( SNAME ( " close " ) ) ;
Rect2 cb_rect ;
cb_rect . size = style - > get_minimum_size ( ) + cb - > get_size ( ) ;
2022-01-19 16:11:44 +00:00
cb_rect . position . x = rtl ? p_x - cb_rect . size . width : p_x ;
2022-01-12 23:43:31 +00:00
cb_rect . position . y = p_tab_style - > get_margin ( SIDE_TOP ) + ( ( sb_rect . size . y - sb_ms . y ) - ( cb_rect . size . y ) ) / 2 ;
2022-01-19 16:11:44 +00:00
tabs . write [ p_index ] . cb_rect = cb_rect ;
if ( ! tabs [ p_index ] . disabled & & cb_hover = = p_index ) {
2022-01-12 23:43:31 +00:00
if ( cb_pressing ) {
2022-01-19 16:11:44 +00:00
get_theme_stylebox ( SNAME ( " button_pressed " ) ) - > draw ( ci , cb_rect ) ;
2022-01-12 23:43:31 +00:00
} else {
style - > draw ( ci , cb_rect ) ;
}
}
2022-01-19 16:11:44 +00:00
cb - > draw ( ci , Point2i ( cb_rect . position . x + style - > get_margin ( SIDE_LEFT ) , cb_rect . position . y + style - > get_margin ( SIDE_TOP ) ) ) ;
2022-01-12 23:43:31 +00:00
}
}
2021-11-03 18:05:28 +00:00
void TabBar : : set_tab_count ( int p_count ) {
2022-01-19 16:11:44 +00:00
if ( p_count = = tabs . size ( ) ) {
return ;
}
2021-11-03 18:05:28 +00:00
ERR_FAIL_COND ( p_count < 0 ) ;
tabs . resize ( p_count ) ;
2022-01-19 16:11:44 +00:00
if ( p_count = = 0 ) {
offset = 0 ;
max_drawn_tab = 0 ;
current = 0 ;
previous = 0 ;
} else {
offset = MIN ( offset , p_count - 1 ) ;
max_drawn_tab = MIN ( max_drawn_tab , p_count - 1 ) ;
current = MIN ( current , p_count - 1 ) ;
2022-02-14 04:58:17 +00:00
_update_cache ( ) ;
_ensure_no_over_offset ( ) ;
if ( scroll_to_selected ) {
ensure_tab_visible ( current ) ;
}
2022-01-19 16:11:44 +00:00
}
2022-02-14 04:58:17 +00:00
2021-11-03 18:05:28 +00:00
update ( ) ;
2022-01-19 16:11:44 +00:00
update_minimum_size ( ) ;
2021-11-03 18:05:28 +00:00
notify_property_list_changed ( ) ;
}
2021-10-17 16:55:44 +00:00
int TabBar : : get_tab_count ( ) const {
2014-02-10 01:10:30 +00:00
return tabs . size ( ) ;
}
2021-10-17 16:55:44 +00:00
void TabBar : : set_current_tab ( int p_current ) {
2017-03-05 15:44:50 +00:00
ERR_FAIL_INDEX ( p_current , get_tab_count ( ) ) ;
2014-02-10 01:10:30 +00:00
2020-09-08 20:30:47 +00:00
previous = current ;
2017-03-05 15:44:50 +00:00
current = p_current ;
2014-02-10 01:10:30 +00:00
2022-01-19 16:11:44 +00:00
if ( current = = previous ) {
emit_signal ( SNAME ( " tab_selected " ) , current ) ;
return ;
}
emit_signal ( SNAME ( " tab_selected " ) , current ) ;
2017-06-15 15:30:03 +00:00
_update_cache ( ) ;
2022-01-19 16:11:44 +00:00
if ( scroll_to_selected ) {
ensure_tab_visible ( current ) ;
}
2014-02-10 01:10:30 +00:00
update ( ) ;
2017-08-05 15:56:00 +00:00
2021-07-17 21:22:52 +00:00
emit_signal ( SNAME ( " tab_changed " ) , p_current ) ;
2014-02-10 01:10:30 +00:00
}
2021-10-17 16:55:44 +00:00
int TabBar : : get_current_tab ( ) const {
2014-02-10 01:10:30 +00:00
return current ;
}
2021-10-17 16:55:44 +00:00
int TabBar : : get_previous_tab ( ) const {
2020-09-08 20:30:47 +00:00
return previous ;
}
2021-10-17 16:55:44 +00:00
int TabBar : : get_hovered_tab ( ) const {
2017-06-15 15:30:03 +00:00
return hover ;
}
2021-10-17 16:55:44 +00:00
int TabBar : : get_tab_offset ( ) const {
2017-11-16 22:57:57 +00:00
return offset ;
}
2021-10-17 16:55:44 +00:00
bool TabBar : : get_offset_buttons_visible ( ) const {
2017-11-16 22:57:57 +00:00
return buttons_visible ;
}
2021-10-17 16:55:44 +00:00
void TabBar : : set_tab_title ( int p_tab , const String & p_title ) {
2017-03-05 15:44:50 +00:00
ERR_FAIL_INDEX ( p_tab , tabs . size ( ) ) ;
2018-07-25 01:11:03 +00:00
tabs . write [ p_tab ] . text = p_title ;
2022-01-19 16:11:44 +00:00
2020-09-03 11:22:16 +00:00
_shape ( p_tab ) ;
2022-01-12 00:33:43 +00:00
_update_cache ( ) ;
2022-01-19 16:11:44 +00:00
_ensure_no_over_offset ( ) ;
if ( scroll_to_selected ) {
ensure_tab_visible ( current ) ;
}
2014-02-10 01:10:30 +00:00
update ( ) ;
2021-12-06 13:02:34 +00:00
update_minimum_size ( ) ;
2014-02-10 01:10:30 +00:00
}
2021-10-17 16:55:44 +00:00
String TabBar : : get_tab_title ( int p_tab ) const {
2017-03-05 15:44:50 +00:00
ERR_FAIL_INDEX_V ( p_tab , tabs . size ( ) , " " ) ;
2014-02-10 01:10:30 +00:00
return tabs [ p_tab ] . text ;
}
2021-10-17 16:55:44 +00:00
void TabBar : : set_tab_text_direction ( int p_tab , Control : : TextDirection p_text_direction ) {
2020-09-03 11:22:16 +00:00
ERR_FAIL_INDEX ( p_tab , tabs . size ( ) ) ;
ERR_FAIL_COND ( ( int ) p_text_direction < - 1 | | ( int ) p_text_direction > 3 ) ;
2022-01-19 16:11:44 +00:00
2020-09-03 11:22:16 +00:00
if ( tabs [ p_tab ] . text_direction ! = p_text_direction ) {
tabs . write [ p_tab ] . text_direction = p_text_direction ;
_shape ( p_tab ) ;
update ( ) ;
}
}
2021-10-17 16:55:44 +00:00
Control : : TextDirection TabBar : : get_tab_text_direction ( int p_tab ) const {
2020-09-03 11:22:16 +00:00
ERR_FAIL_INDEX_V ( p_tab , tabs . size ( ) , Control : : TEXT_DIRECTION_INHERITED ) ;
return tabs [ p_tab ] . text_direction ;
}
2021-10-17 16:55:44 +00:00
void TabBar : : clear_tab_opentype_features ( int p_tab ) {
2020-09-03 11:22:16 +00:00
ERR_FAIL_INDEX ( p_tab , tabs . size ( ) ) ;
tabs . write [ p_tab ] . opentype_features . clear ( ) ;
2022-01-19 16:11:44 +00:00
2020-09-03 11:22:16 +00:00
_shape ( p_tab ) ;
2022-01-12 00:33:43 +00:00
_update_cache ( ) ;
2022-01-19 16:11:44 +00:00
_ensure_no_over_offset ( ) ;
if ( scroll_to_selected ) {
ensure_tab_visible ( current ) ;
}
2020-09-03 11:22:16 +00:00
update ( ) ;
2022-01-19 16:11:44 +00:00
update_minimum_size ( ) ;
2020-09-03 11:22:16 +00:00
}
2021-10-17 16:55:44 +00:00
void TabBar : : set_tab_opentype_feature ( int p_tab , const String & p_name , int p_value ) {
2020-09-03 11:22:16 +00:00
ERR_FAIL_INDEX ( p_tab , tabs . size ( ) ) ;
2022-01-19 16:11:44 +00:00
2020-09-03 11:22:16 +00:00
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 ;
2022-01-19 16:11:44 +00:00
2020-09-03 11:22:16 +00:00
_shape ( p_tab ) ;
2022-01-12 00:33:43 +00:00
_update_cache ( ) ;
2022-01-19 16:11:44 +00:00
_ensure_no_over_offset ( ) ;
if ( scroll_to_selected ) {
ensure_tab_visible ( current ) ;
}
2020-09-03 11:22:16 +00:00
update ( ) ;
2022-01-19 16:11:44 +00:00
update_minimum_size ( ) ;
2020-09-03 11:22:16 +00:00
}
}
2021-10-17 16:55:44 +00:00
int TabBar : : get_tab_opentype_feature ( int p_tab , const String & p_name ) const {
2020-09-03 11:22:16 +00:00
ERR_FAIL_INDEX_V ( p_tab , tabs . size ( ) , - 1 ) ;
2022-01-19 16:11:44 +00:00
2020-09-03 11:22:16 +00:00
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 ] ;
}
2021-10-17 16:55:44 +00:00
void TabBar : : set_tab_language ( int p_tab , const String & p_language ) {
2020-09-03 11:22:16 +00:00
ERR_FAIL_INDEX ( p_tab , tabs . size ( ) ) ;
2022-01-19 16:11:44 +00:00
2020-09-03 11:22:16 +00:00
if ( tabs [ p_tab ] . language ! = p_language ) {
tabs . write [ p_tab ] . language = p_language ;
_shape ( p_tab ) ;
2022-01-19 16:11:44 +00:00
_update_cache ( ) ;
_ensure_no_over_offset ( ) ;
if ( scroll_to_selected ) {
ensure_tab_visible ( current ) ;
}
2020-09-03 11:22:16 +00:00
update ( ) ;
2022-01-19 16:11:44 +00:00
update_minimum_size ( ) ;
2020-09-03 11:22:16 +00:00
}
}
2021-10-17 16:55:44 +00:00
String TabBar : : get_tab_language ( int p_tab ) const {
2020-09-03 11:22:16 +00:00
ERR_FAIL_INDEX_V ( p_tab , tabs . size ( ) , " " ) ;
return tabs [ p_tab ] . language ;
}
2021-10-17 16:55:44 +00:00
void TabBar : : set_tab_icon ( int p_tab , const Ref < Texture2D > & p_icon ) {
2017-03-05 15:44:50 +00:00
ERR_FAIL_INDEX ( p_tab , tabs . size ( ) ) ;
2018-07-25 01:11:03 +00:00
tabs . write [ p_tab ] . icon = p_icon ;
2022-01-19 16:11:44 +00:00
2022-01-12 00:33:43 +00:00
_update_cache ( ) ;
2022-01-19 16:11:44 +00:00
_ensure_no_over_offset ( ) ;
if ( scroll_to_selected ) {
ensure_tab_visible ( current ) ;
}
2014-02-10 01:10:30 +00:00
update ( ) ;
2021-12-06 13:02:34 +00:00
update_minimum_size ( ) ;
2014-02-10 01:10:30 +00:00
}
2016-05-01 14:27:33 +00:00
2021-10-17 16:55:44 +00:00
Ref < Texture2D > TabBar : : get_tab_icon ( int p_tab ) const {
2019-06-11 18:43:37 +00:00
ERR_FAIL_INDEX_V ( p_tab , tabs . size ( ) , Ref < Texture2D > ( ) ) ;
2014-02-10 01:10:30 +00:00
return tabs [ p_tab ] . icon ;
}
2021-10-17 16:55:44 +00:00
void TabBar : : set_tab_disabled ( int p_tab , bool p_disabled ) {
2017-02-27 18:07:50 +00:00
ERR_FAIL_INDEX ( p_tab , tabs . size ( ) ) ;
2018-07-25 01:11:03 +00:00
tabs . write [ p_tab ] . disabled = p_disabled ;
2022-01-19 16:11:44 +00:00
_update_cache ( ) ;
_ensure_no_over_offset ( ) ;
if ( scroll_to_selected ) {
ensure_tab_visible ( current ) ;
}
2017-02-27 18:07:50 +00:00
update ( ) ;
2022-01-19 16:11:44 +00:00
update_minimum_size ( ) ;
2017-02-27 18:07:50 +00:00
}
2020-05-14 12:29:06 +00:00
2021-11-03 18:05:28 +00:00
bool TabBar : : is_tab_disabled ( int p_tab ) const {
2017-02-27 18:07:50 +00:00
ERR_FAIL_INDEX_V ( p_tab , tabs . size ( ) , false ) ;
return tabs [ p_tab ] . disabled ;
}
2022-01-19 16:11:44 +00:00
void TabBar : : set_tab_hidden ( int p_tab , bool p_hidden ) {
ERR_FAIL_INDEX ( p_tab , tabs . size ( ) ) ;
tabs . write [ p_tab ] . hidden = p_hidden ;
_update_cache ( ) ;
_ensure_no_over_offset ( ) ;
if ( scroll_to_selected ) {
ensure_tab_visible ( current ) ;
}
update ( ) ;
update_minimum_size ( ) ;
}
bool TabBar : : is_tab_hidden ( int p_tab ) const {
ERR_FAIL_INDEX_V ( p_tab , tabs . size ( ) , false ) ;
return tabs [ p_tab ] . hidden ;
}
void TabBar : : set_tab_button_icon ( int p_tab , const Ref < Texture2D > & p_icon ) {
2017-03-05 15:44:50 +00:00
ERR_FAIL_INDEX ( p_tab , tabs . size ( ) ) ;
2022-01-19 16:11:44 +00:00
tabs . write [ p_tab ] . right_button = p_icon ;
2017-11-16 22:57:57 +00:00
_update_cache ( ) ;
2022-01-19 16:11:44 +00:00
_ensure_no_over_offset ( ) ;
if ( scroll_to_selected ) {
ensure_tab_visible ( current ) ;
}
2015-07-26 13:44:10 +00:00
update ( ) ;
2021-12-06 13:02:34 +00:00
update_minimum_size ( ) ;
2015-07-26 13:44:10 +00:00
}
2020-05-14 12:29:06 +00:00
2022-01-19 16:11:44 +00:00
Ref < Texture2D > TabBar : : get_tab_button_icon ( int p_tab ) const {
2019-06-11 18:43:37 +00:00
ERR_FAIL_INDEX_V ( p_tab , tabs . size ( ) , Ref < Texture2D > ( ) ) ;
2015-07-26 13:44:10 +00:00
return tabs [ p_tab ] . right_button ;
}
2021-10-17 16:55:44 +00:00
void TabBar : : _update_hover ( ) {
2018-10-21 13:57:55 +00:00
if ( ! is_inside_tree ( ) ) {
return ;
}
2022-02-07 20:40:16 +00:00
ERR_FAIL_COND ( tabs . is_empty ( ) ) ;
2018-10-21 13:57:55 +00:00
const Point2 & pos = get_local_mouse_position ( ) ;
2022-01-19 16:11:44 +00:00
// Test hovering to display right or close button.
2018-10-21 13:57:55 +00:00
int hover_now = - 1 ;
int hover_buttons = - 1 ;
2022-01-19 16:11:44 +00:00
for ( int i = offset ; i < = max_drawn_tab ; i + + ) {
if ( tabs [ i ] . hidden ) {
continue ;
}
2018-10-21 13:57:55 +00:00
Rect2 rect = get_tab_rect ( i ) ;
if ( rect . has_point ( pos ) ) {
hover_now = i ;
}
2022-01-19 16:11:44 +00:00
2018-10-21 13:57:55 +00:00
if ( tabs [ i ] . rb_rect . has_point ( pos ) ) {
rb_hover = i ;
cb_hover = - 1 ;
hover_buttons = i ;
} else if ( ! tabs [ i ] . disabled & & tabs [ i ] . cb_rect . has_point ( pos ) ) {
cb_hover = i ;
rb_hover = - 1 ;
hover_buttons = i ;
2022-01-19 16:11:44 +00:00
}
if ( hover_buttons ! = - 1 ) {
update ( ) ;
2018-10-21 13:57:55 +00:00
break ;
}
}
2022-01-19 16:11:44 +00:00
2018-10-21 13:57:55 +00:00
if ( hover ! = hover_now ) {
hover = hover_now ;
2022-01-19 16:11:44 +00:00
if ( hover ! = - 1 ) {
emit_signal ( SNAME ( " tab_hovered " ) , hover ) ;
}
2018-10-21 13:57:55 +00:00
}
2021-11-29 20:12:05 +00:00
if ( hover_buttons = = - 1 ) { // No hover.
2022-01-19 16:11:44 +00:00
int rb_hover_old = rb_hover ;
int cb_hover_old = cb_hover ;
2018-10-21 13:57:55 +00:00
rb_hover = hover_buttons ;
cb_hover = hover_buttons ;
2022-01-19 16:11:44 +00:00
if ( rb_hover ! = rb_hover_old | | cb_hover ! = cb_hover_old ) {
update ( ) ;
}
2018-10-21 13:57:55 +00:00
}
}
2021-10-17 16:55:44 +00:00
void TabBar : : _update_cache ( ) {
2022-01-12 00:33:43 +00:00
if ( tabs . is_empty ( ) ) {
return ;
}
2021-07-17 21:22:52 +00:00
Ref < StyleBox > tab_disabled = get_theme_stylebox ( SNAME ( " tab_disabled " ) ) ;
Ref < StyleBox > tab_unselected = get_theme_stylebox ( SNAME ( " tab_unselected " ) ) ;
Ref < StyleBox > tab_selected = get_theme_stylebox ( SNAME ( " tab_selected " ) ) ;
Ref < Texture2D > incr = get_theme_icon ( SNAME ( " increment " ) ) ;
Ref < Texture2D > decr = get_theme_icon ( SNAME ( " decrement " ) ) ;
2022-01-12 00:33:43 +00:00
int limit = get_size ( ) . width ;
int limit_minus_buttons = limit - incr - > get_width ( ) - decr - > get_width ( ) ;
2017-06-15 15:30:03 +00:00
int w = 0 ;
int mw = 0 ;
int size_fixed = 0 ;
int count_resize = 0 ;
2022-01-12 00:33:43 +00:00
2017-06-15 15:30:03 +00:00
for ( int i = 0 ; i < tabs . size ( ) ; i + + ) {
2020-09-03 11:22:16 +00:00
tabs . write [ i ] . size_text = Math : : ceil ( tabs [ i ] . text_buf - > get_size ( ) . x ) ;
tabs . write [ i ] . text_buf - > set_width ( - 1 ) ;
2022-01-12 00:33:43 +00:00
2022-01-19 16:11:44 +00:00
tabs . write [ i ] . ofs_cache = 0 ;
tabs . write [ i ] . size_cache = get_tab_width ( i ) ;
if ( ! tabs [ i ] . hidden ) {
mw + = tabs [ i ] . size_cache ;
if ( tabs [ i ] . size_cache < = min_width | | i = = current ) {
size_fixed + = tabs [ i ] . size_cache ;
} else {
count_resize + + ;
}
2017-06-15 15:30:03 +00:00
}
}
2022-01-12 00:33:43 +00:00
2017-06-15 15:30:03 +00:00
int m_width = min_width ;
if ( count_resize > 0 ) {
2021-03-30 10:20:00 +00:00
m_width = MAX ( ( limit_minus_buttons - size_fixed ) / count_resize , min_width ) ;
2017-06-15 15:30:03 +00:00
}
2022-01-12 00:33:43 +00:00
2020-04-08 09:14:46 +00:00
for ( int i = offset ; i < tabs . size ( ) ; i + + ) {
2022-01-19 16:11:44 +00:00
if ( tabs [ i ] . hidden ) {
tabs . write [ i ] . ofs_cache = w ;
max_drawn_tab = i ;
continue ;
2017-06-15 15:30:03 +00:00
}
2022-01-12 00:33:43 +00:00
2017-06-15 15:30:03 +00:00
int lsize = tabs [ i ] . size_cache ;
int slen = tabs [ i ] . size_text ;
2022-01-12 00:33:43 +00:00
2022-01-19 16:11:44 +00:00
// FIXME: This is completely broken.
if ( min_width > 0 & & ( mw > limit | | ( offset > 0 & & mw > limit_minus_buttons ) ) & & i ! = current & & lsize > m_width ) {
slen = MAX ( m_width - tabs [ i ] . size_cache + tabs [ i ] . size_text , 1 ) ;
lsize = m_width ;
2017-06-15 15:30:03 +00:00
}
2022-01-12 00:33:43 +00:00
2018-07-25 01:11:03 +00:00
tabs . write [ i ] . ofs_cache = w ;
tabs . write [ i ] . size_cache = lsize ;
tabs . write [ i ] . size_text = slen ;
2020-09-03 11:22:16 +00:00
tabs . write [ i ] . text_buf - > set_width ( slen ) ;
2022-01-12 00:33:43 +00:00
2017-06-15 15:30:03 +00:00
w + = lsize ;
2022-01-12 00:33:43 +00:00
max_drawn_tab = i ;
// Check if all tabs would fit inside the area.
2022-01-19 16:11:44 +00:00
if ( clip_tabs & & i > offset & & ( w > limit | | ( offset > 0 & & w > limit_minus_buttons ) ) ) {
tabs . write [ i ] . ofs_cache = 0 ;
tabs . write [ i ] . text_buf - > set_width ( - 1 ) ;
w - = tabs [ i ] . size_cache ;
max_drawn_tab - - ;
2022-01-12 00:33:43 +00:00
while ( w > limit_minus_buttons & & max_drawn_tab > offset ) {
2022-01-19 16:11:44 +00:00
tabs . write [ max_drawn_tab ] . ofs_cache = 0 ;
if ( ! tabs [ max_drawn_tab ] . hidden ) {
tabs . write [ max_drawn_tab ] . text_buf - > set_width ( - 1 ) ;
w - = tabs [ max_drawn_tab ] . size_cache ;
}
max_drawn_tab - - ;
2022-01-12 00:33:43 +00:00
}
break ;
}
}
missing_right = max_drawn_tab < tabs . size ( ) - 1 ;
buttons_visible = offset > 0 | | missing_right ;
if ( tab_alignment = = ALIGNMENT_LEFT ) {
2022-01-19 16:11:44 +00:00
_update_hover ( ) ;
2022-01-12 00:33:43 +00:00
return ;
2022-01-19 16:11:44 +00:00
}
if ( tab_alignment = = ALIGNMENT_CENTER ) {
2022-01-12 00:33:43 +00:00
w = ( ( buttons_visible ? limit_minus_buttons : limit ) - w ) / 2 ;
} else if ( tab_alignment = = ALIGNMENT_RIGHT ) {
w = ( buttons_visible ? limit_minus_buttons : limit ) - w ;
}
for ( int i = offset ; i < = max_drawn_tab ; i + + ) {
tabs . write [ i ] . ofs_cache = w ;
2022-01-19 16:11:44 +00:00
if ( ! tabs [ i ] . hidden ) {
w + = tabs [ i ] . size_cache ;
}
2017-06-15 15:30:03 +00:00
}
2022-01-19 16:11:44 +00:00
_update_hover ( ) ;
2017-06-15 15:30:03 +00:00
}
2021-10-17 16:55:44 +00:00
void TabBar : : _on_mouse_exited ( ) {
2019-09-23 02:07:00 +00:00
rb_hover = - 1 ;
cb_hover = - 1 ;
hover = - 1 ;
highlight_arrow = - 1 ;
update ( ) ;
}
2021-10-17 16:55:44 +00:00
void TabBar : : add_tab ( const String & p_str , const Ref < Texture2D > & p_icon ) {
2014-02-10 01:10:30 +00:00
Tab t ;
2017-03-05 15:44:50 +00:00
t . text = p_str ;
2021-05-27 17:31:33 +00:00
t . xl_text = atr ( p_str ) ;
2020-09-03 11:22:16 +00:00
t . text_buf - > set_direction ( is_layout_rtl ( ) ? TextServer : : DIRECTION_RTL : TextServer : : DIRECTION_LTR ) ;
2021-07-17 21:22:52 +00:00
t . text_buf - > add_string ( t . xl_text , get_theme_font ( SNAME ( " font " ) ) , get_theme_font_size ( SNAME ( " font_size " ) ) , Dictionary ( ) , TranslationServer : : get_singleton ( ) - > get_tool_locale ( ) ) ;
2017-03-05 15:44:50 +00:00
t . icon = p_icon ;
2014-02-10 01:10:30 +00:00
tabs . push_back ( t ) ;
2022-01-19 16:11:44 +00:00
2017-06-15 15:30:03 +00:00
_update_cache ( ) ;
2022-01-19 16:11:44 +00:00
if ( scroll_to_selected ) {
ensure_tab_visible ( current ) ;
}
2014-02-10 01:10:30 +00:00
update ( ) ;
2021-12-06 13:02:34 +00:00
update_minimum_size ( ) ;
2022-03-02 14:37:10 +00:00
if ( tabs . size ( ) = = 1 & & is_inside_tree ( ) ) {
emit_signal ( SNAME ( " tab_changed " ) , 0 ) ;
}
2014-02-10 01:10:30 +00:00
}
2021-10-17 16:55:44 +00:00
void TabBar : : clear_tabs ( ) {
2022-03-02 14:37:10 +00:00
if ( tabs . is_empty ( ) ) {
return ;
}
2015-06-22 03:03:19 +00:00
tabs . clear ( ) ;
2022-01-19 16:11:44 +00:00
offset = 0 ;
max_drawn_tab = 0 ;
2017-03-05 15:44:50 +00:00
current = 0 ;
2020-09-08 20:30:47 +00:00
previous = 0 ;
2022-01-19 16:11:44 +00:00
2015-06-22 03:03:19 +00:00
update ( ) ;
2022-01-19 16:11:44 +00:00
update_minimum_size ( ) ;
2021-11-03 18:05:28 +00:00
notify_property_list_changed ( ) ;
2015-06-22 03:03:19 +00:00
}
2021-10-17 16:55:44 +00:00
void TabBar : : remove_tab ( int p_idx ) {
2017-03-05 15:44:50 +00:00
ERR_FAIL_INDEX ( p_idx , tabs . size ( ) ) ;
2021-07-03 22:17:03 +00:00
tabs . remove_at ( p_idx ) ;
2022-03-02 14:37:10 +00:00
bool is_tab_changing = current = = p_idx & & ! tabs . is_empty ( ) ;
if ( current > = p_idx & & current > 0 ) {
2014-02-10 01:10:30 +00:00
current - - ;
2020-05-14 14:41:43 +00:00
}
2014-02-10 01:10:30 +00:00
2022-03-02 14:37:10 +00:00
if ( tabs . is_empty ( ) ) {
2022-02-14 04:58:17 +00:00
offset = 0 ;
max_drawn_tab = 0 ;
2020-09-08 20:30:47 +00:00
previous = 0 ;
2022-02-14 04:58:17 +00:00
} else {
offset = MIN ( offset , tabs . size ( ) - 1 ) ;
max_drawn_tab = MIN ( max_drawn_tab , tabs . size ( ) - 1 ) ;
2014-02-10 01:10:30 +00:00
2022-02-14 04:58:17 +00:00
_update_cache ( ) ;
_ensure_no_over_offset ( ) ;
2022-03-02 14:37:10 +00:00
if ( scroll_to_selected ) {
2022-02-14 04:58:17 +00:00
ensure_tab_visible ( current ) ;
}
2022-01-19 16:11:44 +00:00
}
2022-02-14 04:58:17 +00:00
2022-01-19 16:11:44 +00:00
update ( ) ;
update_minimum_size ( ) ;
2021-11-03 18:05:28 +00:00
notify_property_list_changed ( ) ;
2022-03-02 14:37:10 +00:00
if ( is_tab_changing & & is_inside_tree ( ) ) {
emit_signal ( SNAME ( " tab_changed " ) , current ) ;
}
2015-06-22 03:03:19 +00:00
}
2021-10-17 16:55:44 +00:00
Variant TabBar : : get_drag_data ( const Point2 & p_point ) {
2020-05-14 14:41:43 +00:00
if ( ! drag_to_rearrange_enabled ) {
2022-03-02 14:37:10 +00:00
return Control : : get_drag_data ( p_point ) ; // Allow stuff like TabContainer to override it.
2020-05-14 14:41:43 +00:00
}
2018-02-07 13:01:45 +00:00
int tab_over = get_tab_idx_at_point ( p_point ) ;
2020-05-14 14:41:43 +00:00
if ( tab_over < 0 ) {
2018-02-07 13:01:45 +00:00
return Variant ( ) ;
2020-05-14 14:41:43 +00:00
}
2018-02-07 13:01:45 +00:00
HBoxContainer * drag_preview = memnew ( HBoxContainer ) ;
if ( ! tabs [ tab_over ] . icon . is_null ( ) ) {
TextureRect * tf = memnew ( TextureRect ) ;
tf - > set_texture ( tabs [ tab_over ] . icon ) ;
2022-01-19 16:11:44 +00:00
tf - > set_stretch_mode ( TextureRect : : STRETCH_KEEP_CENTERED ) ;
2018-02-07 13:01:45 +00:00
drag_preview - > add_child ( tf ) ;
}
2022-01-19 16:11:44 +00:00
2019-09-30 09:56:20 +00:00
Label * label = memnew ( Label ( tabs [ tab_over ] . xl_text ) ) ;
2018-02-07 13:01:45 +00:00
drag_preview - > add_child ( label ) ;
2022-01-19 16:11:44 +00:00
2018-02-07 13:01:45 +00:00
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 ( ) ;
2022-03-02 14:37:10 +00:00
2018-02-07 13:01:45 +00:00
return drag_data ;
2017-07-05 13:44:53 +00:00
}
2021-10-17 16:55:44 +00:00
bool TabBar : : can_drop_data ( const Point2 & p_point , const Variant & p_data ) const {
2020-05-14 14:41:43 +00:00
if ( ! drag_to_rearrange_enabled ) {
2022-03-02 14:37:10 +00:00
return Control : : can_drop_data ( p_point , p_data ) ; // Allow stuff like TabContainer to override it.
2020-05-14 14:41:43 +00:00
}
2018-02-07 13:01:45 +00:00
Dictionary d = p_data ;
2020-05-14 14:41:43 +00:00
if ( ! d . has ( " type " ) ) {
2018-02-07 13:01:45 +00:00
return false ;
2020-05-14 14:41:43 +00:00
}
2018-02-07 13:01:45 +00: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 ) {
2021-11-29 20:12:05 +00:00
// Drag and drop between other TabBars.
2018-02-07 13:01:45 +00:00
Node * from_node = get_node ( from_path ) ;
2021-10-17 16:55:44 +00:00
TabBar * from_tabs = Object : : cast_to < TabBar > ( from_node ) ;
2018-02-07 13:01:45 +00:00
if ( from_tabs & & from_tabs - > get_tabs_rearrange_group ( ) = = get_tabs_rearrange_group ( ) ) {
return true ;
}
}
}
2022-03-02 14:37:10 +00:00
2018-02-07 13:01:45 +00:00
return false ;
2017-07-05 13:44:53 +00:00
}
2021-10-17 16:55:44 +00:00
void TabBar : : drop_data ( const Point2 & p_point , const Variant & p_data ) {
2020-05-14 14:41:43 +00:00
if ( ! drag_to_rearrange_enabled ) {
2022-03-02 14:37:10 +00:00
Control : : drop_data ( p_point , p_data ) ; // Allow stuff like TabContainer to override it.
2018-02-07 13:01:45 +00:00
return ;
2020-05-14 14:41:43 +00:00
}
2018-02-07 13:01:45 +00:00
Dictionary d = p_data ;
2020-05-14 14:41:43 +00:00
if ( ! d . has ( " type " ) ) {
2018-02-07 13:01:45 +00:00
return ;
2020-05-14 14:41:43 +00:00
}
2018-02-07 13:01:45 +00:00
if ( String ( d [ " type " ] ) = = " tab_element " ) {
int tab_from_id = d [ " tab_element " ] ;
2022-03-02 14:37:10 +00:00
int hover_now = get_tab_idx_at_point ( p_point ) ;
2018-02-07 13:01:45 +00:00
NodePath from_path = d [ " from_path " ] ;
NodePath to_path = get_path ( ) ;
2022-01-19 16:11:44 +00:00
2018-02-07 13:01:45 +00:00
if ( from_path = = to_path ) {
2020-05-14 14:41:43 +00:00
if ( hover_now < 0 ) {
2018-02-07 13:01:45 +00:00
hover_now = get_tab_count ( ) - 1 ;
2020-05-14 14:41:43 +00:00
}
2022-01-19 16:11:44 +00:00
2018-02-07 13:01:45 +00:00
move_tab ( tab_from_id , hover_now ) ;
2021-09-30 23:18:23 +00:00
emit_signal ( SNAME ( " active_tab_rearranged " ) , hover_now ) ;
2018-02-07 13:01:45 +00:00
set_current_tab ( hover_now ) ;
} else if ( get_tabs_rearrange_group ( ) ! = - 1 ) {
2021-11-29 20:12:05 +00:00
// Drag and drop between Tabs.
2022-01-19 16:11:44 +00:00
2018-02-07 13:01:45 +00:00
Node * from_node = get_node ( from_path ) ;
2021-10-17 16:55:44 +00:00
TabBar * from_tabs = Object : : cast_to < TabBar > ( from_node ) ;
2022-01-19 16:11:44 +00:00
2018-02-07 13:01:45 +00:00
if ( from_tabs & & from_tabs - > get_tabs_rearrange_group ( ) = = get_tabs_rearrange_group ( ) ) {
2020-05-14 14:41:43 +00:00
if ( tab_from_id > = from_tabs - > get_tab_count ( ) ) {
2018-02-07 13:01:45 +00:00
return ;
2020-05-14 14:41:43 +00:00
}
2022-01-19 16:11:44 +00:00
2018-02-07 13:01:45 +00:00
Tab moving_tab = from_tabs - > tabs [ tab_from_id ] ;
2020-05-14 14:41:43 +00:00
if ( hover_now < 0 ) {
2018-02-07 13:01:45 +00:00
hover_now = get_tab_count ( ) ;
2020-05-14 14:41:43 +00:00
}
2022-01-19 16:11:44 +00:00
2022-03-02 14:37:10 +00:00
from_tabs - > remove_tab ( tab_from_id ) ;
tabs . insert ( hover_now , moving_tab ) ;
if ( tabs . size ( ) > 1 ) {
if ( current > = hover_now ) {
current + + ;
}
if ( previous > = hover_now ) {
previous + + ;
}
2022-01-19 16:11:44 +00:00
}
2018-02-07 13:01:45 +00:00
set_current_tab ( hover_now ) ;
2022-01-19 16:11:44 +00:00
update_minimum_size ( ) ;
2022-03-02 14:37:10 +00:00
if ( tabs . size ( ) = = 1 ) {
emit_signal ( SNAME ( " tab_selected " ) , 0 ) ;
emit_signal ( SNAME ( " tab_changed " ) , 0 ) ;
}
2018-02-07 13:01:45 +00:00
}
}
}
2017-07-05 13:44:53 +00:00
}
2021-10-17 16:55:44 +00:00
int TabBar : : get_tab_idx_at_point ( const Point2 & p_point ) const {
2017-07-05 13:44:53 +00:00
int hover_now = - 1 ;
2021-10-08 01:38:04 +00:00
for ( int i = offset ; i < = max_drawn_tab ; i + + ) {
2017-07-05 13:44:53 +00:00
Rect2 rect = get_tab_rect ( i ) ;
if ( rect . has_point ( p_point ) ) {
hover_now = i ;
}
}
return hover_now ;
}
2021-11-25 02:58:47 +00:00
void TabBar : : set_tab_alignment ( AlignmentMode p_alignment ) {
ERR_FAIL_INDEX ( p_alignment , ALIGNMENT_MAX ) ;
tab_alignment = p_alignment ;
2022-01-19 16:11:44 +00:00
2022-01-12 00:33:43 +00:00
_update_cache ( ) ;
2015-06-22 03:03:19 +00:00
update ( ) ;
}
2021-11-25 02:58:47 +00:00
TabBar : : AlignmentMode TabBar : : get_tab_alignment ( ) const {
return tab_alignment ;
2014-02-10 01:10:30 +00:00
}
2021-10-17 16:55:44 +00:00
void TabBar : : set_clip_tabs ( bool p_clip_tabs ) {
2021-03-30 10:20:00 +00:00
if ( clip_tabs = = p_clip_tabs ) {
return ;
}
clip_tabs = p_clip_tabs ;
2022-01-19 16:11:44 +00:00
if ( ! clip_tabs ) {
offset = 0 ;
max_drawn_tab = 0 ;
}
2022-01-12 00:33:43 +00:00
_update_cache ( ) ;
2022-01-19 16:11:44 +00:00
if ( scroll_to_selected ) {
ensure_tab_visible ( current ) ;
}
2021-03-30 10:20:00 +00:00
update ( ) ;
2021-12-06 13:02:34 +00:00
update_minimum_size ( ) ;
2021-03-30 10:20:00 +00:00
}
2021-10-17 16:55:44 +00:00
bool TabBar : : get_clip_tabs ( ) const {
2021-03-30 10:20:00 +00:00
return clip_tabs ;
}
2022-03-02 14:37:10 +00:00
void TabBar : : move_tab ( int p_from , int p_to ) {
if ( p_from = = p_to ) {
2017-07-12 14:48:43 +00:00
return ;
2020-05-14 14:41:43 +00:00
}
2017-07-12 14:48:43 +00:00
2022-03-02 14:37:10 +00:00
ERR_FAIL_INDEX ( p_from , tabs . size ( ) ) ;
ERR_FAIL_INDEX ( p_to , tabs . size ( ) ) ;
Tab tab_from = tabs [ p_from ] ;
tabs . remove_at ( p_from ) ;
tabs . insert ( p_to , tab_from ) ;
2017-07-12 14:48:43 +00:00
2022-03-02 14:37:10 +00:00
if ( current = = p_from ) {
current = p_to ;
} else if ( current > p_from & & current < = p_to ) {
current - - ;
} else if ( current < p_from & & current > = p_to ) {
current + + ;
}
if ( previous = = p_from ) {
previous = p_to ;
} else if ( previous > p_from & & previous > = p_to ) {
previous - - ;
} else if ( previous < p_from & & previous < = p_to ) {
previous + + ;
}
2017-07-12 14:48:43 +00:00
_update_cache ( ) ;
2022-01-19 16:11:44 +00:00
_ensure_no_over_offset ( ) ;
if ( scroll_to_selected ) {
ensure_tab_visible ( current ) ;
}
2017-07-12 14:48:43 +00:00
update ( ) ;
2021-11-03 18:05:28 +00:00
notify_property_list_changed ( ) ;
2017-07-12 14:48:43 +00:00
}
2021-10-17 16:55:44 +00:00
int TabBar : : get_tab_width ( int p_idx ) const {
2017-03-05 15:44:50 +00:00
ERR_FAIL_INDEX_V ( p_idx , tabs . size ( ) , 0 ) ;
2016-01-13 10:39:31 +00:00
2021-07-17 21:22:52 +00:00
Ref < StyleBox > tab_unselected = get_theme_stylebox ( SNAME ( " tab_unselected " ) ) ;
Ref < StyleBox > tab_selected = get_theme_stylebox ( SNAME ( " tab_selected " ) ) ;
Ref < StyleBox > tab_disabled = get_theme_stylebox ( SNAME ( " tab_disabled " ) ) ;
2022-01-19 16:11:44 +00:00
int hseparation = get_theme_constant ( SNAME ( " hseparation " ) ) ;
2016-05-01 14:27:33 +00:00
2022-01-19 16:11:44 +00:00
Ref < StyleBox > style ;
if ( tabs [ p_idx ] . disabled ) {
style = tab_disabled ;
} else if ( current = = p_idx ) {
style = tab_selected ;
} else {
style = tab_unselected ;
}
int x = style - > get_minimum_size ( ) . width ;
2016-01-13 10:39:31 +00:00
2019-06-11 18:43:37 +00:00
Ref < Texture2D > tex = tabs [ p_idx ] . icon ;
2016-01-13 10:39:31 +00:00
if ( tex . is_valid ( ) ) {
2022-01-19 16:11:44 +00:00
x + = tex - > get_width ( ) + hseparation ;
2016-01-13 10:39:31 +00:00
}
2022-01-19 16:11:44 +00:00
if ( ! tabs [ p_idx ] . text . is_empty ( ) ) {
x + = tabs [ p_idx ] . size_text + hseparation ;
2020-05-14 14:41:43 +00:00
}
2016-01-13 10:39:31 +00:00
2022-01-19 16:11:44 +00:00
bool close_visible = cb_displaypolicy = = CLOSE_BUTTON_SHOW_ALWAYS | | ( cb_displaypolicy = = CLOSE_BUTTON_SHOW_ACTIVE_ONLY & & p_idx = = current ) ;
2016-01-13 10:39:31 +00:00
if ( tabs [ p_idx ] . right_button . is_valid ( ) ) {
2022-01-19 16:11:44 +00:00
Ref < StyleBox > btn_style = get_theme_stylebox ( SNAME ( " button_highlight " ) ) ;
2019-06-11 18:43:37 +00:00
Ref < Texture2D > rb = tabs [ p_idx ] . right_button ;
2022-01-19 16:11:44 +00:00
if ( close_visible ) {
x + = btn_style - > get_minimum_size ( ) . width + rb - > get_width ( ) ;
} else {
x + = btn_style - > get_margin ( SIDE_LEFT ) + rb - > get_width ( ) + hseparation ;
}
2016-01-13 10:39:31 +00:00
}
2022-01-19 16:11:44 +00:00
if ( close_visible ) {
Ref < StyleBox > btn_style = get_theme_stylebox ( SNAME ( " button_highlight " ) ) ;
2021-07-17 21:22:52 +00:00
Ref < Texture2D > cb = get_theme_icon ( SNAME ( " close " ) ) ;
2022-01-19 16:11:44 +00:00
x + = btn_style - > get_margin ( SIDE_LEFT ) + cb - > get_width ( ) + hseparation ;
}
if ( x > style - > get_minimum_size ( ) . width ) {
x - = hseparation ;
2016-01-13 10:39:31 +00:00
}
return x ;
}
2014-02-10 01:10:30 +00:00
2021-10-17 16:55:44 +00:00
void TabBar : : _ensure_no_over_offset ( ) {
2022-01-12 00:33:43 +00:00
if ( ! is_inside_tree ( ) | | ! buttons_visible ) {
2016-01-22 23:19:57 +00:00
return ;
2020-05-14 14:41:43 +00:00
}
2016-01-22 23:19:57 +00:00
2021-07-17 21:22:52 +00:00
Ref < Texture2D > incr = get_theme_icon ( SNAME ( " increment " ) ) ;
Ref < Texture2D > decr = get_theme_icon ( SNAME ( " decrement " ) ) ;
2021-03-30 10:20:00 +00:00
int limit_minus_buttons = get_size ( ) . width - incr - > get_width ( ) - decr - > get_width ( ) ;
2016-01-22 23:19:57 +00:00
2022-01-12 00:33:43 +00:00
int prev_offset = offset ;
int total_w = tabs [ max_drawn_tab ] . ofs_cache + tabs [ max_drawn_tab ] . size_cache - tabs [ offset ] . ofs_cache ;
2022-01-19 16:11:44 +00:00
for ( int i = offset ; i > 0 ; i - - ) {
if ( tabs [ i - 1 ] . hidden ) {
continue ;
}
total_w + = tabs [ i - 1 ] . size_cache ;
2016-01-22 23:19:57 +00:00
2022-01-12 00:33:43 +00:00
if ( total_w < limit_minus_buttons ) {
2016-01-22 23:19:57 +00:00
offset - - ;
} else {
break ;
}
}
2022-01-12 00:33:43 +00:00
if ( prev_offset ! = offset ) {
_update_cache ( ) ;
update ( ) ;
2020-05-14 14:41:43 +00:00
}
2022-01-12 00:33:43 +00:00
}
2016-01-11 00:45:11 +00:00
2022-01-12 00:33:43 +00:00
void TabBar : : ensure_tab_visible ( int p_idx ) {
if ( ! is_inside_tree ( ) | | ! buttons_visible ) {
2020-05-10 10:56:01 +00:00
return ;
2020-05-14 14:41:43 +00:00
}
2017-03-05 15:44:50 +00:00
ERR_FAIL_INDEX ( p_idx , tabs . size ( ) ) ;
2016-01-11 00:45:11 +00:00
2022-01-19 16:11:44 +00:00
if ( tabs [ p_idx ] . hidden | | ( p_idx > = offset & & p_idx < = max_drawn_tab ) ) {
2017-06-15 15:30:03 +00:00
return ;
}
2022-01-12 00:33:43 +00:00
2017-06-15 15:30:03 +00:00
if ( p_idx < offset ) {
2017-03-05 15:44:50 +00:00
offset = p_idx ;
2022-01-12 00:33:43 +00:00
_update_cache ( ) ;
2016-01-11 00:45:11 +00:00
update ( ) ;
2022-01-12 00:33:43 +00:00
2016-01-11 00:45:11 +00:00
return ;
}
2021-07-17 21:22:52 +00:00
Ref < Texture2D > incr = get_theme_icon ( SNAME ( " increment " ) ) ;
Ref < Texture2D > decr = get_theme_icon ( SNAME ( " decrement " ) ) ;
2021-03-30 10:20:00 +00:00
int limit_minus_buttons = get_size ( ) . width - incr - > get_width ( ) - decr - > get_width ( ) ;
2022-01-12 00:33:43 +00:00
int total_w = tabs [ max_drawn_tab ] . ofs_cache - tabs [ offset ] . ofs_cache ;
for ( int i = max_drawn_tab ; i < = p_idx ; i + + ) {
2022-01-19 16:11:44 +00:00
if ( tabs [ i ] . hidden ) {
continue ;
}
2022-01-12 00:33:43 +00:00
total_w + = tabs [ i ] . size_cache ;
}
int prev_offset = offset ;
for ( int i = offset ; i < p_idx ; i + + ) {
2022-01-19 16:11:44 +00:00
if ( tabs [ i ] . hidden ) {
continue ;
}
2022-01-12 00:33:43 +00:00
if ( total_w > limit_minus_buttons ) {
total_w - = tabs [ i ] . size_cache ;
2017-06-15 15:30:03 +00:00
offset + + ;
2022-01-12 00:33:43 +00:00
} else {
break ;
2017-06-15 15:30:03 +00:00
}
2016-01-11 00:45:11 +00:00
}
2017-06-15 15:30:03 +00:00
if ( prev_offset ! = offset ) {
2022-01-12 00:33:43 +00:00
_update_cache ( ) ;
2017-06-15 15:30:03 +00:00
update ( ) ;
2016-01-11 00:45:11 +00:00
}
2017-06-15 15:30:03 +00:00
}
2016-01-11 00:45:11 +00:00
2021-10-17 16:55:44 +00:00
Rect2 TabBar : : get_tab_rect ( int p_tab ) const {
2019-06-21 09:34:32 +00:00
ERR_FAIL_INDEX_V ( p_tab , tabs . size ( ) , Rect2 ( ) ) ;
2020-09-03 11:22:16 +00: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 00:45:11 +00:00
}
2021-10-17 16:55:44 +00:00
void TabBar : : set_tab_close_display_policy ( CloseButtonDisplayPolicy p_policy ) {
2017-07-22 16:15:31 +00:00
ERR_FAIL_INDEX ( p_policy , CLOSE_BUTTON_MAX ) ;
2017-03-05 15:44:50 +00:00
cb_displaypolicy = p_policy ;
2022-01-19 16:11:44 +00:00
2022-01-12 00:33:43 +00:00
_update_cache ( ) ;
2022-01-19 16:11:44 +00:00
_ensure_no_over_offset ( ) ;
if ( scroll_to_selected ) {
ensure_tab_visible ( current ) ;
}
2016-01-13 10:39:31 +00:00
update ( ) ;
2022-01-19 16:11:44 +00:00
update_minimum_size ( ) ;
2016-01-13 10:39:31 +00:00
}
2021-10-17 16:55:44 +00:00
TabBar : : CloseButtonDisplayPolicy TabBar : : get_tab_close_display_policy ( ) const {
2017-07-22 16:15:31 +00:00
return cb_displaypolicy ;
}
2021-10-17 16:55:44 +00:00
void TabBar : : set_min_width ( int p_width ) {
2017-06-15 15:30:03 +00:00
min_width = p_width ;
}
2021-10-17 16:55:44 +00:00
void TabBar : : set_scrolling_enabled ( bool p_enabled ) {
2017-11-16 22:57:57 +00:00
scrolling_enabled = p_enabled ;
}
2021-10-17 16:55:44 +00:00
bool TabBar : : get_scrolling_enabled ( ) const {
2017-11-16 22:57:57 +00:00
return scrolling_enabled ;
}
2021-10-17 16:55:44 +00:00
void TabBar : : set_drag_to_rearrange_enabled ( bool p_enabled ) {
2018-02-07 13:01:45 +00:00
drag_to_rearrange_enabled = p_enabled ;
}
2021-10-17 16:55:44 +00:00
bool TabBar : : get_drag_to_rearrange_enabled ( ) const {
2018-02-07 13:01:45 +00:00
return drag_to_rearrange_enabled ;
}
2020-05-14 12:29:06 +00:00
2021-10-17 16:55:44 +00:00
void TabBar : : set_tabs_rearrange_group ( int p_group_id ) {
2018-02-07 13:01:45 +00:00
tabs_rearrange_group = p_group_id ;
}
2021-10-17 16:55:44 +00:00
int TabBar : : get_tabs_rearrange_group ( ) const {
2018-02-07 13:01:45 +00:00
return tabs_rearrange_group ;
}
2022-01-19 16:11:44 +00:00
void TabBar : : set_scroll_to_selected ( bool p_enabled ) {
scroll_to_selected = p_enabled ;
if ( p_enabled ) {
ensure_tab_visible ( current ) ;
}
}
bool TabBar : : get_scroll_to_selected ( ) const {
return scroll_to_selected ;
}
2021-10-17 16:55:44 +00:00
void TabBar : : set_select_with_rmb ( bool p_enabled ) {
2018-01-02 07:10:49 +00:00
select_with_rmb = p_enabled ;
}
2021-10-17 16:55:44 +00:00
bool TabBar : : get_select_with_rmb ( ) const {
2018-01-02 07:10:49 +00:00
return select_with_rmb ;
}
2021-11-03 18:05:28 +00:00
bool TabBar : : _set ( const StringName & p_name , const Variant & p_value ) {
Vector < String > components = String ( p_name ) . split ( " / " , true , 2 ) ;
if ( components . size ( ) > = 2 & & components [ 0 ] . begins_with ( " tab_ " ) & & components [ 0 ] . trim_prefix ( " tab_ " ) . is_valid_int ( ) ) {
int tab_index = components [ 0 ] . trim_prefix ( " tab_ " ) . to_int ( ) ;
String property = components [ 1 ] ;
if ( property = = " title " ) {
set_tab_title ( tab_index , p_value ) ;
return true ;
} else if ( property = = " icon " ) {
set_tab_icon ( tab_index , p_value ) ;
return true ;
} else if ( components [ 1 ] = = " disabled " ) {
set_tab_disabled ( tab_index , p_value ) ;
return true ;
}
}
return false ;
}
bool TabBar : : _get ( const StringName & p_name , Variant & r_ret ) const {
Vector < String > components = String ( p_name ) . split ( " / " , true , 2 ) ;
if ( components . size ( ) > = 2 & & components [ 0 ] . begins_with ( " tab_ " ) & & components [ 0 ] . trim_prefix ( " tab_ " ) . is_valid_int ( ) ) {
int tab_index = components [ 0 ] . trim_prefix ( " tab_ " ) . to_int ( ) ;
String property = components [ 1 ] ;
if ( property = = " title " ) {
r_ret = get_tab_title ( tab_index ) ;
return true ;
} else if ( property = = " icon " ) {
r_ret = get_tab_icon ( tab_index ) ;
return true ;
} else if ( components [ 1 ] = = " disabled " ) {
r_ret = is_tab_disabled ( tab_index ) ;
return true ;
}
}
return false ;
}
void TabBar : : _get_property_list ( List < PropertyInfo > * p_list ) const {
for ( int i = 0 ; i < tabs . size ( ) ; i + + ) {
p_list - > push_back ( PropertyInfo ( Variant : : STRING , vformat ( " tab_%d/title " , i ) ) ) ;
PropertyInfo pi = PropertyInfo ( Variant : : OBJECT , vformat ( " tab_%d/icon " , i ) , PROPERTY_HINT_RESOURCE_TYPE , " Texture2D " ) ;
pi . usage & = ~ ( get_tab_icon ( i ) . is_null ( ) ? PROPERTY_USAGE_STORAGE : 0 ) ;
p_list - > push_back ( pi ) ;
pi = PropertyInfo ( Variant : : BOOL , vformat ( " tab_%d/disabled " , i ) ) ;
pi . usage & = ~ ( ! is_tab_disabled ( i ) ? PROPERTY_USAGE_STORAGE : 0 ) ;
p_list - > push_back ( pi ) ;
}
}
2021-10-17 16:55:44 +00:00
void TabBar : : _bind_methods ( ) {
2022-01-07 12:58:28 +00:00
ClassDB : : bind_method ( D_METHOD ( " set_tab_count " , " count " ) , & TabBar : : set_tab_count ) ;
2021-10-17 16:55:44 +00:00
ClassDB : : bind_method ( D_METHOD ( " get_tab_count " ) , & TabBar : : get_tab_count ) ;
ClassDB : : bind_method ( D_METHOD ( " set_current_tab " , " tab_idx " ) , & TabBar : : set_current_tab ) ;
ClassDB : : bind_method ( D_METHOD ( " get_current_tab " ) , & TabBar : : get_current_tab ) ;
ClassDB : : bind_method ( D_METHOD ( " get_previous_tab " ) , & TabBar : : get_previous_tab ) ;
ClassDB : : bind_method ( D_METHOD ( " set_tab_title " , " tab_idx " , " title " ) , & TabBar : : set_tab_title ) ;
ClassDB : : bind_method ( D_METHOD ( " get_tab_title " , " tab_idx " ) , & TabBar : : get_tab_title ) ;
ClassDB : : bind_method ( D_METHOD ( " set_tab_text_direction " , " tab_idx " , " direction " ) , & TabBar : : set_tab_text_direction ) ;
ClassDB : : bind_method ( D_METHOD ( " get_tab_text_direction " , " tab_idx " ) , & TabBar : : get_tab_text_direction ) ;
ClassDB : : bind_method ( D_METHOD ( " set_tab_opentype_feature " , " tab_idx " , " tag " , " values " ) , & TabBar : : set_tab_opentype_feature ) ;
ClassDB : : bind_method ( D_METHOD ( " get_tab_opentype_feature " , " tab_idx " , " tag " ) , & TabBar : : get_tab_opentype_feature ) ;
ClassDB : : bind_method ( D_METHOD ( " clear_tab_opentype_features " , " tab_idx " ) , & TabBar : : clear_tab_opentype_features ) ;
ClassDB : : bind_method ( D_METHOD ( " set_tab_language " , " tab_idx " , " language " ) , & TabBar : : set_tab_language ) ;
ClassDB : : bind_method ( D_METHOD ( " get_tab_language " , " tab_idx " ) , & TabBar : : get_tab_language ) ;
ClassDB : : bind_method ( D_METHOD ( " set_tab_icon " , " tab_idx " , " icon " ) , & TabBar : : set_tab_icon ) ;
ClassDB : : bind_method ( D_METHOD ( " get_tab_icon " , " tab_idx " ) , & TabBar : : get_tab_icon ) ;
2022-01-19 16:11:44 +00:00
ClassDB : : bind_method ( D_METHOD ( " set_tab_button_icon " , " tab_idx " , " icon " ) , & TabBar : : set_tab_button_icon ) ;
ClassDB : : bind_method ( D_METHOD ( " get_tab_button_icon " , " tab_idx " ) , & TabBar : : get_tab_button_icon ) ;
2021-10-17 16:55:44 +00:00
ClassDB : : bind_method ( D_METHOD ( " set_tab_disabled " , " tab_idx " , " disabled " ) , & TabBar : : set_tab_disabled ) ;
2021-11-03 18:05:28 +00:00
ClassDB : : bind_method ( D_METHOD ( " is_tab_disabled " , " tab_idx " ) , & TabBar : : is_tab_disabled ) ;
2022-01-19 16:11:44 +00:00
ClassDB : : bind_method ( D_METHOD ( " set_tab_hidden " , " tab_idx " , " hidden " ) , & TabBar : : set_tab_hidden ) ;
ClassDB : : bind_method ( D_METHOD ( " is_tab_hidden " , " tab_idx " ) , & TabBar : : is_tab_hidden ) ;
2021-10-17 16:55:44 +00:00
ClassDB : : bind_method ( D_METHOD ( " remove_tab " , " tab_idx " ) , & TabBar : : remove_tab ) ;
ClassDB : : bind_method ( D_METHOD ( " add_tab " , " title " , " icon " ) , & TabBar : : add_tab , DEFVAL ( " " ) , DEFVAL ( Ref < Texture2D > ( ) ) ) ;
2022-03-02 14:37:10 +00:00
ClassDB : : bind_method ( D_METHOD ( " get_tab_idx_at_point " , " point " ) , & TabBar : : get_tab_idx_at_point ) ;
2021-11-25 02:58:47 +00:00
ClassDB : : bind_method ( D_METHOD ( " set_tab_alignment " , " alignment " ) , & TabBar : : set_tab_alignment ) ;
ClassDB : : bind_method ( D_METHOD ( " get_tab_alignment " ) , & TabBar : : get_tab_alignment ) ;
2021-10-17 16:55:44 +00:00
ClassDB : : bind_method ( D_METHOD ( " set_clip_tabs " , " clip_tabs " ) , & TabBar : : set_clip_tabs ) ;
ClassDB : : bind_method ( D_METHOD ( " get_clip_tabs " ) , & TabBar : : get_clip_tabs ) ;
ClassDB : : bind_method ( D_METHOD ( " get_tab_offset " ) , & TabBar : : get_tab_offset ) ;
ClassDB : : bind_method ( D_METHOD ( " get_offset_buttons_visible " ) , & TabBar : : get_offset_buttons_visible ) ;
ClassDB : : bind_method ( D_METHOD ( " ensure_tab_visible " , " idx " ) , & TabBar : : ensure_tab_visible ) ;
ClassDB : : bind_method ( D_METHOD ( " get_tab_rect " , " tab_idx " ) , & TabBar : : get_tab_rect ) ;
ClassDB : : bind_method ( D_METHOD ( " move_tab " , " from " , " to " ) , & TabBar : : move_tab ) ;
ClassDB : : bind_method ( D_METHOD ( " set_tab_close_display_policy " , " policy " ) , & TabBar : : set_tab_close_display_policy ) ;
ClassDB : : bind_method ( D_METHOD ( " get_tab_close_display_policy " ) , & TabBar : : get_tab_close_display_policy ) ;
ClassDB : : bind_method ( D_METHOD ( " set_scrolling_enabled " , " enabled " ) , & TabBar : : set_scrolling_enabled ) ;
ClassDB : : bind_method ( D_METHOD ( " get_scrolling_enabled " ) , & TabBar : : get_scrolling_enabled ) ;
ClassDB : : bind_method ( D_METHOD ( " set_drag_to_rearrange_enabled " , " enabled " ) , & TabBar : : set_drag_to_rearrange_enabled ) ;
ClassDB : : bind_method ( D_METHOD ( " get_drag_to_rearrange_enabled " ) , & TabBar : : get_drag_to_rearrange_enabled ) ;
ClassDB : : bind_method ( D_METHOD ( " set_tabs_rearrange_group " , " group_id " ) , & TabBar : : set_tabs_rearrange_group ) ;
ClassDB : : bind_method ( D_METHOD ( " get_tabs_rearrange_group " ) , & TabBar : : get_tabs_rearrange_group ) ;
2022-01-19 16:11:44 +00:00
ClassDB : : bind_method ( D_METHOD ( " set_scroll_to_selected " , " enabled " ) , & TabBar : : set_scroll_to_selected ) ;
ClassDB : : bind_method ( D_METHOD ( " get_scroll_to_selected " ) , & TabBar : : get_scroll_to_selected ) ;
2021-10-17 16:55:44 +00:00
ClassDB : : bind_method ( D_METHOD ( " set_select_with_rmb " , " enabled " ) , & TabBar : : set_select_with_rmb ) ;
ClassDB : : bind_method ( D_METHOD ( " get_select_with_rmb " ) , & TabBar : : get_select_with_rmb ) ;
2018-01-02 07:10:49 +00:00
2022-01-19 16:11:44 +00:00
ADD_SIGNAL ( MethodInfo ( " tab_selected " , PropertyInfo ( Variant : : INT , " tab " ) ) ) ;
2017-03-05 15:44:50 +00:00
ADD_SIGNAL ( MethodInfo ( " tab_changed " , PropertyInfo ( Variant : : INT , " tab " ) ) ) ;
2022-01-19 16:11:44 +00:00
ADD_SIGNAL ( MethodInfo ( " tab_clicked " , PropertyInfo ( Variant : : INT , " tab " ) ) ) ;
2021-09-30 23:18:23 +00:00
ADD_SIGNAL ( MethodInfo ( " tab_rmb_clicked " , PropertyInfo ( Variant : : INT , " tab " ) ) ) ;
2021-10-27 22:39:13 +00:00
ADD_SIGNAL ( MethodInfo ( " tab_close_pressed " , PropertyInfo ( Variant : : INT , " tab " ) ) ) ;
2022-01-19 16:11:44 +00:00
ADD_SIGNAL ( MethodInfo ( " tab_button_pressed " , PropertyInfo ( Variant : : INT , " tab " ) ) ) ;
2020-12-08 09:51:06 +00:00
ADD_SIGNAL ( MethodInfo ( " tab_hovered " , PropertyInfo ( Variant : : INT , " tab " ) ) ) ;
2021-09-30 23:18:23 +00:00
ADD_SIGNAL ( MethodInfo ( " active_tab_rearranged " , PropertyInfo ( Variant : : INT , " idx_to " ) ) ) ;
2014-02-10 01:10:30 +00:00
2017-03-05 15:44:50 +00:00
ADD_PROPERTY ( PropertyInfo ( Variant : : INT , " current_tab " , PROPERTY_HINT_RANGE , " -1,4096,1 " , PROPERTY_USAGE_EDITOR ) , " set_current_tab " , " get_current_tab " ) ;
2021-11-25 02:58:47 +00:00
ADD_PROPERTY ( PropertyInfo ( Variant : : INT , " tab_alignment " , PROPERTY_HINT_ENUM , " Left,Center,Right " ) , " set_tab_alignment " , " get_tab_alignment " ) ;
2021-03-30 10:20:00 +00:00
ADD_PROPERTY ( PropertyInfo ( Variant : : BOOL , " clip_tabs " ) , " set_clip_tabs " , " get_clip_tabs " ) ;
2018-11-08 14:30:02 +00: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 22:57:57 +00:00
ADD_PROPERTY ( PropertyInfo ( Variant : : BOOL , " scrolling_enabled " ) , " set_scrolling_enabled " , " get_scrolling_enabled " ) ;
2018-02-07 13:01:45 +00:00
ADD_PROPERTY ( PropertyInfo ( Variant : : BOOL , " drag_to_rearrange_enabled " ) , " set_drag_to_rearrange_enabled " , " get_drag_to_rearrange_enabled " ) ;
2022-01-19 16:11:44 +00:00
ADD_PROPERTY ( PropertyInfo ( Variant : : BOOL , " scroll_to_selected " ) , " set_scroll_to_selected " , " get_scroll_to_selected " ) ;
ADD_PROPERTY ( PropertyInfo ( Variant : : BOOL , " select_with_rmb " ) , " set_select_with_rmb " , " get_select_with_rmb " ) ;
2015-08-18 18:27:01 +00:00
2021-11-03 18:05:28 +00:00
ADD_ARRAY_COUNT ( " Tabs " , " tab_count " , " set_tab_count " , " get_tab_count " , " tab_ " ) ;
2021-11-25 02:58:47 +00:00
BIND_ENUM_CONSTANT ( ALIGNMENT_LEFT ) ;
BIND_ENUM_CONSTANT ( ALIGNMENT_CENTER ) ;
BIND_ENUM_CONSTANT ( ALIGNMENT_RIGHT ) ;
BIND_ENUM_CONSTANT ( ALIGNMENT_MAX ) ;
2017-08-20 15:45:01 +00:00
2017-10-21 18:58:02 +00:00
BIND_ENUM_CONSTANT ( CLOSE_BUTTON_SHOW_NEVER ) ;
2017-08-20 15:45:01 +00: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 01:10:30 +00:00
}
2021-10-17 16:55:44 +00:00
TabBar : : TabBar ( ) {
2022-01-19 16:11:44 +00:00
set_size ( Size2 ( get_size ( ) . width , get_minimum_size ( ) . height ) ) ;
2021-10-17 16:55:44 +00:00
connect ( " mouse_exited " , callable_mp ( this , & TabBar : : _on_mouse_exited ) ) ;
2014-02-10 01:10:30 +00:00
}