Separate underscore from grapheme punctuation to enable doubleclick and caret jump over snakecase variables in editor
This commit is contained in:
parent
9e0e74e58e
commit
8b91828e4f
7 changed files with 26 additions and 12 deletions
|
@ -150,7 +150,7 @@ typedef struct {
|
||||||
godot_packed_glyph_array (*shaped_text_sort_logical)(void *, godot_rid *);
|
godot_packed_glyph_array (*shaped_text_sort_logical)(void *, godot_rid *);
|
||||||
godot_packed_vector2i_array (*shaped_text_get_line_breaks_adv)(void *, godot_rid *, godot_packed_float32_array *, int, bool, uint8_t);
|
godot_packed_vector2i_array (*shaped_text_get_line_breaks_adv)(void *, godot_rid *, godot_packed_float32_array *, int, bool, uint8_t);
|
||||||
godot_packed_vector2i_array (*shaped_text_get_line_breaks)(void *, godot_rid *, float, int, uint8_t);
|
godot_packed_vector2i_array (*shaped_text_get_line_breaks)(void *, godot_rid *, float, int, uint8_t);
|
||||||
godot_packed_vector2i_array (*shaped_text_get_word_breaks)(void *, godot_rid *);
|
godot_packed_vector2i_array (*shaped_text_get_word_breaks)(void *, godot_rid *, int);
|
||||||
godot_array (*shaped_text_get_objects)(void *, godot_rid *);
|
godot_array (*shaped_text_get_objects)(void *, godot_rid *);
|
||||||
godot_rect2 (*shaped_text_get_object_rect)(void *, godot_rid *, const godot_variant *);
|
godot_rect2 (*shaped_text_get_object_rect)(void *, godot_rid *, const godot_variant *);
|
||||||
godot_vector2 (*shaped_text_get_size)(void *, godot_rid *);
|
godot_vector2 (*shaped_text_get_size)(void *, godot_rid *);
|
||||||
|
|
|
@ -555,15 +555,15 @@ Vector<Vector2i> TextServerGDNative::shaped_text_get_line_breaks(RID p_shaped, f
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
Vector<Vector2i> TextServerGDNative::shaped_text_get_word_breaks(RID p_shaped) const {
|
Vector<Vector2i> TextServerGDNative::shaped_text_get_word_breaks(RID p_shaped, int p_grapheme_flags) const {
|
||||||
ERR_FAIL_COND_V(interface == nullptr, Vector<Vector2i>());
|
ERR_FAIL_COND_V(interface == nullptr, Vector<Vector2i>());
|
||||||
if (interface->shaped_text_get_word_breaks != nullptr) {
|
if (interface->shaped_text_get_word_breaks != nullptr) {
|
||||||
godot_packed_vector2i_array result = interface->shaped_text_get_word_breaks(data, (godot_rid *)&p_shaped);
|
godot_packed_vector2i_array result = interface->shaped_text_get_word_breaks(data, (godot_rid *)&p_shaped, p_grapheme_flags);
|
||||||
Vector<Vector2i> breaks = *(Vector<Vector2i> *)&result;
|
Vector<Vector2i> breaks = *(Vector<Vector2i> *)&result;
|
||||||
godot_packed_vector2i_array_destroy(&result);
|
godot_packed_vector2i_array_destroy(&result);
|
||||||
return breaks;
|
return breaks;
|
||||||
} else {
|
} else {
|
||||||
return TextServer::shaped_text_get_word_breaks(p_shaped);
|
return TextServer::shaped_text_get_word_breaks(p_shaped, p_grapheme_flags);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -178,7 +178,7 @@ public:
|
||||||
virtual Vector<Glyph> shaped_text_sort_logical(RID p_shaped) override;
|
virtual Vector<Glyph> shaped_text_sort_logical(RID p_shaped) override;
|
||||||
virtual Vector<Vector2i> shaped_text_get_line_breaks_adv(RID p_shaped, const Vector<float> &p_width, int p_start = 0, bool p_once = true, uint8_t /*TextBreakFlag*/ p_break_flags = BREAK_MANDATORY | BREAK_WORD_BOUND) const override;
|
virtual Vector<Vector2i> shaped_text_get_line_breaks_adv(RID p_shaped, const Vector<float> &p_width, int p_start = 0, bool p_once = true, uint8_t /*TextBreakFlag*/ p_break_flags = BREAK_MANDATORY | BREAK_WORD_BOUND) const override;
|
||||||
virtual Vector<Vector2i> shaped_text_get_line_breaks(RID p_shaped, float p_width, int p_start = 0, uint8_t p_break_flags = BREAK_MANDATORY | BREAK_WORD_BOUND) const override;
|
virtual Vector<Vector2i> shaped_text_get_line_breaks(RID p_shaped, float p_width, int p_start = 0, uint8_t p_break_flags = BREAK_MANDATORY | BREAK_WORD_BOUND) const override;
|
||||||
virtual Vector<Vector2i> shaped_text_get_word_breaks(RID p_shaped) const override;
|
virtual Vector<Vector2i> shaped_text_get_word_breaks(RID p_shaped, int p_grapheme_flags = GRAPHEME_IS_SPACE | GRAPHEME_IS_PUNCTUATION) const override;
|
||||||
virtual Array shaped_text_get_objects(RID p_shaped) const override;
|
virtual Array shaped_text_get_objects(RID p_shaped) const override;
|
||||||
virtual Rect2 shaped_text_get_object_rect(RID p_shaped, Variant p_key) const override;
|
virtual Rect2 shaped_text_get_object_rect(RID p_shaped, Variant p_key) const override;
|
||||||
|
|
||||||
|
|
|
@ -129,6 +129,10 @@ _FORCE_INLINE_ bool is_linebreak(char32_t p_char) {
|
||||||
return (p_char >= 0x000a && p_char <= 0x000d) || (p_char == 0x0085) || (p_char == 0x2028) || (p_char == 0x2029);
|
return (p_char >= 0x000a && p_char <= 0x000d) || (p_char == 0x0085) || (p_char == 0x2028) || (p_char == 0x2029);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
_FORCE_INLINE_ bool is_underscore(char32_t p_char) {
|
||||||
|
return (p_char == 0x005F);
|
||||||
|
}
|
||||||
|
|
||||||
/*************************************************************************/
|
/*************************************************************************/
|
||||||
|
|
||||||
String TextServerAdvanced::interface_name = "ICU / HarfBuzz / Graphite";
|
String TextServerAdvanced::interface_name = "ICU / HarfBuzz / Graphite";
|
||||||
|
@ -1883,7 +1887,10 @@ bool TextServerAdvanced::shaped_text_update_breaks(RID p_shaped) {
|
||||||
if (is_whitespace(c)) {
|
if (is_whitespace(c)) {
|
||||||
sd_glyphs[i].flags |= GRAPHEME_IS_SPACE;
|
sd_glyphs[i].flags |= GRAPHEME_IS_SPACE;
|
||||||
}
|
}
|
||||||
if (u_ispunct(c)) {
|
if (is_underscore(c)) {
|
||||||
|
sd_glyphs[i].flags |= GRAPHEME_IS_UNDERSCORE;
|
||||||
|
}
|
||||||
|
if (u_ispunct(c) && c != 0x005F) {
|
||||||
sd_glyphs[i].flags |= GRAPHEME_IS_PUNCTUATION;
|
sd_glyphs[i].flags |= GRAPHEME_IS_PUNCTUATION;
|
||||||
}
|
}
|
||||||
if (breaks.has(sd->glyphs[i].start)) {
|
if (breaks.has(sd->glyphs[i].start)) {
|
||||||
|
|
|
@ -46,7 +46,11 @@ _FORCE_INLINE_ bool is_linebreak(char32_t p_char) {
|
||||||
}
|
}
|
||||||
|
|
||||||
_FORCE_INLINE_ bool is_punct(char32_t p_char) {
|
_FORCE_INLINE_ bool is_punct(char32_t p_char) {
|
||||||
return (p_char >= 0x0020 && p_char <= 0x002F) || (p_char >= 0x003A && p_char <= 0x0040) || (p_char >= 0x005B && p_char <= 0x0060) || (p_char >= 0x007B && p_char <= 0x007E) || (p_char >= 0x2000 && p_char <= 0x206F) || (p_char >= 0x3000 && p_char <= 0x303F);
|
return (p_char >= 0x0020 && p_char <= 0x002F) || (p_char >= 0x003A && p_char <= 0x0040) || (p_char >= 0x005B && p_char <= 0x005E) || (p_char == 0x0060) || (p_char >= 0x007B && p_char <= 0x007E) || (p_char >= 0x2000 && p_char <= 0x206F) || (p_char >= 0x3000 && p_char <= 0x303F);
|
||||||
|
}
|
||||||
|
|
||||||
|
_FORCE_INLINE_ bool is_underscore(char32_t p_char) {
|
||||||
|
return (p_char == 0x005F);
|
||||||
}
|
}
|
||||||
|
|
||||||
/*************************************************************************/
|
/*************************************************************************/
|
||||||
|
@ -1108,6 +1112,9 @@ bool TextServerFallback::shaped_text_update_breaks(RID p_shaped) {
|
||||||
if (is_punct(c)) {
|
if (is_punct(c)) {
|
||||||
sd->glyphs.write[i].flags |= GRAPHEME_IS_PUNCTUATION;
|
sd->glyphs.write[i].flags |= GRAPHEME_IS_PUNCTUATION;
|
||||||
}
|
}
|
||||||
|
if (is_underscore(c)) {
|
||||||
|
sd->glyphs.write[i].flags |= GRAPHEME_IS_UNDERSCORE;
|
||||||
|
}
|
||||||
if (is_whitespace(c) && !is_linebreak(c)) {
|
if (is_whitespace(c) && !is_linebreak(c)) {
|
||||||
sd->glyphs.write[i].flags |= GRAPHEME_IS_SPACE;
|
sd->glyphs.write[i].flags |= GRAPHEME_IS_SPACE;
|
||||||
sd->glyphs.write[i].flags |= GRAPHEME_IS_BREAK_SOFT;
|
sd->glyphs.write[i].flags |= GRAPHEME_IS_BREAK_SOFT;
|
||||||
|
|
|
@ -712,7 +712,7 @@ Vector<Vector2i> TextServer::shaped_text_get_line_breaks(RID p_shaped, float p_w
|
||||||
return lines;
|
return lines;
|
||||||
}
|
}
|
||||||
|
|
||||||
Vector<Vector2i> TextServer::shaped_text_get_word_breaks(RID p_shaped) const {
|
Vector<Vector2i> TextServer::shaped_text_get_word_breaks(RID p_shaped, int p_grapheme_flags) const {
|
||||||
Vector<Vector2i> words;
|
Vector<Vector2i> words;
|
||||||
|
|
||||||
const_cast<TextServer *>(this)->shaped_text_update_justification_ops(p_shaped);
|
const_cast<TextServer *>(this)->shaped_text_update_justification_ops(p_shaped);
|
||||||
|
@ -726,7 +726,7 @@ Vector<Vector2i> TextServer::shaped_text_get_word_breaks(RID p_shaped) const {
|
||||||
|
|
||||||
for (int i = 0; i < l_size; i++) {
|
for (int i = 0; i < l_size; i++) {
|
||||||
if (l_gl[i].count > 0) {
|
if (l_gl[i].count > 0) {
|
||||||
if (((l_gl[i].flags & GRAPHEME_IS_SPACE) == GRAPHEME_IS_SPACE) || ((l_gl[i].flags & GRAPHEME_IS_PUNCTUATION) == GRAPHEME_IS_PUNCTUATION)) {
|
if ((l_gl[i].flags & p_grapheme_flags) != 0) {
|
||||||
words.push_back(Vector2i(word_start, l_gl[i].start));
|
words.push_back(Vector2i(word_start, l_gl[i].start));
|
||||||
word_start = l_gl[i].end;
|
word_start = l_gl[i].end;
|
||||||
}
|
}
|
||||||
|
|
|
@ -87,7 +87,8 @@ public:
|
||||||
GRAPHEME_IS_BREAK_SOFT = 1 << 5, // Is line break (optional break, e.g. space).
|
GRAPHEME_IS_BREAK_SOFT = 1 << 5, // Is line break (optional break, e.g. space).
|
||||||
GRAPHEME_IS_TAB = 1 << 6, // Is tab or vertical tab.
|
GRAPHEME_IS_TAB = 1 << 6, // Is tab or vertical tab.
|
||||||
GRAPHEME_IS_ELONGATION = 1 << 7, // Elongation (e.g. kashida), glyph can be duplicated or truncated to fit line to width.
|
GRAPHEME_IS_ELONGATION = 1 << 7, // Elongation (e.g. kashida), glyph can be duplicated or truncated to fit line to width.
|
||||||
GRAPHEME_IS_PUNCTUATION = 1 << 8 // Punctuation (can be used as word break, but not line break or justifiction).
|
GRAPHEME_IS_PUNCTUATION = 1 << 8, // Punctuation, except underscore (can be used as word break, but not line break or justifiction).
|
||||||
|
GRAPHEME_IS_UNDERSCORE = 1 << 9, // Underscore (can be used as word break).
|
||||||
};
|
};
|
||||||
|
|
||||||
enum Hinting {
|
enum Hinting {
|
||||||
|
@ -354,10 +355,9 @@ public:
|
||||||
|
|
||||||
virtual Vector<Vector2i> shaped_text_get_line_breaks_adv(RID p_shaped, const Vector<float> &p_width, int p_start = 0, bool p_once = true, uint8_t /*TextBreakFlag*/ p_break_flags = BREAK_MANDATORY | BREAK_WORD_BOUND) const;
|
virtual Vector<Vector2i> shaped_text_get_line_breaks_adv(RID p_shaped, const Vector<float> &p_width, int p_start = 0, bool p_once = true, uint8_t /*TextBreakFlag*/ p_break_flags = BREAK_MANDATORY | BREAK_WORD_BOUND) const;
|
||||||
virtual Vector<Vector2i> shaped_text_get_line_breaks(RID p_shaped, float p_width, int p_start = 0, uint8_t /*TextBreakFlag*/ p_break_flags = BREAK_MANDATORY | BREAK_WORD_BOUND) const;
|
virtual Vector<Vector2i> shaped_text_get_line_breaks(RID p_shaped, float p_width, int p_start = 0, uint8_t /*TextBreakFlag*/ p_break_flags = BREAK_MANDATORY | BREAK_WORD_BOUND) const;
|
||||||
virtual Vector<Vector2i> shaped_text_get_word_breaks(RID p_shaped) const;
|
virtual Vector<Vector2i> shaped_text_get_word_breaks(RID p_shaped, int p_grapheme_flags = GRAPHEME_IS_SPACE | GRAPHEME_IS_PUNCTUATION) const;
|
||||||
|
|
||||||
virtual void shaped_text_overrun_trim_to_width(RID p_shaped, float p_width, uint8_t p_clip_flags) = 0;
|
virtual void shaped_text_overrun_trim_to_width(RID p_shaped, float p_width, uint8_t p_clip_flags) = 0;
|
||||||
|
|
||||||
virtual Array shaped_text_get_objects(RID p_shaped) const = 0;
|
virtual Array shaped_text_get_objects(RID p_shaped) const = 0;
|
||||||
virtual Rect2 shaped_text_get_object_rect(RID p_shaped, Variant p_key) const = 0;
|
virtual Rect2 shaped_text_get_object_rect(RID p_shaped, Variant p_key) const = 0;
|
||||||
|
|
||||||
|
|
Loading…
Reference in a new issue