code_editor.cpp 43 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972973974975976977978979980981982983984985986987988989990991992993994995996997998999100010011002100310041005100610071008100910101011101210131014101510161017101810191020102110221023102410251026102710281029103010311032103310341035103610371038103910401041104210431044104510461047104810491050105110521053105410551056105710581059106010611062106310641065106610671068106910701071107210731074107510761077107810791080108110821083108410851086108710881089109010911092109310941095109610971098109911001101110211031104110511061107110811091110111111121113111411151116111711181119112011211122112311241125112611271128112911301131113211331134113511361137113811391140114111421143114411451146114711481149115011511152115311541155115611571158115911601161116211631164116511661167116811691170117111721173117411751176117711781179118011811182118311841185118611871188118911901191119211931194119511961197119811991200120112021203120412051206120712081209121012111212121312141215121612171218121912201221122212231224122512261227122812291230123112321233123412351236123712381239124012411242124312441245124612471248124912501251125212531254125512561257125812591260126112621263126412651266126712681269127012711272127312741275127612771278127912801281128212831284128512861287128812891290129112921293129412951296129712981299130013011302130313041305130613071308130913101311131213131314131513161317131813191320132113221323132413251326132713281329133013311332133313341335133613371338133913401341134213431344134513461347134813491350135113521353135413551356135713581359136013611362136313641365136613671368136913701371137213731374137513761377
  1. /*************************************************************************/
  2. /* code_editor.cpp */
  3. /*************************************************************************/
  4. /* This file is part of: */
  5. /* GODOT ENGINE */
  6. /* https://godotengine.org */
  7. /*************************************************************************/
  8. /* Copyright (c) 2007-2019 Juan Linietsky, Ariel Manzur. */
  9. /* Copyright (c) 2014-2019 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. #include "code_editor.h"
  31. #include "core/os/keyboard.h"
  32. #include "editor/editor_scale.h"
  33. #include "editor_node.h"
  34. #include "editor_settings.h"
  35. #include "scene/gui/margin_container.h"
  36. #include "scene/gui/separator.h"
  37. #include "scene/resources/dynamic_font.h"
  38. void GotoLineDialog::popup_find_line(TextEdit *p_edit) {
  39. text_editor = p_edit;
  40. line->set_text(itos(text_editor->cursor_get_line()));
  41. line->select_all();
  42. popup_centered(Size2(180, 80) * EDSCALE);
  43. line->grab_focus();
  44. }
  45. int GotoLineDialog::get_line() const {
  46. return line->get_text().to_int();
  47. }
  48. void GotoLineDialog::ok_pressed() {
  49. if (get_line() < 1 || get_line() > text_editor->get_line_count())
  50. return;
  51. text_editor->unfold_line(get_line() - 1);
  52. text_editor->cursor_set_line(get_line() - 1);
  53. hide();
  54. }
  55. GotoLineDialog::GotoLineDialog() {
  56. set_title(TTR("Go to Line"));
  57. VBoxContainer *vbc = memnew(VBoxContainer);
  58. vbc->set_anchor_and_margin(MARGIN_LEFT, ANCHOR_BEGIN, 8 * EDSCALE);
  59. vbc->set_anchor_and_margin(MARGIN_TOP, ANCHOR_BEGIN, 8 * EDSCALE);
  60. vbc->set_anchor_and_margin(MARGIN_RIGHT, ANCHOR_END, -8 * EDSCALE);
  61. vbc->set_anchor_and_margin(MARGIN_BOTTOM, ANCHOR_END, -8 * EDSCALE);
  62. add_child(vbc);
  63. Label *l = memnew(Label);
  64. l->set_text(TTR("Line Number:"));
  65. vbc->add_child(l);
  66. line = memnew(LineEdit);
  67. vbc->add_child(line);
  68. register_text_enter(line);
  69. text_editor = NULL;
  70. set_hide_on_ok(false);
  71. }
  72. void FindReplaceBar::_notification(int p_what) {
  73. if (p_what == NOTIFICATION_READY) {
  74. find_prev->set_icon(get_icon("MoveUp", "EditorIcons"));
  75. find_next->set_icon(get_icon("MoveDown", "EditorIcons"));
  76. hide_button->set_normal_texture(get_icon("Close", "EditorIcons"));
  77. hide_button->set_hover_texture(get_icon("Close", "EditorIcons"));
  78. hide_button->set_pressed_texture(get_icon("Close", "EditorIcons"));
  79. hide_button->set_custom_minimum_size(hide_button->get_normal_texture()->get_size());
  80. } else if (p_what == NOTIFICATION_VISIBILITY_CHANGED) {
  81. set_process_unhandled_input(is_visible_in_tree());
  82. if (is_visible_in_tree()) {
  83. _update_size();
  84. }
  85. } else if (p_what == EditorSettings::NOTIFICATION_EDITOR_SETTINGS_CHANGED) {
  86. find_prev->set_icon(get_icon("MoveUp", "EditorIcons"));
  87. find_next->set_icon(get_icon("MoveDown", "EditorIcons"));
  88. hide_button->set_normal_texture(get_icon("Close", "EditorIcons"));
  89. hide_button->set_hover_texture(get_icon("Close", "EditorIcons"));
  90. hide_button->set_pressed_texture(get_icon("Close", "EditorIcons"));
  91. hide_button->set_custom_minimum_size(hide_button->get_normal_texture()->get_size());
  92. }
  93. }
  94. void FindReplaceBar::_unhandled_input(const Ref<InputEvent> &p_event) {
  95. Ref<InputEventKey> k = p_event;
  96. if (k.is_valid()) {
  97. if (k->is_pressed() && (text_edit->has_focus() || vbc_lineedit->is_a_parent_of(get_focus_owner()))) {
  98. bool accepted = true;
  99. switch (k->get_scancode()) {
  100. case KEY_ESCAPE: {
  101. _hide_bar();
  102. } break;
  103. default: {
  104. accepted = false;
  105. } break;
  106. }
  107. if (accepted) {
  108. accept_event();
  109. }
  110. }
  111. }
  112. }
  113. bool FindReplaceBar::_search(uint32_t p_flags, int p_from_line, int p_from_col) {
  114. int line, col;
  115. String text = get_search_text();
  116. bool found = text_edit->search(text, p_flags, p_from_line, p_from_col, line, col);
  117. if (found) {
  118. if (!preserve_cursor) {
  119. text_edit->unfold_line(line);
  120. text_edit->cursor_set_line(line, false);
  121. text_edit->cursor_set_column(col + text.length(), false);
  122. text_edit->center_viewport_to_cursor();
  123. }
  124. text_edit->set_search_text(text);
  125. text_edit->set_search_flags(p_flags);
  126. text_edit->set_current_search_result(line, col);
  127. result_line = line;
  128. result_col = col;
  129. set_error("");
  130. } else {
  131. result_line = -1;
  132. result_col = -1;
  133. text_edit->set_search_text("");
  134. set_error(text.empty() ? "" : TTR("No Matches"));
  135. }
  136. return found;
  137. }
  138. void FindReplaceBar::_replace() {
  139. if (result_line != -1 && result_col != -1) {
  140. text_edit->begin_complex_operation();
  141. text_edit->unfold_line(result_line);
  142. text_edit->select(result_line, result_col, result_line, result_col + get_search_text().length());
  143. text_edit->insert_text_at_cursor(get_replace_text());
  144. text_edit->end_complex_operation();
  145. }
  146. search_current();
  147. }
  148. void FindReplaceBar::_replace_all() {
  149. text_edit->disconnect("text_changed", this, "_editor_text_changed");
  150. // line as x so it gets priority in comparison, column as y
  151. Point2i orig_cursor(text_edit->cursor_get_line(), text_edit->cursor_get_column());
  152. Point2i prev_match = Point2(-1, -1);
  153. bool selection_enabled = text_edit->is_selection_active();
  154. Point2i selection_begin, selection_end;
  155. if (selection_enabled) {
  156. selection_begin = Point2i(text_edit->get_selection_from_line(), text_edit->get_selection_from_column());
  157. selection_end = Point2i(text_edit->get_selection_to_line(), text_edit->get_selection_to_column());
  158. }
  159. int vsval = text_edit->get_v_scroll();
  160. text_edit->cursor_set_line(0);
  161. text_edit->cursor_set_column(0);
  162. String replace_text = get_replace_text();
  163. int search_text_len = get_search_text().length();
  164. int rc = 0;
  165. replace_all_mode = true;
  166. text_edit->begin_complex_operation();
  167. if (search_current()) {
  168. do {
  169. // replace area
  170. Point2i match_from(result_line, result_col);
  171. Point2i match_to(result_line, result_col + search_text_len);
  172. if (match_from < prev_match) {
  173. break; // done
  174. }
  175. prev_match = Point2i(result_line, result_col + replace_text.length());
  176. text_edit->unfold_line(result_line);
  177. text_edit->select(result_line, result_col, result_line, match_to.y);
  178. if (selection_enabled && is_selection_only()) {
  179. if (match_from < selection_begin || match_to > selection_end) {
  180. continue;
  181. }
  182. // replace but adjust selection bounds
  183. text_edit->insert_text_at_cursor(replace_text);
  184. if (match_to.x == selection_end.x) {
  185. selection_end.y += replace_text.length() - search_text_len;
  186. }
  187. } else {
  188. // just replace
  189. text_edit->insert_text_at_cursor(replace_text);
  190. }
  191. rc++;
  192. } while (search_next());
  193. }
  194. text_edit->end_complex_operation();
  195. replace_all_mode = false;
  196. // restore editor state (selection, cursor, scroll)
  197. text_edit->cursor_set_line(orig_cursor.x);
  198. text_edit->cursor_set_column(orig_cursor.y);
  199. if (selection_enabled && is_selection_only()) {
  200. // reselect
  201. text_edit->select(selection_begin.x, selection_begin.y, selection_end.x, selection_end.y);
  202. } else {
  203. text_edit->deselect();
  204. }
  205. text_edit->set_v_scroll(vsval);
  206. set_error(vformat(TTR("Replaced %d occurrence(s)."), rc));
  207. text_edit->call_deferred("connect", "text_changed", this, "_editor_text_changed");
  208. }
  209. void FindReplaceBar::_get_search_from(int &r_line, int &r_col) {
  210. r_line = text_edit->cursor_get_line();
  211. r_col = text_edit->cursor_get_column();
  212. if (text_edit->is_selection_active() && !replace_all_mode) {
  213. int selection_line = text_edit->get_selection_from_line();
  214. if (text_edit->get_selection_text() == get_search_text() && r_line == selection_line) {
  215. int selection_from_col = text_edit->get_selection_from_column();
  216. if (r_col >= selection_from_col && r_col <= text_edit->get_selection_to_column()) {
  217. r_col = selection_from_col;
  218. }
  219. }
  220. }
  221. if (r_line == result_line && r_col >= result_col && r_col <= result_col + get_search_text().length()) {
  222. r_col = result_col;
  223. }
  224. }
  225. bool FindReplaceBar::search_current() {
  226. uint32_t flags = 0;
  227. if (is_whole_words())
  228. flags |= TextEdit::SEARCH_WHOLE_WORDS;
  229. if (is_case_sensitive())
  230. flags |= TextEdit::SEARCH_MATCH_CASE;
  231. int line, col;
  232. _get_search_from(line, col);
  233. return _search(flags, line, col);
  234. }
  235. bool FindReplaceBar::search_prev() {
  236. uint32_t flags = 0;
  237. String text = get_search_text();
  238. if (is_whole_words())
  239. flags |= TextEdit::SEARCH_WHOLE_WORDS;
  240. if (is_case_sensitive())
  241. flags |= TextEdit::SEARCH_MATCH_CASE;
  242. flags |= TextEdit::SEARCH_BACKWARDS;
  243. int line, col;
  244. _get_search_from(line, col);
  245. col -= text.length();
  246. if (col < 0) {
  247. line -= 1;
  248. if (line < 0)
  249. line = text_edit->get_line_count() - 1;
  250. col = text_edit->get_line(line).length();
  251. }
  252. return _search(flags, line, col);
  253. }
  254. bool FindReplaceBar::search_next() {
  255. uint32_t flags = 0;
  256. String text = get_search_text();
  257. if (is_whole_words())
  258. flags |= TextEdit::SEARCH_WHOLE_WORDS;
  259. if (is_case_sensitive())
  260. flags |= TextEdit::SEARCH_MATCH_CASE;
  261. int line, col;
  262. _get_search_from(line, col);
  263. if (line == result_line && col == result_col) {
  264. col += text.length();
  265. if (col > text_edit->get_line(line).length()) {
  266. line += 1;
  267. if (line >= text_edit->get_line_count())
  268. line = 0;
  269. col = 0;
  270. }
  271. }
  272. return _search(flags, line, col);
  273. }
  274. void FindReplaceBar::_hide_bar() {
  275. if (replace_text->has_focus() || search_text->has_focus())
  276. text_edit->grab_focus();
  277. text_edit->set_search_text("");
  278. result_line = -1;
  279. result_col = -1;
  280. set_error("");
  281. hide();
  282. }
  283. void FindReplaceBar::_show_search() {
  284. show();
  285. search_text->call_deferred("grab_focus");
  286. if (text_edit->is_selection_active() && !selection_only->is_pressed()) {
  287. search_text->set_text(text_edit->get_selection_text());
  288. }
  289. if (!get_search_text().empty()) {
  290. search_text->select_all();
  291. search_text->set_cursor_position(search_text->get_text().length());
  292. search_current();
  293. }
  294. call_deferred("_update_size");
  295. }
  296. void FindReplaceBar::popup_search() {
  297. replace_text->hide();
  298. hbc_button_replace->hide();
  299. hbc_option_replace->hide();
  300. _show_search();
  301. }
  302. void FindReplaceBar::popup_replace() {
  303. if (!replace_text->is_visible_in_tree()) {
  304. replace_text->clear();
  305. replace_text->show();
  306. hbc_button_replace->show();
  307. hbc_option_replace->show();
  308. }
  309. selection_only->set_pressed((text_edit->is_selection_active() && text_edit->get_selection_from_line() < text_edit->get_selection_to_line()));
  310. _show_search();
  311. }
  312. void FindReplaceBar::_search_options_changed(bool p_pressed) {
  313. search_current();
  314. }
  315. void FindReplaceBar::_editor_text_changed() {
  316. if (is_visible_in_tree()) {
  317. preserve_cursor = true;
  318. search_current();
  319. preserve_cursor = false;
  320. }
  321. }
  322. void FindReplaceBar::_search_text_changed(const String &p_text) {
  323. search_current();
  324. }
  325. void FindReplaceBar::_search_text_entered(const String &p_text) {
  326. search_next();
  327. }
  328. void FindReplaceBar::_replace_text_entered(const String &p_text) {
  329. if (selection_only->is_pressed() && text_edit->is_selection_active()) {
  330. _replace_all();
  331. _hide_bar();
  332. }
  333. }
  334. String FindReplaceBar::get_search_text() const {
  335. return search_text->get_text();
  336. }
  337. String FindReplaceBar::get_replace_text() const {
  338. return replace_text->get_text();
  339. }
  340. bool FindReplaceBar::is_case_sensitive() const {
  341. return case_sensitive->is_pressed();
  342. }
  343. bool FindReplaceBar::is_whole_words() const {
  344. return whole_words->is_pressed();
  345. }
  346. bool FindReplaceBar::is_selection_only() const {
  347. return selection_only->is_pressed();
  348. }
  349. void FindReplaceBar::set_error(const String &p_label) {
  350. emit_signal("error", p_label);
  351. }
  352. void FindReplaceBar::set_text_edit(TextEdit *p_text_edit) {
  353. text_edit = p_text_edit;
  354. text_edit->connect("text_changed", this, "_editor_text_changed");
  355. }
  356. void FindReplaceBar::_update_size() {
  357. container->set_size(Size2(hbc->get_size().width, 1));
  358. }
  359. void FindReplaceBar::_bind_methods() {
  360. ClassDB::bind_method("_unhandled_input", &FindReplaceBar::_unhandled_input);
  361. ClassDB::bind_method("_editor_text_changed", &FindReplaceBar::_editor_text_changed);
  362. ClassDB::bind_method("_search_text_changed", &FindReplaceBar::_search_text_changed);
  363. ClassDB::bind_method("_search_text_entered", &FindReplaceBar::_search_text_entered);
  364. ClassDB::bind_method("_replace_text_entered", &FindReplaceBar::_replace_text_entered);
  365. ClassDB::bind_method("_search_current", &FindReplaceBar::search_current);
  366. ClassDB::bind_method("_search_next", &FindReplaceBar::search_next);
  367. ClassDB::bind_method("_search_prev", &FindReplaceBar::search_prev);
  368. ClassDB::bind_method("_replace_pressed", &FindReplaceBar::_replace);
  369. ClassDB::bind_method("_replace_all_pressed", &FindReplaceBar::_replace_all);
  370. ClassDB::bind_method("_search_options_changed", &FindReplaceBar::_search_options_changed);
  371. ClassDB::bind_method("_hide_pressed", &FindReplaceBar::_hide_bar);
  372. ClassDB::bind_method("_update_size", &FindReplaceBar::_update_size);
  373. ADD_SIGNAL(MethodInfo("search"));
  374. ADD_SIGNAL(MethodInfo("error"));
  375. }
  376. FindReplaceBar::FindReplaceBar() {
  377. container = memnew(MarginContainer);
  378. container->add_constant_override("margin_bottom", 5 * EDSCALE);
  379. add_child(container);
  380. container->set_clip_contents(true);
  381. container->set_h_size_flags(SIZE_EXPAND_FILL);
  382. replace_all_mode = false;
  383. preserve_cursor = false;
  384. hbc = memnew(HBoxContainer);
  385. container->add_child(hbc);
  386. hbc->set_anchor_and_margin(MARGIN_RIGHT, 1, 0);
  387. vbc_lineedit = memnew(VBoxContainer);
  388. hbc->add_child(vbc_lineedit);
  389. vbc_lineedit->set_h_size_flags(SIZE_EXPAND_FILL);
  390. VBoxContainer *vbc_button = memnew(VBoxContainer);
  391. hbc->add_child(vbc_button);
  392. VBoxContainer *vbc_option = memnew(VBoxContainer);
  393. hbc->add_child(vbc_option);
  394. HBoxContainer *hbc_button_search = memnew(HBoxContainer);
  395. vbc_button->add_child(hbc_button_search);
  396. hbc_button_replace = memnew(HBoxContainer);
  397. vbc_button->add_child(hbc_button_replace);
  398. HBoxContainer *hbc_option_search = memnew(HBoxContainer);
  399. vbc_option->add_child(hbc_option_search);
  400. hbc_option_replace = memnew(HBoxContainer);
  401. vbc_option->add_child(hbc_option_replace);
  402. // search toolbar
  403. search_text = memnew(LineEdit);
  404. vbc_lineedit->add_child(search_text);
  405. search_text->set_custom_minimum_size(Size2(100 * EDSCALE, 0));
  406. search_text->connect("text_changed", this, "_search_text_changed");
  407. search_text->connect("text_entered", this, "_search_text_entered");
  408. find_prev = memnew(ToolButton);
  409. hbc_button_search->add_child(find_prev);
  410. find_prev->set_focus_mode(FOCUS_NONE);
  411. find_prev->connect("pressed", this, "_search_prev");
  412. find_next = memnew(ToolButton);
  413. hbc_button_search->add_child(find_next);
  414. find_next->set_focus_mode(FOCUS_NONE);
  415. find_next->connect("pressed", this, "_search_next");
  416. case_sensitive = memnew(CheckBox);
  417. hbc_option_search->add_child(case_sensitive);
  418. case_sensitive->set_text(TTR("Match Case"));
  419. case_sensitive->set_focus_mode(FOCUS_NONE);
  420. case_sensitive->connect("toggled", this, "_search_options_changed");
  421. whole_words = memnew(CheckBox);
  422. hbc_option_search->add_child(whole_words);
  423. whole_words->set_text(TTR("Whole Words"));
  424. whole_words->set_focus_mode(FOCUS_NONE);
  425. whole_words->connect("toggled", this, "_search_options_changed");
  426. // replace toolbar
  427. replace_text = memnew(LineEdit);
  428. vbc_lineedit->add_child(replace_text);
  429. replace_text->set_custom_minimum_size(Size2(100 * EDSCALE, 0));
  430. replace_text->connect("text_entered", this, "_replace_text_entered");
  431. replace = memnew(Button);
  432. hbc_button_replace->add_child(replace);
  433. replace->set_text(TTR("Replace"));
  434. replace->connect("pressed", this, "_replace_pressed");
  435. replace_all = memnew(Button);
  436. hbc_button_replace->add_child(replace_all);
  437. replace_all->set_text(TTR("Replace All"));
  438. replace_all->connect("pressed", this, "_replace_all_pressed");
  439. selection_only = memnew(CheckBox);
  440. hbc_option_replace->add_child(selection_only);
  441. selection_only->set_text(TTR("Selection Only"));
  442. selection_only->set_focus_mode(FOCUS_NONE);
  443. selection_only->connect("toggled", this, "_search_options_changed");
  444. hide_button = memnew(TextureButton);
  445. add_child(hide_button);
  446. hide_button->set_focus_mode(FOCUS_NONE);
  447. hide_button->connect("pressed", this, "_hide_pressed");
  448. hide_button->set_v_size_flags(SIZE_SHRINK_CENTER);
  449. }
  450. /*** CODE EDITOR ****/
  451. void CodeTextEditor::_text_editor_gui_input(const Ref<InputEvent> &p_event) {
  452. Ref<InputEventMouseButton> mb = p_event;
  453. if (mb.is_valid()) {
  454. if (mb->is_pressed() && mb->get_command()) {
  455. if (mb->get_button_index() == BUTTON_WHEEL_UP) {
  456. _zoom_in();
  457. } else if (mb->get_button_index() == BUTTON_WHEEL_DOWN) {
  458. _zoom_out();
  459. }
  460. }
  461. }
  462. Ref<InputEventMagnifyGesture> magnify_gesture = p_event;
  463. if (magnify_gesture.is_valid()) {
  464. Ref<DynamicFont> font = text_editor->get_font("font");
  465. if (font.is_valid()) {
  466. if (font->get_size() != (int)font_size) {
  467. font_size = font->get_size();
  468. }
  469. font_size *= powf(magnify_gesture->get_factor(), 0.25);
  470. _add_font_size((int)font_size - font->get_size());
  471. }
  472. return;
  473. }
  474. Ref<InputEventKey> k = p_event;
  475. if (k.is_valid()) {
  476. if (k->is_pressed()) {
  477. if (ED_IS_SHORTCUT("script_editor/zoom_in", p_event)) {
  478. _zoom_in();
  479. }
  480. if (ED_IS_SHORTCUT("script_editor/zoom_out", p_event)) {
  481. _zoom_out();
  482. }
  483. if (ED_IS_SHORTCUT("script_editor/reset_zoom", p_event)) {
  484. _reset_zoom();
  485. }
  486. }
  487. }
  488. }
  489. void CodeTextEditor::_zoom_in() {
  490. font_resize_val += MAX(EDSCALE, 1.0f);
  491. _zoom_changed();
  492. }
  493. void CodeTextEditor::_zoom_out() {
  494. font_resize_val -= MAX(EDSCALE, 1.0f);
  495. _zoom_changed();
  496. }
  497. void CodeTextEditor::_zoom_changed() {
  498. if (font_resize_timer->get_time_left() == 0)
  499. font_resize_timer->start();
  500. }
  501. void CodeTextEditor::_reset_zoom() {
  502. Ref<DynamicFont> font = text_editor->get_font("font"); // reset source font size to default
  503. if (font.is_valid()) {
  504. EditorSettings::get_singleton()->set("interface/editor/code_font_size", 14);
  505. font->set_size(14);
  506. font_size_nb->set_text("14 (100%)");
  507. }
  508. }
  509. void CodeTextEditor::_line_col_changed() {
  510. line_nb->set_text(itos(text_editor->cursor_get_line() + 1));
  511. String line = text_editor->get_line(text_editor->cursor_get_line());
  512. int positional_column = 0;
  513. for (int i = 0; i < text_editor->cursor_get_column(); i++) {
  514. if (line[i] == '\t') {
  515. positional_column += text_editor->get_indent_size(); //tab size
  516. } else {
  517. positional_column += 1;
  518. }
  519. }
  520. col_nb->set_text(itos(positional_column + 1));
  521. }
  522. void CodeTextEditor::_text_changed() {
  523. if (text_editor->is_insert_text_operation()) {
  524. code_complete_timer->start();
  525. }
  526. idle->start();
  527. }
  528. void CodeTextEditor::_code_complete_timer_timeout() {
  529. if (!is_visible_in_tree())
  530. return;
  531. text_editor->query_code_comple();
  532. }
  533. void CodeTextEditor::_complete_request() {
  534. List<String> entries;
  535. String ctext = text_editor->get_text_for_completion();
  536. _code_complete_script(ctext, &entries);
  537. bool forced = false;
  538. if (code_complete_func) {
  539. code_complete_func(code_complete_ud, ctext, &entries, forced);
  540. }
  541. if (entries.size() == 0)
  542. return;
  543. Vector<String> strs;
  544. strs.resize(entries.size());
  545. int i = 0;
  546. for (List<String>::Element *E = entries.front(); E; E = E->next()) {
  547. strs.write[i++] = E->get();
  548. }
  549. text_editor->code_complete(strs, forced);
  550. }
  551. void CodeTextEditor::_font_resize_timeout() {
  552. if (_add_font_size(font_resize_val)) {
  553. font_resize_val = 0;
  554. }
  555. }
  556. bool CodeTextEditor::_add_font_size(int p_delta) {
  557. Ref<DynamicFont> font = text_editor->get_font("font");
  558. if (font.is_valid()) {
  559. int new_size = CLAMP(font->get_size() + p_delta, 8 * EDSCALE, 96 * EDSCALE);
  560. font_size_nb->set_text(itos(new_size) + " (" + itos(100 * new_size / (14 * EDSCALE)) + "%)");
  561. if (new_size != font->get_size()) {
  562. EditorSettings::get_singleton()->set("interface/editor/code_font_size", new_size / EDSCALE);
  563. font->set_size(new_size);
  564. }
  565. return true;
  566. } else {
  567. return false;
  568. }
  569. }
  570. void CodeTextEditor::update_editor_settings() {
  571. text_editor->set_auto_brace_completion(EditorSettings::get_singleton()->get("text_editor/completion/auto_brace_complete"));
  572. text_editor->set_scroll_pass_end_of_file(EditorSettings::get_singleton()->get("text_editor/cursor/scroll_past_end_of_file"));
  573. text_editor->set_indent_using_spaces(EditorSettings::get_singleton()->get("text_editor/indent/type"));
  574. text_editor->set_indent_size(EditorSettings::get_singleton()->get("text_editor/indent/size"));
  575. text_editor->set_auto_indent(EditorSettings::get_singleton()->get("text_editor/indent/auto_indent"));
  576. text_editor->set_draw_tabs(EditorSettings::get_singleton()->get("text_editor/indent/draw_tabs"));
  577. text_editor->set_show_line_numbers(EditorSettings::get_singleton()->get("text_editor/line_numbers/show_line_numbers"));
  578. text_editor->set_line_numbers_zero_padded(EditorSettings::get_singleton()->get("text_editor/line_numbers/line_numbers_zero_padded"));
  579. text_editor->set_show_line_length_guideline(EditorSettings::get_singleton()->get("text_editor/line_numbers/show_line_length_guideline"));
  580. text_editor->set_line_length_guideline_column(EditorSettings::get_singleton()->get("text_editor/line_numbers/line_length_guideline_column"));
  581. text_editor->set_syntax_coloring(EditorSettings::get_singleton()->get("text_editor/highlighting/syntax_highlighting"));
  582. text_editor->set_highlight_all_occurrences(EditorSettings::get_singleton()->get("text_editor/highlighting/highlight_all_occurrences"));
  583. text_editor->set_highlight_current_line(EditorSettings::get_singleton()->get("text_editor/highlighting/highlight_current_line"));
  584. text_editor->cursor_set_blink_enabled(EditorSettings::get_singleton()->get("text_editor/cursor/caret_blink"));
  585. text_editor->cursor_set_blink_speed(EditorSettings::get_singleton()->get("text_editor/cursor/caret_blink_speed"));
  586. text_editor->set_breakpoint_gutter_enabled(EditorSettings::get_singleton()->get("text_editor/line_numbers/show_breakpoint_gutter"));
  587. text_editor->set_hiding_enabled(EditorSettings::get_singleton()->get("text_editor/line_numbers/code_folding"));
  588. text_editor->set_draw_fold_gutter(EditorSettings::get_singleton()->get("text_editor/line_numbers/code_folding"));
  589. text_editor->set_wrap_enabled(EditorSettings::get_singleton()->get("text_editor/line_numbers/word_wrap"));
  590. text_editor->cursor_set_block_mode(EditorSettings::get_singleton()->get("text_editor/cursor/block_caret"));
  591. text_editor->set_smooth_scroll_enabled(EditorSettings::get_singleton()->get("text_editor/open_scripts/smooth_scrolling"));
  592. text_editor->set_v_scroll_speed(EditorSettings::get_singleton()->get("text_editor/open_scripts/v_scroll_speed"));
  593. }
  594. void CodeTextEditor::trim_trailing_whitespace() {
  595. bool trimed_whitespace = false;
  596. for (int i = 0; i < text_editor->get_line_count(); i++) {
  597. String line = text_editor->get_line(i);
  598. if (line.ends_with(" ") || line.ends_with("\t")) {
  599. if (!trimed_whitespace) {
  600. text_editor->begin_complex_operation();
  601. trimed_whitespace = true;
  602. }
  603. int end = 0;
  604. for (int j = line.length() - 1; j > -1; j--) {
  605. if (line[j] != ' ' && line[j] != '\t') {
  606. end = j + 1;
  607. break;
  608. }
  609. }
  610. text_editor->set_line(i, line.substr(0, end));
  611. }
  612. }
  613. if (trimed_whitespace) {
  614. text_editor->end_complex_operation();
  615. text_editor->update();
  616. }
  617. }
  618. void CodeTextEditor::convert_indent_to_spaces() {
  619. int indent_size = EditorSettings::get_singleton()->get("text_editor/indent/size");
  620. String indent = "";
  621. for (int i = 0; i < indent_size; i++) {
  622. indent += " ";
  623. }
  624. int cursor_line = text_editor->cursor_get_line();
  625. int cursor_column = text_editor->cursor_get_column();
  626. bool changed_indentation = false;
  627. for (int i = 0; i < text_editor->get_line_count(); i++) {
  628. String line = text_editor->get_line(i);
  629. if (line.length() <= 0) {
  630. continue;
  631. }
  632. int j = 0;
  633. while (j < line.length() && (line[j] == ' ' || line[j] == '\t')) {
  634. if (line[j] == '\t') {
  635. if (!changed_indentation) {
  636. text_editor->begin_complex_operation();
  637. changed_indentation = true;
  638. }
  639. if (cursor_line == i && cursor_column > j) {
  640. cursor_column += indent_size - 1;
  641. }
  642. line = line.left(j) + indent + line.right(j + 1);
  643. }
  644. j++;
  645. }
  646. if (changed_indentation) {
  647. text_editor->set_line(i, line);
  648. }
  649. }
  650. if (changed_indentation) {
  651. text_editor->cursor_set_column(cursor_column);
  652. text_editor->end_complex_operation();
  653. text_editor->update();
  654. }
  655. }
  656. void CodeTextEditor::convert_indent_to_tabs() {
  657. int indent_size = EditorSettings::get_singleton()->get("text_editor/indent/size");
  658. indent_size -= 1;
  659. int cursor_line = text_editor->cursor_get_line();
  660. int cursor_column = text_editor->cursor_get_column();
  661. bool changed_indentation = false;
  662. for (int i = 0; i < text_editor->get_line_count(); i++) {
  663. String line = text_editor->get_line(i);
  664. if (line.length() <= 0) {
  665. continue;
  666. }
  667. int j = 0;
  668. int space_count = -1;
  669. while (j < line.length() && (line[j] == ' ' || line[j] == '\t')) {
  670. if (line[j] != '\t') {
  671. space_count++;
  672. if (space_count == indent_size) {
  673. if (!changed_indentation) {
  674. text_editor->begin_complex_operation();
  675. changed_indentation = true;
  676. }
  677. if (cursor_line == i && cursor_column > j) {
  678. cursor_column -= indent_size;
  679. }
  680. line = line.left(j - indent_size) + "\t" + line.right(j + 1);
  681. j = 0;
  682. space_count = -1;
  683. }
  684. } else {
  685. space_count = -1;
  686. }
  687. j++;
  688. }
  689. if (changed_indentation) {
  690. text_editor->set_line(i, line);
  691. }
  692. }
  693. if (changed_indentation) {
  694. text_editor->cursor_set_column(cursor_column);
  695. text_editor->end_complex_operation();
  696. text_editor->update();
  697. }
  698. }
  699. void CodeTextEditor::convert_case(CaseStyle p_case) {
  700. if (!text_editor->is_selection_active()) {
  701. return;
  702. }
  703. text_editor->begin_complex_operation();
  704. int begin = text_editor->get_selection_from_line();
  705. int end = text_editor->get_selection_to_line();
  706. int begin_col = text_editor->get_selection_from_column();
  707. int end_col = text_editor->get_selection_to_column();
  708. for (int i = begin; i <= end; i++) {
  709. int len = text_editor->get_line(i).length();
  710. if (i == end)
  711. len -= len - end_col;
  712. if (i == begin)
  713. len -= begin_col;
  714. String new_line = text_editor->get_line(i).substr(i == begin ? begin_col : 0, len);
  715. switch (p_case) {
  716. case UPPER: {
  717. new_line = new_line.to_upper();
  718. } break;
  719. case LOWER: {
  720. new_line = new_line.to_lower();
  721. } break;
  722. case CAPITALIZE: {
  723. new_line = new_line.capitalize();
  724. } break;
  725. }
  726. if (i == begin) {
  727. new_line = text_editor->get_line(i).left(begin_col) + new_line;
  728. }
  729. if (i == end) {
  730. new_line = new_line + text_editor->get_line(i).right(end_col);
  731. }
  732. text_editor->set_line(i, new_line);
  733. }
  734. text_editor->end_complex_operation();
  735. }
  736. void CodeTextEditor::move_lines_up() {
  737. text_editor->begin_complex_operation();
  738. if (text_editor->is_selection_active()) {
  739. int from_line = text_editor->get_selection_from_line();
  740. int from_col = text_editor->get_selection_from_column();
  741. int to_line = text_editor->get_selection_to_line();
  742. int to_column = text_editor->get_selection_to_column();
  743. for (int i = from_line; i <= to_line; i++) {
  744. int line_id = i;
  745. int next_id = i - 1;
  746. if (line_id == 0 || next_id < 0)
  747. return;
  748. text_editor->unfold_line(line_id);
  749. text_editor->unfold_line(next_id);
  750. text_editor->swap_lines(line_id, next_id);
  751. text_editor->cursor_set_line(next_id);
  752. }
  753. int from_line_up = from_line > 0 ? from_line - 1 : from_line;
  754. int to_line_up = to_line > 0 ? to_line - 1 : to_line;
  755. text_editor->select(from_line_up, from_col, to_line_up, to_column);
  756. } else {
  757. int line_id = text_editor->cursor_get_line();
  758. int next_id = line_id - 1;
  759. if (line_id == 0 || next_id < 0)
  760. return;
  761. text_editor->unfold_line(line_id);
  762. text_editor->unfold_line(next_id);
  763. text_editor->swap_lines(line_id, next_id);
  764. text_editor->cursor_set_line(next_id);
  765. }
  766. text_editor->end_complex_operation();
  767. text_editor->update();
  768. }
  769. void CodeTextEditor::move_lines_down() {
  770. text_editor->begin_complex_operation();
  771. if (text_editor->is_selection_active()) {
  772. int from_line = text_editor->get_selection_from_line();
  773. int from_col = text_editor->get_selection_from_column();
  774. int to_line = text_editor->get_selection_to_line();
  775. int to_column = text_editor->get_selection_to_column();
  776. for (int i = to_line; i >= from_line; i--) {
  777. int line_id = i;
  778. int next_id = i + 1;
  779. if (line_id == text_editor->get_line_count() - 1 || next_id > text_editor->get_line_count())
  780. return;
  781. text_editor->unfold_line(line_id);
  782. text_editor->unfold_line(next_id);
  783. text_editor->swap_lines(line_id, next_id);
  784. text_editor->cursor_set_line(next_id);
  785. }
  786. int from_line_down = from_line < text_editor->get_line_count() ? from_line + 1 : from_line;
  787. int to_line_down = to_line < text_editor->get_line_count() ? to_line + 1 : to_line;
  788. text_editor->select(from_line_down, from_col, to_line_down, to_column);
  789. } else {
  790. int line_id = text_editor->cursor_get_line();
  791. int next_id = line_id + 1;
  792. if (line_id == text_editor->get_line_count() - 1 || next_id > text_editor->get_line_count())
  793. return;
  794. text_editor->unfold_line(line_id);
  795. text_editor->unfold_line(next_id);
  796. text_editor->swap_lines(line_id, next_id);
  797. text_editor->cursor_set_line(next_id);
  798. }
  799. text_editor->end_complex_operation();
  800. text_editor->update();
  801. }
  802. void CodeTextEditor::delete_lines() {
  803. text_editor->begin_complex_operation();
  804. if (text_editor->is_selection_active()) {
  805. int to_line = text_editor->get_selection_to_line();
  806. int from_line = text_editor->get_selection_from_line();
  807. int count = Math::abs(to_line - from_line) + 1;
  808. text_editor->cursor_set_line(to_line, false);
  809. while (count) {
  810. text_editor->set_line(text_editor->cursor_get_line(), "");
  811. text_editor->backspace_at_cursor();
  812. count--;
  813. if (count)
  814. text_editor->unfold_line(from_line);
  815. }
  816. text_editor->cursor_set_line(from_line - 1);
  817. text_editor->deselect();
  818. } else {
  819. int line = text_editor->cursor_get_line();
  820. text_editor->set_line(text_editor->cursor_get_line(), "");
  821. text_editor->backspace_at_cursor();
  822. text_editor->unfold_line(line);
  823. text_editor->cursor_set_line(line);
  824. }
  825. text_editor->end_complex_operation();
  826. }
  827. void CodeTextEditor::clone_lines_down() {
  828. int from_line = text_editor->cursor_get_line();
  829. int to_line = text_editor->cursor_get_line();
  830. int column = text_editor->cursor_get_column();
  831. if (text_editor->is_selection_active()) {
  832. from_line = text_editor->get_selection_from_line();
  833. to_line = text_editor->get_selection_to_line();
  834. column = text_editor->cursor_get_column();
  835. }
  836. int next_line = to_line + 1;
  837. bool caret_at_start = text_editor->cursor_get_line() == from_line;
  838. text_editor->begin_complex_operation();
  839. for (int i = from_line; i <= to_line; i++) {
  840. text_editor->unfold_line(i);
  841. text_editor->set_line(next_line - 1, text_editor->get_line(next_line - 1) + "\n");
  842. text_editor->set_line(next_line, text_editor->get_line(i));
  843. next_line++;
  844. }
  845. if (caret_at_start) {
  846. text_editor->cursor_set_line(to_line + 1);
  847. } else {
  848. text_editor->cursor_set_line(next_line - 1);
  849. }
  850. text_editor->cursor_set_column(column);
  851. if (text_editor->is_selection_active()) {
  852. text_editor->select(to_line + 1, text_editor->get_selection_from_column(), next_line - 1, text_editor->get_selection_to_column());
  853. }
  854. text_editor->end_complex_operation();
  855. text_editor->update();
  856. }
  857. void CodeTextEditor::goto_line(int p_line) {
  858. text_editor->deselect();
  859. text_editor->unfold_line(p_line);
  860. text_editor->call_deferred("cursor_set_line", p_line);
  861. }
  862. void CodeTextEditor::goto_line_selection(int p_line, int p_begin, int p_end) {
  863. text_editor->unfold_line(p_line);
  864. text_editor->call_deferred("cursor_set_line", p_line);
  865. text_editor->call_deferred("cursor_set_column", p_begin);
  866. text_editor->select(p_line, p_begin, p_line, p_end);
  867. }
  868. Variant CodeTextEditor::get_edit_state() {
  869. Dictionary state;
  870. state["scroll_position"] = text_editor->get_v_scroll();
  871. state["column"] = text_editor->cursor_get_column();
  872. state["row"] = text_editor->cursor_get_line();
  873. return state;
  874. }
  875. void CodeTextEditor::set_edit_state(const Variant &p_state) {
  876. Dictionary state = p_state;
  877. text_editor->cursor_set_column(state["column"]);
  878. text_editor->cursor_set_line(state["row"]);
  879. text_editor->set_v_scroll(state["scroll_position"]);
  880. text_editor->grab_focus();
  881. }
  882. void CodeTextEditor::set_error(const String &p_error) {
  883. error->set_text(p_error);
  884. if (p_error != "") {
  885. error->set_default_cursor_shape(CURSOR_POINTING_HAND);
  886. } else {
  887. error->set_default_cursor_shape(CURSOR_ARROW);
  888. }
  889. }
  890. void CodeTextEditor::set_error_pos(int p_line, int p_column) {
  891. error_line = p_line;
  892. error_column = p_column;
  893. }
  894. void CodeTextEditor::goto_error() {
  895. if (error->get_text() != "") {
  896. text_editor->cursor_set_line(error_line);
  897. text_editor->cursor_set_column(error_column);
  898. text_editor->center_viewport_to_cursor();
  899. }
  900. }
  901. void CodeTextEditor::_update_font() {
  902. text_editor->add_font_override("font", get_font("source", "EditorFonts"));
  903. Ref<Font> status_bar_font = get_font("status_source", "EditorFonts");
  904. int count = status_bar->get_child_count();
  905. for (int i = 0; i < count; i++) {
  906. Control *n = Object::cast_to<Control>(status_bar->get_child(i));
  907. if (n)
  908. n->add_font_override("font", status_bar_font);
  909. }
  910. }
  911. void CodeTextEditor::_on_settings_change() {
  912. _update_font();
  913. font_size = EditorSettings::get_singleton()->get("interface/editor/code_font_size");
  914. font_size_nb->set_text(itos(font_size) + " (" + itos(100 * font_size / (14 * EDSCALE)) + "%)");
  915. // AUTO BRACE COMPLETION
  916. text_editor->set_auto_brace_completion(
  917. EDITOR_DEF("text_editor/completion/auto_brace_complete", true));
  918. code_complete_timer->set_wait_time(
  919. EDITOR_DEF("text_editor/completion/code_complete_delay", .3f));
  920. // call hint settings
  921. text_editor->set_callhint_settings(
  922. EDITOR_DEF("text_editor/completion/put_callhint_tooltip_below_current_line", true),
  923. EDITOR_DEF("text_editor/completion/callhint_tooltip_offset", Vector2()));
  924. }
  925. void CodeTextEditor::_text_changed_idle_timeout() {
  926. _validate_script();
  927. emit_signal("validate_script");
  928. }
  929. void CodeTextEditor::_notification(int p_what) {
  930. if (p_what == EditorSettings::NOTIFICATION_EDITOR_SETTINGS_CHANGED) {
  931. _load_theme_settings();
  932. emit_signal("load_theme_settings");
  933. }
  934. if (p_what == NOTIFICATION_THEME_CHANGED) {
  935. _update_font();
  936. }
  937. }
  938. void CodeTextEditor::_bind_methods() {
  939. ClassDB::bind_method("_text_editor_gui_input", &CodeTextEditor::_text_editor_gui_input);
  940. ClassDB::bind_method("_line_col_changed", &CodeTextEditor::_line_col_changed);
  941. ClassDB::bind_method("_text_changed", &CodeTextEditor::_text_changed);
  942. ClassDB::bind_method("_on_settings_change", &CodeTextEditor::_on_settings_change);
  943. ClassDB::bind_method("_text_changed_idle_timeout", &CodeTextEditor::_text_changed_idle_timeout);
  944. ClassDB::bind_method("_code_complete_timer_timeout", &CodeTextEditor::_code_complete_timer_timeout);
  945. ClassDB::bind_method("_complete_request", &CodeTextEditor::_complete_request);
  946. ClassDB::bind_method("_font_resize_timeout", &CodeTextEditor::_font_resize_timeout);
  947. ADD_SIGNAL(MethodInfo("validate_script"));
  948. ADD_SIGNAL(MethodInfo("load_theme_settings"));
  949. }
  950. void CodeTextEditor::set_code_complete_func(CodeTextEditorCodeCompleteFunc p_code_complete_func, void *p_ud) {
  951. code_complete_func = p_code_complete_func;
  952. code_complete_ud = p_ud;
  953. }
  954. CodeTextEditor::CodeTextEditor() {
  955. code_complete_func = NULL;
  956. ED_SHORTCUT("script_editor/zoom_in", TTR("Zoom In"), KEY_MASK_CMD | KEY_EQUAL);
  957. ED_SHORTCUT("script_editor/zoom_out", TTR("Zoom Out"), KEY_MASK_CMD | KEY_MINUS);
  958. ED_SHORTCUT("script_editor/reset_zoom", TTR("Reset Zoom"), KEY_MASK_CMD | KEY_0);
  959. find_replace_bar = memnew(FindReplaceBar);
  960. add_child(find_replace_bar);
  961. find_replace_bar->set_h_size_flags(SIZE_EXPAND_FILL);
  962. find_replace_bar->hide();
  963. text_editor = memnew(TextEdit);
  964. add_child(text_editor);
  965. text_editor->set_v_size_flags(SIZE_EXPAND_FILL);
  966. find_replace_bar->set_text_edit(text_editor);
  967. text_editor->set_show_line_numbers(true);
  968. text_editor->set_brace_matching(true);
  969. text_editor->set_auto_indent(true);
  970. status_bar = memnew(HBoxContainer);
  971. add_child(status_bar);
  972. status_bar->set_h_size_flags(SIZE_EXPAND_FILL);
  973. idle = memnew(Timer);
  974. add_child(idle);
  975. idle->set_one_shot(true);
  976. idle->set_wait_time(EDITOR_DEF("text_editor/completion/idle_parse_delay", 2));
  977. code_complete_timer = memnew(Timer);
  978. add_child(code_complete_timer);
  979. code_complete_timer->set_one_shot(true);
  980. code_complete_timer->set_wait_time(EDITOR_DEF("text_editor/completion/code_complete_delay", .3f));
  981. error_line = 0;
  982. error_column = 0;
  983. error = memnew(Label);
  984. status_bar->add_child(error);
  985. error->set_autowrap(true);
  986. error->set_valign(Label::VALIGN_CENTER);
  987. error->set_h_size_flags(SIZE_EXPAND_FILL); //required for it to display, given now it's clipping contents, do not touch
  988. error->add_color_override("font_color", EditorNode::get_singleton()->get_gui_base()->get_color("error_color", "Editor"));
  989. error->add_font_override("font", EditorNode::get_singleton()->get_gui_base()->get_font("status_source", "EditorFonts"));
  990. error->set_mouse_filter(MOUSE_FILTER_STOP);
  991. find_replace_bar->connect("error", error, "set_text");
  992. status_bar->add_child(memnew(Label)); //to keep the height if the other labels are not visible
  993. warning_label = memnew(Label);
  994. status_bar->add_child(warning_label);
  995. warning_label->set_align(Label::ALIGN_RIGHT);
  996. warning_label->set_valign(Label::VALIGN_CENTER);
  997. warning_label->set_v_size_flags(SIZE_FILL);
  998. warning_label->set_default_cursor_shape(CURSOR_POINTING_HAND);
  999. warning_label->set_mouse_filter(MOUSE_FILTER_STOP);
  1000. warning_label->set_text(TTR("Warnings:"));
  1001. warning_label->add_font_override("font", EditorNode::get_singleton()->get_gui_base()->get_font("status_source", "EditorFonts"));
  1002. warning_count_label = memnew(Label);
  1003. status_bar->add_child(warning_count_label);
  1004. warning_count_label->set_valign(Label::VALIGN_CENTER);
  1005. warning_count_label->set_v_size_flags(SIZE_FILL);
  1006. warning_count_label->set_autowrap(true); // workaround to prevent resizing the label on each change, do not touch
  1007. warning_count_label->set_clip_text(true); // workaround to prevent resizing the label on each change, do not touch
  1008. warning_count_label->set_custom_minimum_size(Size2(40, 1) * EDSCALE);
  1009. warning_count_label->set_align(Label::ALIGN_RIGHT);
  1010. warning_count_label->set_default_cursor_shape(CURSOR_POINTING_HAND);
  1011. warning_count_label->set_mouse_filter(MOUSE_FILTER_STOP);
  1012. warning_count_label->add_font_override("font", EditorNode::get_singleton()->get_gui_base()->get_font("status_source", "EditorFonts"));
  1013. warning_count_label->set_text("0");
  1014. Label *font_size_txt = memnew(Label);
  1015. status_bar->add_child(font_size_txt);
  1016. font_size_txt->set_align(Label::ALIGN_RIGHT);
  1017. font_size_txt->set_valign(Label::VALIGN_CENTER);
  1018. font_size_txt->set_v_size_flags(SIZE_FILL);
  1019. font_size_txt->set_text(TTR("Font Size:"));
  1020. font_size_txt->add_font_override("font", EditorNode::get_singleton()->get_gui_base()->get_font("status_source", "EditorFonts"));
  1021. font_size_nb = memnew(Label);
  1022. status_bar->add_child(font_size_nb);
  1023. font_size_nb->set_valign(Label::VALIGN_CENTER);
  1024. font_size_nb->set_v_size_flags(SIZE_FILL);
  1025. font_size_nb->set_autowrap(true); // workaround to prevent resizing the label on each change, do not touch
  1026. font_size_nb->set_clip_text(true); // workaround to prevent resizing the label on each change, do not touch
  1027. font_size_nb->set_custom_minimum_size(Size2(100, 1) * EDSCALE);
  1028. font_size_nb->set_align(Label::ALIGN_RIGHT);
  1029. font_size_nb->add_font_override("font", EditorNode::get_singleton()->get_gui_base()->get_font("status_source", "EditorFonts"));
  1030. Label *line_txt = memnew(Label);
  1031. status_bar->add_child(line_txt);
  1032. line_txt->set_align(Label::ALIGN_RIGHT);
  1033. line_txt->set_valign(Label::VALIGN_CENTER);
  1034. line_txt->set_v_size_flags(SIZE_FILL);
  1035. line_txt->set_text(TTR("Line:"));
  1036. line_txt->add_font_override("font", EditorNode::get_singleton()->get_gui_base()->get_font("status_source", "EditorFonts"));
  1037. line_nb = memnew(Label);
  1038. status_bar->add_child(line_nb);
  1039. line_nb->set_valign(Label::VALIGN_CENTER);
  1040. line_nb->set_v_size_flags(SIZE_FILL);
  1041. line_nb->set_autowrap(true); // workaround to prevent resizing the label on each change, do not touch
  1042. line_nb->set_clip_text(true); // workaround to prevent resizing the label on each change, do not touch
  1043. line_nb->set_custom_minimum_size(Size2(40, 1) * EDSCALE);
  1044. line_nb->set_align(Label::ALIGN_RIGHT);
  1045. line_nb->add_font_override("font", EditorNode::get_singleton()->get_gui_base()->get_font("status_source", "EditorFonts"));
  1046. Label *col_txt = memnew(Label);
  1047. status_bar->add_child(col_txt);
  1048. col_txt->set_align(Label::ALIGN_RIGHT);
  1049. col_txt->set_valign(Label::VALIGN_CENTER);
  1050. col_txt->set_v_size_flags(SIZE_FILL);
  1051. col_txt->set_text(TTR("Col:"));
  1052. col_txt->add_font_override("font", EditorNode::get_singleton()->get_gui_base()->get_font("status_source", "EditorFonts"));
  1053. col_nb = memnew(Label);
  1054. status_bar->add_child(col_nb);
  1055. col_nb->set_valign(Label::VALIGN_CENTER);
  1056. col_nb->set_v_size_flags(SIZE_FILL);
  1057. col_nb->set_autowrap(true); // workaround to prevent resizing the label on each change, do not touch
  1058. col_nb->set_clip_text(true); // workaround to prevent resizing the label on each change, do not touch
  1059. col_nb->set_custom_minimum_size(Size2(40, 1) * EDSCALE);
  1060. col_nb->set_align(Label::ALIGN_RIGHT);
  1061. col_nb->set("custom_constants/margin_right", 0);
  1062. col_nb->add_font_override("font", EditorNode::get_singleton()->get_gui_base()->get_font("status_source", "EditorFonts"));
  1063. text_editor->connect("gui_input", this, "_text_editor_gui_input");
  1064. text_editor->connect("cursor_changed", this, "_line_col_changed");
  1065. text_editor->connect("text_changed", this, "_text_changed");
  1066. text_editor->connect("request_completion", this, "_complete_request");
  1067. Vector<String> cs;
  1068. cs.push_back(".");
  1069. cs.push_back(",");
  1070. cs.push_back("(");
  1071. cs.push_back("=");
  1072. cs.push_back("$");
  1073. text_editor->set_completion(true, cs);
  1074. idle->connect("timeout", this, "_text_changed_idle_timeout");
  1075. code_complete_timer->connect("timeout", this, "_code_complete_timer_timeout");
  1076. font_resize_val = 0;
  1077. font_size = EditorSettings::get_singleton()->get("interface/editor/code_font_size");
  1078. font_size_nb->set_text(itos(font_size) + " (" + itos(100 * font_size / (14 * EDSCALE)) + "%)");
  1079. font_resize_timer = memnew(Timer);
  1080. add_child(font_resize_timer);
  1081. font_resize_timer->set_one_shot(true);
  1082. font_resize_timer->set_wait_time(0.07);
  1083. font_resize_timer->connect("timeout", this, "_font_resize_timeout");
  1084. EditorSettings::get_singleton()->connect("settings_changed", this, "_on_settings_change");
  1085. }