@tool extends VBoxContainer # ############################################################################## # Keeps search results from the TextEdit # ############################################################################## class TextEditSearcher: var te: TextEdit var _last_term = "" var _last_pos = Vector2(-1, -1) var _ignore_caret_change = false func set_text_edit(which): te = which te.caret_changed.connect(_on_caret_changed) func _on_caret_changed(): if _ignore_caret_change: _ignore_caret_change = false else: _last_pos = _get_caret() func _get_caret(): return Vector2(te.get_caret_column(), te.get_caret_line()) func _set_caret_and_sel(pos, len): te.set_caret_line(pos.y) te.set_caret_column(pos.x) if len > 0: te.select(pos.y, pos.x, pos.y, pos.x + len) func _find(term, search_flags): var pos = _get_caret() if term == _last_term: if search_flags == 0: pos = _last_pos pos.x += 1 else: pos = _last_pos pos.x -= 1 var result = te.search(term, search_flags, pos.y, pos.x) # print('searching from ', pos, ' for "', term, '" = ', result) if result.y != -1: _ignore_caret_change = true _set_caret_and_sel(result, term.length()) _last_pos = result _last_term = term func find_next(term): _find(term, 0) func find_prev(term): _find(term, te.SEARCH_BACKWARDS) # ############################################################################## # Start OutputText control code # ############################################################################## @onready var _ctrls = { output = $Output, copy_button = $Toolbar/CopyButton, use_colors = $Toolbar/UseColors, clear_button = $Toolbar/ClearButton, word_wrap = $Toolbar/WordWrap, show_search = $Toolbar/ShowSearch, caret_position = $Toolbar/LblPosition, search_bar = { bar = $Search, search_term = $Search/SearchTerm, } } var _sr = TextEditSearcher.new() var _highlighter: CodeHighlighter var _font_name = null # Automatically used when running the OutputText scene from the editor. Changes # to this method only affect test-running the control through the editor. func _test_running_setup(): _ctrls.use_colors.text = "use colors" _ctrls.show_search.text = "search" _ctrls.word_wrap.text = "ww" set_all_fonts("CourierPrime") set_font_size(30) _ctrls.output.queue_redraw() load_file("user://.gut_editor.bbcode") await get_tree().process_frame show_search(true) _ctrls.output.set_caret_line(0) _ctrls.output.scroll_vertical = 0 _ctrls.output.caret_changed.connect(_on_caret_changed) func _ready(): _sr.set_text_edit(_ctrls.output) _ctrls.use_colors.icon = get_theme_icon("RichTextEffect", "EditorIcons") _ctrls.show_search.icon = get_theme_icon("Search", "EditorIcons") _ctrls.word_wrap.icon = get_theme_icon("Loop", "EditorIcons") _setup_colors() _ctrls.use_colors.button_pressed = true _use_highlighting(true) if get_parent() == get_tree().root: _test_running_setup() func _on_caret_changed(): var txt = str( "line:", _ctrls.output.get_caret_line(), " col:", _ctrls.output.get_caret_column() ) _ctrls.caret_position.text = str(txt) # ------------------ # Private # ------------------ # Call this after changes in colors and the like to get them to apply. reloads # the text of the output control. func _refresh_output(): var orig_pos = _ctrls.output.scroll_vertical var text = _ctrls.output.text _ctrls.output.text = text _ctrls.output.scroll_vertical = orig_pos func _create_highlighter(default_color = Color(1, 1, 1, 1)): var to_return = CodeHighlighter.new() to_return.function_color = default_color to_return.number_color = default_color to_return.symbol_color = default_color to_return.member_variable_color = default_color var keywords = [ ["Failed", Color.RED], ["Passed", Color.GREEN], ["Pending", Color.YELLOW], ["Orphans", Color.YELLOW], ["WARNING", Color.YELLOW], ["ERROR", Color.RED] ] for keyword in keywords: to_return.add_keyword_color(keyword[0], keyword[1]) return to_return func _setup_colors(): _ctrls.output.clear() var f_color = null if _ctrls.output.theme == null: f_color = get_theme_color("font_color") else: f_color = _ctrls.output.theme.font_color _highlighter = _create_highlighter() _ctrls.output.queue_redraw() func _use_highlighting(should): if should: _ctrls.output.syntax_highlighter = _highlighter else: _ctrls.output.syntax_highlighter = null _refresh_output() # ------------------ # Events # ------------------ func _on_CopyButton_pressed(): copy_to_clipboard() func _on_UseColors_pressed(): _use_highlighting(_ctrls.use_colors.button_pressed) func _on_ClearButton_pressed(): clear() func _on_ShowSearch_pressed(): show_search(_ctrls.show_search.button_pressed) func _on_SearchTerm_focus_entered(): _ctrls.search_bar.search_term.call_deferred("select_all") func _on_SearchNext_pressed(): _sr.find_next(_ctrls.search_bar.search_term.text) func _on_SearchPrev_pressed(): _sr.find_prev(_ctrls.search_bar.search_term.text) func _on_SearchTerm_text_changed(new_text): if new_text == "": _ctrls.output.deselect() else: _sr.find_next(new_text) func _on_SearchTerm_text_entered(new_text): if Input.is_physical_key_pressed(KEY_SHIFT): _sr.find_prev(new_text) else: _sr.find_next(new_text) func _on_SearchTerm_gui_input(event): if event is InputEventKey and !event.pressed and event.keycode == KEY_ESCAPE: show_search(false) func _on_WordWrap_pressed(): if _ctrls.word_wrap.button_pressed: _ctrls.output.wrap_mode = TextEdit.LINE_WRAPPING_BOUNDARY else: _ctrls.output.wrap_mode = TextEdit.LINE_WRAPPING_NONE _ctrls.output.queue_redraw() # ------------------ # Public # ------------------ func show_search(should): _ctrls.search_bar.bar.visible = should if should: _ctrls.search_bar.search_term.grab_focus() _ctrls.search_bar.search_term.select_all() _ctrls.show_search.button_pressed = should func search(text, start_pos, highlight = true): return _sr.find_next(text) func copy_to_clipboard(): var selected = _ctrls.output.get_selected_text() if selected != "": DisplayServer.clipboard_set(selected) else: DisplayServer.clipboard_set(_ctrls.output.text) func clear(): _ctrls.output.text = "" func _set_font(font_name, custom_name): var rtl = _ctrls.output if font_name == null: rtl.add_theme_font_override(custom_name, null) else: var dyn_font = FontFile.new() dyn_font.load_dynamic_font("res://addons/gut/fonts/" + font_name + ".ttf") rtl.add_theme_font_override(custom_name, dyn_font) func set_all_fonts(base_name): _font_name = GutUtils.nvl(base_name, "Default") if base_name == "Default": _set_font(null, "font") _set_font(null, "normal_font") _set_font(null, "bold_font") _set_font(null, "italics_font") _set_font(null, "bold_italics_font") else: _set_font(base_name + "-Regular", "font") _set_font(base_name + "-Regular", "normal_font") _set_font(base_name + "-Bold", "bold_font") _set_font(base_name + "-Italic", "italics_font") _set_font(base_name + "-BoldItalic", "bold_italics_font") func set_font_size(new_size): var rtl = _ctrls.output rtl.set("theme_override_font_sizes/font_size", new_size) # rtl.add_theme_font_size_override("font", new_size) # rtl.add_theme_font_size_override("normal_font", new_size) # rtl.add_theme_font_size_override("bold_font", new_size) # rtl.add_theme_font_size_override("italics_font", new_size) # rtl.add_theme_font_size_override("bold_italics_font", new_size) # if(rtl.get('custom_fonts/font') != null): # rtl.get('custom_fonts/font').size = new_size # rtl.get('custom_fonts/bold_italics_font').size = new_size # rtl.get('custom_fonts/bold_font').size = new_size # rtl.get('custom_fonts/italics_font').size = new_size # rtl.get('custom_fonts/normal_font').size = new_size func set_use_colors(value): pass func get_use_colors(): return false func get_rich_text_edit(): return _ctrls.output func load_file(path): var f = FileAccess.open(path, FileAccess.READ) if f == null: return var t = f.get_as_text() f = null # closes file _ctrls.output.text = t _ctrls.output.scroll_vertical = _ctrls.output.get_line_count() _ctrls.output.set_deferred("scroll_vertical", _ctrls.output.get_line_count()) func add_text(text): if is_inside_tree(): _ctrls.output.text += text func scroll_to_line(line): _ctrls.output.scroll_vertical = line _ctrls.output.set_caret_line(line)