Fix some bugs about search in code editor

1. Fix #61713;
2. Fix the bug when there are consecutive matches, forward searching will skip the adjacent item;
3. Fix the bug that enable the selection-only option will affect the operations in search mode.
This commit is contained in:
Rindbee 2022-06-06 17:47:57 +08:00
parent e413d1c9c3
commit ff0de4e220
2 changed files with 54 additions and 45 deletions

View file

@ -230,6 +230,14 @@ void FindReplaceBar::_replace_all() {
Point2i prev_match = Point2(-1, -1); Point2i prev_match = Point2(-1, -1);
bool selection_enabled = text_editor->has_selection(0); bool selection_enabled = text_editor->has_selection(0);
if (!is_selection_only()) {
text_editor->deselect();
selection_enabled = false;
} else {
result_line = -1;
result_col = -1;
}
Point2i selection_begin, selection_end; Point2i selection_begin, selection_end;
if (selection_enabled) { if (selection_enabled) {
selection_begin = Point2i(text_editor->get_selection_from_line(0), text_editor->get_selection_from_column(0)); selection_begin = Point2i(text_editor->get_selection_from_line(0), text_editor->get_selection_from_column(0));
@ -238,9 +246,6 @@ void FindReplaceBar::_replace_all() {
int vsval = text_editor->get_v_scroll(); int vsval = text_editor->get_v_scroll();
text_editor->set_caret_line(0, false, true, 0, 0);
text_editor->set_caret_column(0, true, 0);
String repl_text = get_replace_text(); String repl_text = get_replace_text();
int search_text_len = get_search_text().length(); int search_text_len = get_search_text().length();
@ -253,7 +258,11 @@ void FindReplaceBar::_replace_all() {
if (selection_enabled && is_selection_only()) { if (selection_enabled && is_selection_only()) {
text_editor->set_caret_line(selection_begin.width, false, true, 0, 0); text_editor->set_caret_line(selection_begin.width, false, true, 0, 0);
text_editor->set_caret_column(selection_begin.height, true, 0); text_editor->set_caret_column(selection_begin.height, true, 0);
} else {
text_editor->set_caret_line(0, false, true, 0, 0);
text_editor->set_caret_column(0, true, 0);
} }
if (search_current()) { if (search_current()) {
do { do {
// replace area // replace area
@ -269,7 +278,7 @@ void FindReplaceBar::_replace_all() {
text_editor->unfold_line(result_line); text_editor->unfold_line(result_line);
text_editor->select(result_line, result_col, result_line, match_to.y, 0); text_editor->select(result_line, result_col, result_line, match_to.y, 0);
if (selection_enabled && is_selection_only()) { if (selection_enabled) {
if (match_from < selection_begin || match_to > selection_end) { if (match_from < selection_begin || match_to > selection_end) {
break; // Done. break; // Done.
} }
@ -297,11 +306,9 @@ void FindReplaceBar::_replace_all() {
text_editor->set_caret_line(orig_cursor.x, false, true, 0, 0); text_editor->set_caret_line(orig_cursor.x, false, true, 0, 0);
text_editor->set_caret_column(orig_cursor.y, true, 0); text_editor->set_caret_column(orig_cursor.y, true, 0);
if (selection_enabled && is_selection_only()) { if (selection_enabled) {
// Reselect. // Reselect.
text_editor->select(selection_begin.x, selection_begin.y, selection_end.x, selection_end.y, 0); text_editor->select(selection_begin.x, selection_begin.y, selection_end.x, selection_end.y, 0);
} else {
text_editor->deselect(0);
} }
text_editor->set_v_scroll(vsval); text_editor->set_v_scroll(vsval);
@ -314,21 +321,28 @@ void FindReplaceBar::_replace_all() {
needs_to_count_results = true; needs_to_count_results = true;
} }
void FindReplaceBar::_get_search_from(int &r_line, int &r_col) { void FindReplaceBar::_get_search_from(int &r_line, int &r_col, bool p_is_searching_next) {
r_line = text_editor->get_caret_line(0); if (!text_editor->has_selection(0) || is_selection_only()) {
r_col = text_editor->get_caret_column(0); r_line = text_editor->get_caret_line(0);
r_col = text_editor->get_caret_column(0);
if (text_editor->has_selection(0) && is_selection_only()) { if (!p_is_searching_next && r_line == result_line && r_col >= result_col && r_col <= result_col + get_search_text().length()) {
r_col = result_col;
}
return; return;
} }
if (r_line == result_line && r_col >= result_col && r_col <= result_col + get_search_text().length()) { if (p_is_searching_next) {
r_col = result_col; r_line = text_editor->get_selection_to_line();
r_col = text_editor->get_selection_to_column();
} else {
r_line = text_editor->get_selection_from_line();
r_col = text_editor->get_selection_from_column();
} }
} }
void FindReplaceBar::_update_results_count() { void FindReplaceBar::_update_results_count() {
if (!needs_to_count_results && (result_line != -1)) { if (!needs_to_count_results && (result_line != -1) && results_count_to_current > 0) {
results_count_to_current += (flags & TextEdit::SEARCH_BACKWARDS) ? -1 : 1; results_count_to_current += (flags & TextEdit::SEARCH_BACKWARDS) ? -1 : 1;
if (results_count_to_current > results_count) { if (results_count_to_current > results_count) {
@ -340,9 +354,6 @@ void FindReplaceBar::_update_results_count() {
return; return;
} }
results_count = 0;
results_count_to_current = 0;
String searched = get_search_text(); String searched = get_search_text();
if (searched.is_empty()) { if (searched.is_empty()) {
return; return;
@ -350,6 +361,8 @@ void FindReplaceBar::_update_results_count() {
needs_to_count_results = false; needs_to_count_results = false;
results_count = 0;
for (int i = 0; i < text_editor->get_line_count(); i++) { for (int i = 0; i < text_editor->get_line_count(); i++) {
String line_text = text_editor->get_line(i); String line_text = text_editor->get_line(i);
@ -373,8 +386,13 @@ void FindReplaceBar::_update_results_count() {
results_count++; results_count++;
if (i == result_line && col_pos == result_col) { if (i == result_line) {
results_count_to_current = results_count; if (col_pos == result_col) {
results_count_to_current = results_count;
} else if (col_pos < result_col && col_pos + searched.length() > result_col) {
col_pos = result_col;
results_count_to_current = results_count;
}
} }
col_pos += searched.length(); col_pos += searched.length();
@ -392,10 +410,10 @@ void FindReplaceBar::_update_matches_label() {
if (results_count == 0) { if (results_count == 0) {
matches_label->set_text("No match"); matches_label->set_text("No match");
} else if (results_count == 1) { } else if (results_count_to_current == -1) {
matches_label->set_text(vformat(TTR("%d match"), results_count)); matches_label->set_text(vformat(TTRN("%d match", "%d matches", results_count), results_count));
} else { } else {
matches_label->set_text(vformat(TTR("%d of %d matches"), results_count_to_current, results_count)); matches_label->set_text(vformat(TTRN("%d of %d match", "%d of %d matches", results_count), results_count_to_current, results_count));
} }
} }
} }
@ -417,6 +435,10 @@ bool FindReplaceBar::search_current() {
} }
bool FindReplaceBar::search_prev() { bool FindReplaceBar::search_prev() {
if (is_selection_only() && !replace_all_mode) {
return false;
}
if (!is_visible()) { if (!is_visible()) {
popup_search(true); popup_search(true);
} }
@ -435,9 +457,6 @@ bool FindReplaceBar::search_prev() {
int line, col; int line, col;
_get_search_from(line, col); _get_search_from(line, col);
if (text_editor->has_selection(0)) {
col--; // Skip currently selected word.
}
col -= text.length(); col -= text.length();
if (col < 0) { if (col < 0) {
@ -452,17 +471,15 @@ bool FindReplaceBar::search_prev() {
} }
bool FindReplaceBar::search_next() { bool FindReplaceBar::search_next() {
if (is_selection_only() && !replace_all_mode) {
return false;
}
if (!is_visible()) { if (!is_visible()) {
popup_search(true); popup_search(true);
} }
flags = 0; flags = 0;
String text;
if (replace_all_mode) {
text = get_replace_text();
} else {
text = get_search_text();
}
if (is_whole_words()) { if (is_whole_words()) {
flags |= TextEdit::SEARCH_WHOLE_WORDS; flags |= TextEdit::SEARCH_WHOLE_WORDS;
@ -472,18 +489,7 @@ bool FindReplaceBar::search_next() {
} }
int line, col; int line, col;
_get_search_from(line, col); _get_search_from(line, col, true);
if (line == result_line && col == result_col) {
col += text.length();
if (col > text_editor->get_line(line).length()) {
line += 1;
if (line >= text_editor->get_line_count()) {
line = 0;
}
col = 0;
}
}
return _search(flags, line, col); return _search(flags, line, col);
} }
@ -513,8 +519,10 @@ void FindReplaceBar::_show_search(bool p_focus_replace, bool p_show_only) {
search_text->call_deferred(SNAME("grab_focus")); search_text->call_deferred(SNAME("grab_focus"));
} }
if (text_editor->has_selection(0) && !selection_only->is_pressed()) { if (text_editor->has_selection(0) && !is_selection_only()) {
search_text->set_text(text_editor->get_selected_text(0)); search_text->set_text(text_editor->get_selected_text(0));
result_line = text_editor->get_selection_from_line();
result_col = text_editor->get_selection_from_column();
} }
if (!get_search_text().is_empty()) { if (!get_search_text().is_empty()) {
@ -538,6 +546,7 @@ void FindReplaceBar::popup_search(bool p_show_only) {
replace_text->hide(); replace_text->hide();
hbc_button_replace->hide(); hbc_button_replace->hide();
hbc_option_replace->hide(); hbc_option_replace->hide();
selection_only->set_pressed(false);
_show_search(false, p_show_only); _show_search(false, p_show_only);
} }

View file

@ -92,7 +92,7 @@ class FindReplaceBar : public HBoxContainer {
bool replace_all_mode = false; bool replace_all_mode = false;
bool preserve_cursor = false; bool preserve_cursor = false;
void _get_search_from(int &r_line, int &r_col); void _get_search_from(int &r_line, int &r_col, bool p_is_searching_next = false);
void _update_results_count(); void _update_results_count();
void _update_matches_label(); void _update_matches_label();