rich_text_label.h 12 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511
  1. /*************************************************************************/
  2. /* rich_text_label.h */
  3. /*************************************************************************/
  4. /* This file is part of: */
  5. /* GODOT ENGINE */
  6. /* https://godotengine.org */
  7. /*************************************************************************/
  8. /* Copyright (c) 2007-2022 Juan Linietsky, Ariel Manzur. */
  9. /* Copyright (c) 2014-2022 Godot Engine contributors (cf. AUTHORS.md). */
  10. /* */
  11. /* Permission is hereby granted, free of charge, to any person obtaining */
  12. /* a copy of this software and associated documentation files (the */
  13. /* "Software"), to deal in the Software without restriction, including */
  14. /* without limitation the rights to use, copy, modify, merge, publish, */
  15. /* distribute, sublicense, and/or sell copies of the Software, and to */
  16. /* permit persons to whom the Software is furnished to do so, subject to */
  17. /* the following conditions: */
  18. /* */
  19. /* The above copyright notice and this permission notice shall be */
  20. /* included in all copies or substantial portions of the Software. */
  21. /* */
  22. /* THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, */
  23. /* EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF */
  24. /* MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT.*/
  25. /* IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY */
  26. /* CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, */
  27. /* TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE */
  28. /* SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE. */
  29. /*************************************************************************/
  30. #ifndef RICH_TEXT_LABEL_H
  31. #define RICH_TEXT_LABEL_H
  32. #include "rich_text_effect.h"
  33. #include "scene/gui/scroll_bar.h"
  34. class RichTextLabel : public Control {
  35. GDCLASS(RichTextLabel, Control);
  36. public:
  37. enum Align {
  38. ALIGN_LEFT,
  39. ALIGN_CENTER,
  40. ALIGN_RIGHT,
  41. ALIGN_FILL
  42. };
  43. enum ListType {
  44. LIST_NUMBERS,
  45. LIST_LETTERS,
  46. LIST_DOTS
  47. };
  48. enum ItemType {
  49. ITEM_FRAME,
  50. ITEM_TEXT,
  51. ITEM_IMAGE,
  52. ITEM_NEWLINE,
  53. ITEM_FONT,
  54. ITEM_COLOR,
  55. ITEM_UNDERLINE,
  56. ITEM_STRIKETHROUGH,
  57. ITEM_ALIGN,
  58. ITEM_INDENT,
  59. ITEM_LIST,
  60. ITEM_TABLE,
  61. ITEM_FADE,
  62. ITEM_SHAKE,
  63. ITEM_WAVE,
  64. ITEM_TORNADO,
  65. ITEM_RAINBOW,
  66. ITEM_META,
  67. ITEM_CUSTOMFX
  68. };
  69. protected:
  70. static void _bind_methods();
  71. private:
  72. struct Item;
  73. struct Line {
  74. Item *from;
  75. Vector<int> offset_caches;
  76. Vector<int> height_caches;
  77. Vector<int> ascent_caches;
  78. Vector<int> descent_caches;
  79. Vector<int> space_caches;
  80. int height_cache;
  81. int height_accum_cache;
  82. int char_count;
  83. int minimum_width;
  84. int maximum_width;
  85. Line() {
  86. from = nullptr;
  87. char_count = 0;
  88. }
  89. };
  90. struct Item {
  91. int index;
  92. Item *parent;
  93. ItemType type;
  94. List<Item *> subitems;
  95. List<Item *>::Element *E;
  96. int line;
  97. void _clear_children() {
  98. while (subitems.size()) {
  99. memdelete(subitems.front()->get());
  100. subitems.pop_front();
  101. }
  102. }
  103. Item() {
  104. parent = nullptr;
  105. E = nullptr;
  106. line = 0;
  107. }
  108. virtual ~Item() { _clear_children(); }
  109. };
  110. struct ItemFrame : public Item {
  111. int parent_line;
  112. bool cell;
  113. Vector<Line> lines;
  114. int first_invalid_line;
  115. ItemFrame *parent_frame;
  116. ItemFrame() {
  117. type = ITEM_FRAME;
  118. parent_frame = nullptr;
  119. cell = false;
  120. parent_line = 0;
  121. }
  122. };
  123. struct ItemText : public Item {
  124. String text;
  125. ItemText() { type = ITEM_TEXT; }
  126. };
  127. struct ItemImage : public Item {
  128. Ref<Texture> image;
  129. Size2 size;
  130. ItemImage() { type = ITEM_IMAGE; }
  131. };
  132. struct ItemFont : public Item {
  133. Ref<Font> font;
  134. ItemFont() { type = ITEM_FONT; }
  135. };
  136. struct ItemColor : public Item {
  137. Color color;
  138. ItemColor() { type = ITEM_COLOR; }
  139. };
  140. struct ItemUnderline : public Item {
  141. ItemUnderline() { type = ITEM_UNDERLINE; }
  142. };
  143. struct ItemStrikethrough : public Item {
  144. ItemStrikethrough() { type = ITEM_STRIKETHROUGH; }
  145. };
  146. struct ItemMeta : public Item {
  147. Variant meta;
  148. ItemMeta() { type = ITEM_META; }
  149. };
  150. struct ItemAlign : public Item {
  151. Align align;
  152. ItemAlign() { type = ITEM_ALIGN; }
  153. };
  154. struct ItemIndent : public Item {
  155. int level;
  156. ItemIndent() { type = ITEM_INDENT; }
  157. };
  158. struct ItemList : public Item {
  159. ListType list_type;
  160. ItemList() { type = ITEM_LIST; }
  161. };
  162. struct ItemNewline : public Item {
  163. ItemNewline() { type = ITEM_NEWLINE; }
  164. };
  165. struct ItemTable : public Item {
  166. struct Column {
  167. bool expand;
  168. int expand_ratio;
  169. int min_width;
  170. int max_width;
  171. int width;
  172. };
  173. Vector<Column> columns;
  174. int total_width;
  175. ItemTable() { type = ITEM_TABLE; }
  176. };
  177. struct ItemFade : public Item {
  178. int starting_index;
  179. int length;
  180. ItemFade() { type = ITEM_FADE; }
  181. };
  182. struct ItemFX : public Item {
  183. float elapsed_time;
  184. ItemFX() {
  185. elapsed_time = 0.0f;
  186. }
  187. };
  188. struct ItemShake : public ItemFX {
  189. int strength;
  190. float rate;
  191. uint64_t _current_rng;
  192. uint64_t _previous_rng;
  193. ItemShake() {
  194. strength = 0;
  195. rate = 0.0f;
  196. _current_rng = 0;
  197. type = ITEM_SHAKE;
  198. }
  199. void reroll_random() {
  200. _previous_rng = _current_rng;
  201. _current_rng = Math::rand();
  202. }
  203. uint64_t offset_random(int index) {
  204. return (_current_rng >> (index % 64)) |
  205. (_current_rng << (64 - (index % 64)));
  206. }
  207. uint64_t offset_previous_random(int index) {
  208. return (_previous_rng >> (index % 64)) |
  209. (_previous_rng << (64 - (index % 64)));
  210. }
  211. };
  212. struct ItemWave : public ItemFX {
  213. float frequency;
  214. float amplitude;
  215. ItemWave() {
  216. frequency = 1.0f;
  217. amplitude = 1.0f;
  218. type = ITEM_WAVE;
  219. }
  220. };
  221. struct ItemTornado : public ItemFX {
  222. float radius;
  223. float frequency;
  224. ItemTornado() {
  225. radius = 1.0f;
  226. frequency = 1.0f;
  227. type = ITEM_TORNADO;
  228. }
  229. };
  230. struct ItemRainbow : public ItemFX {
  231. float saturation;
  232. float value;
  233. float frequency;
  234. ItemRainbow() {
  235. saturation = 0.8f;
  236. value = 0.8f;
  237. frequency = 1.0f;
  238. type = ITEM_RAINBOW;
  239. }
  240. };
  241. struct ItemCustomFX : public ItemFX {
  242. Ref<CharFXTransform> char_fx_transform;
  243. Ref<RichTextEffect> custom_effect;
  244. ItemCustomFX() {
  245. type = ITEM_CUSTOMFX;
  246. char_fx_transform.instance();
  247. }
  248. virtual ~ItemCustomFX() {
  249. _clear_children();
  250. char_fx_transform.unref();
  251. custom_effect.unref();
  252. }
  253. };
  254. ItemFrame *main;
  255. Item *current;
  256. ItemFrame *current_frame;
  257. VScrollBar *vscroll;
  258. bool scroll_visible;
  259. bool scroll_follow;
  260. bool scroll_following;
  261. bool scroll_active;
  262. int scroll_w;
  263. bool scroll_updated;
  264. bool updating_scroll;
  265. int current_idx;
  266. int visible_line_count;
  267. int tab_size;
  268. bool underline_meta;
  269. bool override_selected_font_color;
  270. Align default_align;
  271. ItemMeta *meta_hovering;
  272. Variant current_meta;
  273. Vector<Ref<RichTextEffect>> custom_effects;
  274. void _invalidate_current_line(ItemFrame *p_frame);
  275. void _validate_line_caches(ItemFrame *p_frame);
  276. void _add_item(Item *p_item, bool p_enter = false, bool p_ensure_newline = false);
  277. void _remove_item(Item *p_item, const int p_line, const int p_subitem_line);
  278. struct ProcessState {
  279. int line_width;
  280. };
  281. enum ProcessMode {
  282. PROCESS_CACHE,
  283. PROCESS_DRAW,
  284. PROCESS_POINTER
  285. };
  286. struct Selection {
  287. Item *click;
  288. int click_char;
  289. Item *from;
  290. int from_char;
  291. Item *to;
  292. int to_char;
  293. bool active; // anything selected? i.e. from, to, etc. valid?
  294. bool enabled; // allow selections?
  295. };
  296. Selection selection;
  297. int visible_characters;
  298. float percent_visible;
  299. int _process_line(ItemFrame *p_frame, const Vector2 &p_ofs, int &y, int p_width, int p_line, ProcessMode p_mode, const Ref<Font> &p_base_font, const Color &p_base_color, const Color &p_font_color_shadow, bool p_shadow_as_outline, const Point2 &shadow_ofs, const Point2i &p_click_pos = Point2i(), Item **r_click_item = nullptr, int *r_click_char = nullptr, bool *r_outside = nullptr, int p_char_count = 0);
  300. void _find_click(ItemFrame *p_frame, const Point2i &p_click, Item **r_click_item = nullptr, int *r_click_char = nullptr, bool *r_outside = nullptr);
  301. Ref<Font> _find_font(Item *p_item);
  302. int _find_margin(Item *p_item, const Ref<Font> &p_base_font);
  303. Align _find_align(Item *p_item);
  304. Color _find_color(Item *p_item, const Color &p_default_color);
  305. bool _find_underline(Item *p_item);
  306. bool _find_strikethrough(Item *p_item);
  307. bool _find_meta(Item *p_item, Variant *r_meta, ItemMeta **r_item = nullptr);
  308. bool _find_layout_subitem(Item *from, Item *to);
  309. bool _find_by_type(Item *p_item, ItemType p_type);
  310. void _fetch_item_fx_stack(Item *p_item, Vector<ItemFX *> &r_stack);
  311. void _update_scroll();
  312. void _update_fx(ItemFrame *p_frame, float p_delta_time);
  313. void _scroll_changed(double);
  314. void _gui_input(Ref<InputEvent> p_event);
  315. Item *_get_next_item(Item *p_item, bool p_free = false);
  316. Item *_get_prev_item(Item *p_item, bool p_free = false);
  317. Rect2 _get_text_rect();
  318. Ref<RichTextEffect> _get_custom_effect_by_code(String p_bbcode_identifier);
  319. virtual Dictionary parse_expressions_for_values(Vector<String> p_expressions);
  320. bool use_bbcode;
  321. String bbcode;
  322. void _update_all_lines();
  323. int fixed_width;
  324. bool fit_content_height;
  325. protected:
  326. void _notification(int p_what);
  327. public:
  328. String get_text();
  329. void add_text(const String &p_text);
  330. void add_image(const Ref<Texture> &p_image, const int p_width = 0, const int p_height = 0);
  331. void add_newline();
  332. bool remove_line(const int p_line);
  333. void push_font(const Ref<Font> &p_font);
  334. void push_normal();
  335. void push_bold();
  336. void push_bold_italics();
  337. void push_italics();
  338. void push_mono();
  339. void push_color(const Color &p_color);
  340. void push_underline();
  341. void push_strikethrough();
  342. void push_align(Align p_align);
  343. void push_indent(int p_level);
  344. void push_list(ListType p_list);
  345. void push_meta(const Variant &p_meta);
  346. void push_table(int p_columns);
  347. void push_fade(int p_start_index, int p_length);
  348. void push_shake(int p_strength, float p_rate);
  349. void push_wave(float p_frequency, float p_amplitude);
  350. void push_tornado(float p_frequency, float p_radius);
  351. void push_rainbow(float p_saturation, float p_value, float p_frequency);
  352. void push_customfx(Ref<RichTextEffect> p_custom_effect, Dictionary p_environment);
  353. void set_table_column_expand(int p_column, bool p_expand, int p_ratio = 1);
  354. int get_current_table_column() const;
  355. void push_cell();
  356. void pop();
  357. void clear();
  358. void set_offset(int p_pixel);
  359. void set_meta_underline(bool p_underline);
  360. bool is_meta_underlined() const;
  361. void set_override_selected_font_color(bool p_override_selected_font_color);
  362. bool is_overriding_selected_font_color() const;
  363. void set_scroll_active(bool p_active);
  364. bool is_scroll_active() const;
  365. void set_scroll_follow(bool p_follow);
  366. bool is_scroll_following() const;
  367. void set_tab_size(int p_spaces);
  368. int get_tab_size() const;
  369. void set_fit_content_height(bool p_enabled);
  370. bool is_fit_content_height_enabled() const;
  371. bool search(const String &p_string, bool p_from_selection = false, bool p_search_previous = false);
  372. void scroll_to_line(int p_line);
  373. int get_line_count() const;
  374. int get_visible_line_count() const;
  375. int get_content_height() const;
  376. VScrollBar *get_v_scroll() { return vscroll; }
  377. virtual CursorShape get_cursor_shape(const Point2 &p_pos) const;
  378. void set_selection_enabled(bool p_enabled);
  379. bool is_selection_enabled() const;
  380. String get_selected_text();
  381. void selection_copy();
  382. Error parse_bbcode(const String &p_bbcode);
  383. Error append_bbcode(const String &p_bbcode);
  384. void set_use_bbcode(bool p_enable);
  385. bool is_using_bbcode() const;
  386. void set_bbcode(const String &p_bbcode);
  387. String get_bbcode() const;
  388. void set_text(const String &p_string);
  389. void set_visible_characters(int p_visible);
  390. int get_visible_characters() const;
  391. int get_total_character_count() const;
  392. void set_percent_visible(float p_percent);
  393. float get_percent_visible() const;
  394. void set_effects(const Vector<Variant> &effects);
  395. Vector<Variant> get_effects();
  396. void install_effect(const Variant effect);
  397. void set_fixed_size_to_width(int p_width);
  398. virtual Size2 get_minimum_size() const;
  399. RichTextLabel();
  400. ~RichTextLabel();
  401. };
  402. VARIANT_ENUM_CAST(RichTextLabel::Align);
  403. VARIANT_ENUM_CAST(RichTextLabel::ListType);
  404. VARIANT_ENUM_CAST(RichTextLabel::ItemType);
  405. #endif // RICH_TEXT_LABEL_H