Fix ensure_scoll bug where logical and visual lines were mixed up.

This commit is contained in:
2025-12-10 22:08:30 +00:00
parent 3927899428
commit 260867e225

View File

@@ -347,29 +347,53 @@ void cursor_left(Editor *editor, uint32_t number) {
} }
void ensure_scroll(Editor *editor) { void ensure_scroll(Editor *editor) {
LineIterator *it = begin_l_iter(editor->root, editor->scroll.row); if (editor->cursor.row < editor->scroll.row) {
uint32_t rendered_lines = 0; uint32_t visual_delta = 0;
uint32_t line_index = editor->scroll.row; LineIterator *it = begin_l_iter(editor->root, editor->cursor.row);
char *line_content = nullptr; for (uint32_t i = editor->cursor.row; i < editor->scroll.row; i++) {
while (rendered_lines <= editor->size.row) { char *line = next_line(it);
line_content = next_line(it); if (!line)
if (!line_content)
break; break;
char *line_content_t = line_content; uint32_t len = grapheme_strlen(line);
if (rendered_lines == 0) visual_delta += (len + editor->size.col - 1) / editor->size.col;
line_content_t = line_content + editor->scroll.col; free(line);
uint32_t len = grapheme_strlen(line_content_t);
free(line_content);
uint32_t wrapped_lines = (len + editor->size.col - 1) / editor->size.col;
rendered_lines += wrapped_lines;
line_index++;
} }
line_index -= 2;
free(it); free(it);
if (editor->cursor.row >= line_index && line_content) scroll_up(editor, visual_delta);
scroll_down(editor, editor->cursor.row - line_index); return;
if (editor->cursor.row < editor->scroll.row) }
scroll_up(editor, editor->scroll.row - editor->cursor.row); uint32_t current_visual_y = 0;
LineIterator *it = begin_l_iter(editor->root, editor->scroll.row);
uint32_t i = editor->scroll.row;
char *line = nullptr;
bool found_cursor = false;
while ((line = next_line(it)) != nullptr) {
uint32_t lines_in_chunk;
uint32_t offset = (i == editor->scroll.row) ? editor->scroll.col : 0;
if (i == editor->cursor.row) {
uint32_t cursor_sub_row = editor->cursor.col / editor->size.col;
if (i == editor->scroll.row)
cursor_sub_row -= offset / editor->size.col;
current_visual_y += cursor_sub_row;
found_cursor = true;
free(line);
break;
}
uint32_t len = grapheme_strlen(line);
uint32_t visible_len = len - offset;
if (visible_len == 0)
visible_len = 1;
lines_in_chunk = (visible_len + editor->size.col - 1) / editor->size.col;
current_visual_y += lines_in_chunk;
free(line);
i++;
}
free(it);
if (found_cursor)
if (current_visual_y >= editor->size.row) {
uint32_t needed_scroll = current_visual_y - editor->size.row + 1;
scroll_down(editor, needed_scroll);
}
} }
void fold(Editor *editor, uint32_t start_line, uint32_t end_line) { void fold(Editor *editor, uint32_t start_line, uint32_t end_line) {