2020-09-03 19:22:16 +08:00
/**************************************************************************/
/* text_line.cpp */
/**************************************************************************/
/* This file is part of: */
/* GODOT ENGINE */
/* https://godotengine.org */
/**************************************************************************/
/* Copyright (c) 2014-present Godot Engine contributors (see AUTHORS.md). */
/* Copyright (c) 2007-2014 Juan Linietsky, Ariel Manzur. */
/* */
/* Permission is hereby granted, free of charge, to any person obtaining */
/* a copy of this software and associated documentation files (the */
/* "Software"), to deal in the Software without restriction, including */
/* without limitation the rights to use, copy, modify, merge, publish, */
/* distribute, sublicense, and/or sell copies of the Software, and to */
/* permit persons to whom the Software is furnished to do so, subject to */
/* the following conditions: */
/* */
/* The above copyright notice and this permission notice shall be */
/* included in all copies or substantial portions of the Software. */
/* */
/* THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, */
/* EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF */
/* MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. */
/* IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY */
/* CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, */
/* TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE */
/* SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE. */
/**************************************************************************/
# include "text_line.h"
void TextLine : : _bind_methods ( ) {
ClassDB : : bind_method ( D_METHOD ( " clear " ) , & TextLine : : clear ) ;
ClassDB : : bind_method ( D_METHOD ( " set_direction " , " direction " ) , & TextLine : : set_direction ) ;
ClassDB : : bind_method ( D_METHOD ( " get_direction " ) , & TextLine : : get_direction ) ;
2022-05-13 04:03:16 +08:00
ADD_PROPERTY ( PropertyInfo ( Variant : : INT , " direction " , PROPERTY_HINT_ENUM , " Auto,Left-to-right,Right-to-left " ) , " set_direction " , " get_direction " ) ;
2020-09-03 19:22:16 +08:00
ClassDB : : bind_method ( D_METHOD ( " set_orientation " , " orientation " ) , & TextLine : : set_orientation ) ;
ClassDB : : bind_method ( D_METHOD ( " get_orientation " ) , & TextLine : : get_orientation ) ;
ADD_PROPERTY ( PropertyInfo ( Variant : : INT , " orientation " , PROPERTY_HINT_ENUM , " Horizontal,Orientation " ) , " set_orientation " , " get_orientation " ) ;
ClassDB : : bind_method ( D_METHOD ( " set_preserve_invalid " , " enabled " ) , & TextLine : : set_preserve_invalid ) ;
ClassDB : : bind_method ( D_METHOD ( " get_preserve_invalid " ) , & TextLine : : get_preserve_invalid ) ;
ADD_PROPERTY ( PropertyInfo ( Variant : : BOOL , " preserve_invalid " ) , " set_preserve_invalid " , " get_preserve_invalid " ) ;
ClassDB : : bind_method ( D_METHOD ( " set_preserve_control " , " enabled " ) , & TextLine : : set_preserve_control ) ;
ClassDB : : bind_method ( D_METHOD ( " get_preserve_control " ) , & TextLine : : get_preserve_control ) ;
ADD_PROPERTY ( PropertyInfo ( Variant : : BOOL , " preserve_control " ) , " set_preserve_control " , " get_preserve_control " ) ;
2021-08-28 05:19:51 +08:00
ClassDB : : bind_method ( D_METHOD ( " set_bidi_override " , " override " ) , & TextLine : : set_bidi_override ) ;
2020-09-03 19:22:16 +08:00
2022-05-09 17:47:10 +08:00
ClassDB : : bind_method ( D_METHOD ( " add_string " , " text " , " font " , " font_size " , " language " , " meta " ) , & TextLine : : add_string , DEFVAL ( " " ) , DEFVAL ( Variant ( ) ) ) ;
2022-12-09 04:01:07 +08:00
ClassDB : : bind_method ( D_METHOD ( " add_object " , " key " , " size " , " inline_align " , " length " , " baseline " ) , & TextLine : : add_object , DEFVAL ( INLINE_ALIGNMENT_CENTER ) , DEFVAL ( 1 ) , DEFVAL ( 0.0 ) ) ;
ClassDB : : bind_method ( D_METHOD ( " resize_object " , " key " , " size " , " inline_align " , " baseline " ) , & TextLine : : resize_object , DEFVAL ( INLINE_ALIGNMENT_CENTER ) , DEFVAL ( 0.0 ) ) ;
2020-09-03 19:22:16 +08:00
ClassDB : : bind_method ( D_METHOD ( " set_width " , " width " ) , & TextLine : : set_width ) ;
ClassDB : : bind_method ( D_METHOD ( " get_width " ) , & TextLine : : get_width ) ;
ADD_PROPERTY ( PropertyInfo ( Variant : : FLOAT , " width " ) , " set_width " , " get_width " ) ;
2021-11-25 10:58:47 +08:00
ClassDB : : bind_method ( D_METHOD ( " set_horizontal_alignment " , " alignment " ) , & TextLine : : set_horizontal_alignment ) ;
ClassDB : : bind_method ( D_METHOD ( " get_horizontal_alignment " ) , & TextLine : : get_horizontal_alignment ) ;
2020-09-03 19:22:16 +08:00
2021-11-25 10:58:47 +08:00
ADD_PROPERTY ( PropertyInfo ( Variant : : INT , " alignment " , PROPERTY_HINT_ENUM , " Left,Center,Right,Fill " ) , " set_horizontal_alignment " , " get_horizontal_alignment " ) ;
2020-09-03 19:22:16 +08:00
ClassDB : : bind_method ( D_METHOD ( " tab_align " , " tab_stops " ) , & TextLine : : tab_align ) ;
ClassDB : : bind_method ( D_METHOD ( " set_flags " , " flags " ) , & TextLine : : set_flags ) ;
ClassDB : : bind_method ( D_METHOD ( " get_flags " ) , & TextLine : : get_flags ) ;
2022-07-21 19:37:00 +08:00
ADD_PROPERTY ( PropertyInfo ( Variant : : INT , " flags " , PROPERTY_HINT_FLAGS , " Kashida Justification,Word Justification,Trim Edge Spaces After Justification,Justify Only After Last Tab,Constrain Ellipsis " ) , " set_flags " , " get_flags " ) ;
2020-09-03 19:22:16 +08:00
2021-08-11 06:09:48 +08:00
ClassDB : : bind_method ( D_METHOD ( " set_text_overrun_behavior " , " overrun_behavior " ) , & TextLine : : set_text_overrun_behavior ) ;
ClassDB : : bind_method ( D_METHOD ( " get_text_overrun_behavior " ) , & TextLine : : get_text_overrun_behavior ) ;
ADD_PROPERTY ( PropertyInfo ( Variant : : INT , " text_overrun_behavior " , PROPERTY_HINT_ENUM , " Trim Nothing,Trim Characters,Trim Words,Ellipsis,Word Ellipsis " ) , " set_text_overrun_behavior " , " get_text_overrun_behavior " ) ;
2023-10-01 18:39:13 +08:00
ClassDB : : bind_method ( D_METHOD ( " set_ellipsis_char " , " char " ) , & TextLine : : set_ellipsis_char ) ;
ClassDB : : bind_method ( D_METHOD ( " get_ellipsis_char " ) , & TextLine : : get_ellipsis_char ) ;
ADD_PROPERTY ( PropertyInfo ( Variant : : STRING , " ellipsis_char " ) , " set_ellipsis_char " , " get_ellipsis_char " ) ;
2020-09-03 19:22:16 +08:00
ClassDB : : bind_method ( D_METHOD ( " get_objects " ) , & TextLine : : get_objects ) ;
ClassDB : : bind_method ( D_METHOD ( " get_object_rect " , " key " ) , & TextLine : : get_object_rect ) ;
ClassDB : : bind_method ( D_METHOD ( " get_size " ) , & TextLine : : get_size ) ;
ClassDB : : bind_method ( D_METHOD ( " get_rid " ) , & TextLine : : get_rid ) ;
ClassDB : : bind_method ( D_METHOD ( " get_line_ascent " ) , & TextLine : : get_line_ascent ) ;
ClassDB : : bind_method ( D_METHOD ( " get_line_descent " ) , & TextLine : : get_line_descent ) ;
ClassDB : : bind_method ( D_METHOD ( " get_line_width " ) , & TextLine : : get_line_width ) ;
ClassDB : : bind_method ( D_METHOD ( " get_line_underline_position " ) , & TextLine : : get_line_underline_position ) ;
ClassDB : : bind_method ( D_METHOD ( " get_line_underline_thickness " ) , & TextLine : : get_line_underline_thickness ) ;
ClassDB : : bind_method ( D_METHOD ( " draw " , " canvas " , " pos " , " color " ) , & TextLine : : draw , DEFVAL ( Color ( 1 , 1 , 1 ) ) ) ;
ClassDB : : bind_method ( D_METHOD ( " draw_outline " , " canvas " , " pos " , " outline_size " , " color " ) , & TextLine : : draw_outline , DEFVAL ( 1 ) , DEFVAL ( Color ( 1 , 1 , 1 ) ) ) ;
ClassDB : : bind_method ( D_METHOD ( " hit_test " , " coords " ) , & TextLine : : hit_test ) ;
}
void TextLine : : _shape ( ) {
2023-01-28 10:31:44 +08:00
// When a shaped text is invalidated by an external source, we want to reshape it.
if ( ! TS - > shaped_text_is_ready ( rid ) ) {
dirty = true ;
}
2020-09-03 19:22:16 +08:00
if ( dirty ) {
2020-12-15 20:04:21 +08:00
if ( ! tab_stops . is_empty ( ) ) {
2020-09-03 19:22:16 +08:00
TS - > shaped_text_tab_align ( rid , tab_stops ) ;
}
2021-08-11 06:09:48 +08:00
2022-07-11 17:40:31 +08:00
BitField < TextServer : : TextOverrunFlag > overrun_flags = TextServer : : OVERRUN_NO_TRIM ;
2022-06-15 16:01:45 +08:00
if ( overrun_behavior ! = TextServer : : OVERRUN_NO_TRIMMING ) {
2021-08-11 06:09:48 +08:00
switch ( overrun_behavior ) {
2022-06-15 16:01:45 +08:00
case TextServer : : OVERRUN_TRIM_WORD_ELLIPSIS :
2022-07-11 17:40:31 +08:00
overrun_flags . set_flag ( TextServer : : OVERRUN_TRIM ) ;
overrun_flags . set_flag ( TextServer : : OVERRUN_TRIM_WORD_ONLY ) ;
overrun_flags . set_flag ( TextServer : : OVERRUN_ADD_ELLIPSIS ) ;
2021-08-11 06:09:48 +08:00
break ;
2022-06-15 16:01:45 +08:00
case TextServer : : OVERRUN_TRIM_ELLIPSIS :
2022-07-11 17:40:31 +08:00
overrun_flags . set_flag ( TextServer : : OVERRUN_TRIM ) ;
overrun_flags . set_flag ( TextServer : : OVERRUN_ADD_ELLIPSIS ) ;
2021-08-11 06:09:48 +08:00
break ;
2022-06-15 16:01:45 +08:00
case TextServer : : OVERRUN_TRIM_WORD :
2022-07-11 17:40:31 +08:00
overrun_flags . set_flag ( TextServer : : OVERRUN_TRIM ) ;
overrun_flags . set_flag ( TextServer : : OVERRUN_TRIM_WORD_ONLY ) ;
2021-08-11 06:09:48 +08:00
break ;
2022-06-15 16:01:45 +08:00
case TextServer : : OVERRUN_TRIM_CHAR :
2022-07-11 17:40:31 +08:00
overrun_flags . set_flag ( TextServer : : OVERRUN_TRIM ) ;
2021-08-11 06:09:48 +08:00
break ;
2022-06-15 16:01:45 +08:00
case TextServer : : OVERRUN_NO_TRIMMING :
2021-08-11 06:09:48 +08:00
break ;
}
2021-11-25 10:58:47 +08:00
if ( alignment = = HORIZONTAL_ALIGNMENT_FILL ) {
2021-08-11 06:09:48 +08:00
TS - > shaped_text_fit_to_width ( rid , width , flags ) ;
2022-07-11 17:40:31 +08:00
overrun_flags . set_flag ( TextServer : : OVERRUN_JUSTIFICATION_AWARE ) ;
2023-10-01 18:39:13 +08:00
TS - > shaped_text_set_custom_ellipsis ( rid , ( el_char . length ( ) > 0 ) ? el_char [ 0 ] : 0x2026 ) ;
2021-08-11 06:09:48 +08:00
TS - > shaped_text_overrun_trim_to_width ( rid , width , overrun_flags ) ;
} else {
2023-10-01 18:39:13 +08:00
TS - > shaped_text_set_custom_ellipsis ( rid , ( el_char . length ( ) > 0 ) ? el_char [ 0 ] : 0x2026 ) ;
2021-08-11 06:09:48 +08:00
TS - > shaped_text_overrun_trim_to_width ( rid , width , overrun_flags ) ;
}
2021-11-25 10:58:47 +08:00
} else if ( alignment = = HORIZONTAL_ALIGNMENT_FILL ) {
2020-09-03 19:22:16 +08:00
TS - > shaped_text_fit_to_width ( rid , width , flags ) ;
}
dirty = false ;
}
}
RID TextLine : : get_rid ( ) const {
return rid ;
}
void TextLine : : clear ( ) {
TS - > shaped_text_clear ( rid ) ;
}
void TextLine : : set_preserve_invalid ( bool p_enabled ) {
TS - > shaped_text_set_preserve_invalid ( rid , p_enabled ) ;
dirty = true ;
}
bool TextLine : : get_preserve_invalid ( ) const {
return TS - > shaped_text_get_preserve_invalid ( rid ) ;
}
void TextLine : : set_preserve_control ( bool p_enabled ) {
TS - > shaped_text_set_preserve_control ( rid , p_enabled ) ;
dirty = true ;
}
bool TextLine : : get_preserve_control ( ) const {
return TS - > shaped_text_get_preserve_control ( rid ) ;
}
void TextLine : : set_direction ( TextServer : : Direction p_direction ) {
TS - > shaped_text_set_direction ( rid , p_direction ) ;
dirty = true ;
}
TextServer : : Direction TextLine : : get_direction ( ) const {
return TS - > shaped_text_get_direction ( rid ) ;
}
void TextLine : : set_orientation ( TextServer : : Orientation p_orientation ) {
TS - > shaped_text_set_orientation ( rid , p_orientation ) ;
dirty = true ;
}
TextServer : : Orientation TextLine : : get_orientation ( ) const {
return TS - > shaped_text_get_orientation ( rid ) ;
}
2021-08-28 05:19:51 +08:00
void TextLine : : set_bidi_override ( const Array & p_override ) {
2020-09-03 19:22:16 +08:00
TS - > shaped_text_set_bidi_override ( rid , p_override ) ;
dirty = true ;
}
2022-05-09 17:47:10 +08:00
bool TextLine : : add_string ( const String & p_text , const Ref < Font > & p_font , int p_font_size , const String & p_language , const Variant & p_meta ) {
ERR_FAIL_COND_V ( p_font . is_null ( ) , false ) ;
bool res = TS - > shaped_text_add_string ( rid , p_text , p_font - > get_rids ( ) , p_font_size , p_font - > get_opentype_features ( ) , p_language , p_meta ) ;
2020-09-03 19:22:16 +08:00
dirty = true ;
return res ;
}
2022-12-09 04:01:07 +08:00
bool TextLine : : add_object ( Variant p_key , const Size2 & p_size , InlineAlignment p_inline_align , int p_length , float p_baseline ) {
bool res = TS - > shaped_text_add_object ( rid , p_key , p_size , p_inline_align , p_length , p_baseline ) ;
2020-09-03 19:22:16 +08:00
dirty = true ;
return res ;
}
2022-12-09 04:01:07 +08:00
bool TextLine : : resize_object ( Variant p_key , const Size2 & p_size , InlineAlignment p_inline_align , float p_baseline ) {
2020-09-03 19:22:16 +08:00
const_cast < TextLine * > ( this ) - > _shape ( ) ;
2022-12-09 04:01:07 +08:00
return TS - > shaped_text_resize_object ( rid , p_key , p_size , p_inline_align , p_baseline ) ;
2020-09-03 19:22:16 +08:00
}
Array TextLine : : get_objects ( ) const {
return TS - > shaped_text_get_objects ( rid ) ;
}
Rect2 TextLine : : get_object_rect ( Variant p_key ) const {
2022-10-26 15:46:08 +08:00
Vector2 ofs ;
float length = TS - > shaped_text_get_width ( rid ) ;
if ( width > 0 ) {
switch ( alignment ) {
case HORIZONTAL_ALIGNMENT_FILL :
case HORIZONTAL_ALIGNMENT_LEFT :
break ;
case HORIZONTAL_ALIGNMENT_CENTER : {
if ( length < = width ) {
if ( TS - > shaped_text_get_orientation ( rid ) = = TextServer : : ORIENTATION_HORIZONTAL ) {
ofs . x + = Math : : floor ( ( width - length ) / 2.0 ) ;
} else {
ofs . y + = Math : : floor ( ( width - length ) / 2.0 ) ;
}
} else if ( TS - > shaped_text_get_inferred_direction ( rid ) = = TextServer : : DIRECTION_RTL ) {
if ( TS - > shaped_text_get_orientation ( rid ) = = TextServer : : ORIENTATION_HORIZONTAL ) {
ofs . x + = width - length ;
} else {
ofs . y + = width - length ;
}
}
} break ;
case HORIZONTAL_ALIGNMENT_RIGHT : {
if ( TS - > shaped_text_get_orientation ( rid ) = = TextServer : : ORIENTATION_HORIZONTAL ) {
ofs . x + = width - length ;
} else {
ofs . y + = width - length ;
}
} break ;
}
}
if ( TS - > shaped_text_get_orientation ( rid ) = = TextServer : : ORIENTATION_HORIZONTAL ) {
ofs . y + = TS - > shaped_text_get_ascent ( rid ) ;
} else {
ofs . x + = TS - > shaped_text_get_ascent ( rid ) ;
}
Rect2 rect = TS - > shaped_text_get_object_rect ( rid , p_key ) ;
rect . position + = ofs ;
return rect ;
2020-09-03 19:22:16 +08:00
}
2021-11-25 10:58:47 +08:00
void TextLine : : set_horizontal_alignment ( HorizontalAlignment p_alignment ) {
if ( alignment ! = p_alignment ) {
if ( alignment = = HORIZONTAL_ALIGNMENT_FILL | | p_alignment = = HORIZONTAL_ALIGNMENT_FILL ) {
alignment = p_alignment ;
2020-09-03 19:22:16 +08:00
dirty = true ;
} else {
2021-11-25 10:58:47 +08:00
alignment = p_alignment ;
2020-09-03 19:22:16 +08:00
}
}
}
2021-11-25 10:58:47 +08:00
HorizontalAlignment TextLine : : get_horizontal_alignment ( ) const {
return alignment ;
2020-09-03 19:22:16 +08:00
}
void TextLine : : tab_align ( const Vector < float > & p_tab_stops ) {
tab_stops = p_tab_stops ;
dirty = true ;
}
2022-07-11 17:40:31 +08:00
void TextLine : : set_flags ( BitField < TextServer : : JustificationFlag > p_flags ) {
2020-09-03 19:22:16 +08:00
if ( flags ! = p_flags ) {
flags = p_flags ;
dirty = true ;
}
}
2022-07-11 17:40:31 +08:00
BitField < TextServer : : JustificationFlag > TextLine : : get_flags ( ) const {
2020-09-03 19:22:16 +08:00
return flags ;
}
2022-06-15 16:01:45 +08:00
void TextLine : : set_text_overrun_behavior ( TextServer : : OverrunBehavior p_behavior ) {
2021-08-11 06:09:48 +08:00
if ( overrun_behavior ! = p_behavior ) {
overrun_behavior = p_behavior ;
dirty = true ;
}
}
2022-06-15 16:01:45 +08:00
TextServer : : OverrunBehavior TextLine : : get_text_overrun_behavior ( ) const {
2021-08-11 06:09:48 +08:00
return overrun_behavior ;
}
2023-10-01 18:39:13 +08:00
void TextLine : : set_ellipsis_char ( const String & p_char ) {
String c = p_char ;
if ( c . length ( ) > 1 ) {
WARN_PRINT ( " Ellipsis must be exactly one character long ( " + itos ( c . length ( ) ) + " characters given). " ) ;
c = c . left ( 1 ) ;
}
if ( el_char = = c ) {
return ;
}
el_char = c ;
dirty = true ;
}
String TextLine : : get_ellipsis_char ( ) const {
return el_char ;
}
2020-09-03 19:22:16 +08:00
void TextLine : : set_width ( float p_width ) {
width = p_width ;
2022-06-15 16:01:45 +08:00
if ( alignment = = HORIZONTAL_ALIGNMENT_FILL | | overrun_behavior ! = TextServer : : OVERRUN_NO_TRIMMING ) {
2020-09-03 19:22:16 +08:00
dirty = true ;
}
}
float TextLine : : get_width ( ) const {
return width ;
}
Size2 TextLine : : get_size ( ) const {
const_cast < TextLine * > ( this ) - > _shape ( ) ;
2022-11-04 06:57:07 +08:00
return TS - > shaped_text_get_size ( rid ) ;
2020-09-03 19:22:16 +08:00
}
float TextLine : : get_line_ascent ( ) const {
const_cast < TextLine * > ( this ) - > _shape ( ) ;
2022-05-09 17:47:10 +08:00
return TS - > shaped_text_get_ascent ( rid ) ;
2020-09-03 19:22:16 +08:00
}
float TextLine : : get_line_descent ( ) const {
const_cast < TextLine * > ( this ) - > _shape ( ) ;
2022-05-09 17:47:10 +08:00
return TS - > shaped_text_get_descent ( rid ) ;
2020-09-03 19:22:16 +08:00
}
float TextLine : : get_line_width ( ) const {
const_cast < TextLine * > ( this ) - > _shape ( ) ;
return TS - > shaped_text_get_width ( rid ) ;
}
float TextLine : : get_line_underline_position ( ) const {
const_cast < TextLine * > ( this ) - > _shape ( ) ;
return TS - > shaped_text_get_underline_position ( rid ) ;
}
float TextLine : : get_line_underline_thickness ( ) const {
const_cast < TextLine * > ( this ) - > _shape ( ) ;
return TS - > shaped_text_get_underline_thickness ( rid ) ;
}
void TextLine : : draw ( RID p_canvas , const Vector2 & p_pos , const Color & p_color ) const {
const_cast < TextLine * > ( this ) - > _shape ( ) ;
Vector2 ofs = p_pos ;
float length = TS - > shaped_text_get_width ( rid ) ;
if ( width > 0 ) {
2021-11-25 10:58:47 +08:00
switch ( alignment ) {
case HORIZONTAL_ALIGNMENT_FILL :
case HORIZONTAL_ALIGNMENT_LEFT :
2020-09-03 19:22:16 +08:00
break ;
2021-11-25 10:58:47 +08:00
case HORIZONTAL_ALIGNMENT_CENTER : {
2022-04-04 14:04:05 +08:00
if ( length < = width ) {
if ( TS - > shaped_text_get_orientation ( rid ) = = TextServer : : ORIENTATION_HORIZONTAL ) {
ofs . x + = Math : : floor ( ( width - length ) / 2.0 ) ;
} else {
ofs . y + = Math : : floor ( ( width - length ) / 2.0 ) ;
}
} else if ( TS - > shaped_text_get_inferred_direction ( rid ) = = TextServer : : DIRECTION_RTL ) {
if ( TS - > shaped_text_get_orientation ( rid ) = = TextServer : : ORIENTATION_HORIZONTAL ) {
ofs . x + = width - length ;
} else {
ofs . y + = width - length ;
}
2020-09-03 19:22:16 +08:00
}
} break ;
2021-11-25 10:58:47 +08:00
case HORIZONTAL_ALIGNMENT_RIGHT : {
2020-09-03 19:22:16 +08:00
if ( TS - > shaped_text_get_orientation ( rid ) = = TextServer : : ORIENTATION_HORIZONTAL ) {
ofs . x + = width - length ;
} else {
ofs . y + = width - length ;
}
} break ;
}
}
float clip_l ;
if ( TS - > shaped_text_get_orientation ( rid ) = = TextServer : : ORIENTATION_HORIZONTAL ) {
2022-05-09 17:47:10 +08:00
ofs . y + = TS - > shaped_text_get_ascent ( rid ) ;
2020-09-03 19:22:16 +08:00
clip_l = MAX ( 0 , p_pos . x - ofs . x ) ;
} else {
2022-05-09 17:47:10 +08:00
ofs . x + = TS - > shaped_text_get_ascent ( rid ) ;
2020-09-03 19:22:16 +08:00
clip_l = MAX ( 0 , p_pos . y - ofs . y ) ;
}
return TS - > shaped_text_draw ( rid , p_canvas , ofs , clip_l , clip_l + width , p_color ) ;
}
void TextLine : : draw_outline ( RID p_canvas , const Vector2 & p_pos , int p_outline_size , const Color & p_color ) const {
const_cast < TextLine * > ( this ) - > _shape ( ) ;
Vector2 ofs = p_pos ;
float length = TS - > shaped_text_get_width ( rid ) ;
if ( width > 0 ) {
2021-11-25 10:58:47 +08:00
switch ( alignment ) {
case HORIZONTAL_ALIGNMENT_FILL :
case HORIZONTAL_ALIGNMENT_LEFT :
2020-09-03 19:22:16 +08:00
break ;
2021-11-25 10:58:47 +08:00
case HORIZONTAL_ALIGNMENT_CENTER : {
2022-04-04 14:04:05 +08:00
if ( length < = width ) {
if ( TS - > shaped_text_get_orientation ( rid ) = = TextServer : : ORIENTATION_HORIZONTAL ) {
ofs . x + = Math : : floor ( ( width - length ) / 2.0 ) ;
} else {
ofs . y + = Math : : floor ( ( width - length ) / 2.0 ) ;
}
} else if ( TS - > shaped_text_get_inferred_direction ( rid ) = = TextServer : : DIRECTION_RTL ) {
if ( TS - > shaped_text_get_orientation ( rid ) = = TextServer : : ORIENTATION_HORIZONTAL ) {
ofs . x + = width - length ;
} else {
ofs . y + = width - length ;
}
2020-09-03 19:22:16 +08:00
}
} break ;
2021-11-25 10:58:47 +08:00
case HORIZONTAL_ALIGNMENT_RIGHT : {
2020-09-03 19:22:16 +08:00
if ( TS - > shaped_text_get_orientation ( rid ) = = TextServer : : ORIENTATION_HORIZONTAL ) {
ofs . x + = width - length ;
} else {
ofs . y + = width - length ;
}
} break ;
}
}
float clip_l ;
if ( TS - > shaped_text_get_orientation ( rid ) = = TextServer : : ORIENTATION_HORIZONTAL ) {
2022-05-09 17:47:10 +08:00
ofs . y + = TS - > shaped_text_get_ascent ( rid ) ;
2020-09-03 19:22:16 +08:00
clip_l = MAX ( 0 , p_pos . x - ofs . x ) ;
} else {
2022-05-09 17:47:10 +08:00
ofs . x + = TS - > shaped_text_get_ascent ( rid ) ;
2020-09-03 19:22:16 +08:00
clip_l = MAX ( 0 , p_pos . y - ofs . y ) ;
}
return TS - > shaped_text_draw_outline ( rid , p_canvas , ofs , clip_l , clip_l + width , p_outline_size , p_color ) ;
}
int TextLine : : hit_test ( float p_coords ) const {
const_cast < TextLine * > ( this ) - > _shape ( ) ;
return TS - > shaped_text_hit_test_position ( rid , p_coords ) ;
}
2022-05-09 17:47:10 +08:00
TextLine : : TextLine ( const String & p_text , const Ref < Font > & p_font , int p_font_size , const String & p_language , TextServer : : Direction p_direction , TextServer : : Orientation p_orientation ) {
2020-09-03 19:22:16 +08:00
rid = TS - > create_shaped_text ( p_direction , p_orientation ) ;
2022-05-09 17:47:10 +08:00
if ( p_font . is_valid ( ) ) {
TS - > shaped_text_add_string ( rid , p_text , p_font - > get_rids ( ) , p_font_size , p_font - > get_opentype_features ( ) , p_language ) ;
}
2020-09-03 19:22:16 +08:00
}
TextLine : : TextLine ( ) {
rid = TS - > create_shaped_text ( ) ;
}
TextLine : : ~ TextLine ( ) {
2022-02-13 20:41:29 +08:00
TS - > free_rid ( rid ) ;
2020-09-03 19:22:16 +08:00
}