port deleting text (no range delete yet)

Pull Request: https://github.com/showseeker/pilot-api/pull/67
plugins
Patrick Cleavelin 2023-12-23 20:49:46 -06:00
parent 38539acf5c
commit 16b8b4084d
1 changed files with 172 additions and 58 deletions

View File

@ -61,7 +61,7 @@ FileBufferIndex :: struct {
Cursor :: struct { Cursor :: struct {
col: int, col: int,
line: int, line: int,
buffer_index: FileBufferIndex, index: FileBufferIndex,
} }
Glyph :: struct #packed { Glyph :: struct #packed {
@ -88,7 +88,7 @@ FileBuffer :: struct {
} }
FileBufferIter :: struct { FileBufferIter :: struct {
index: FileBufferIndex, cursor: Cursor,
buffer: ^FileBuffer, buffer: ^FileBuffer,
} }
@ -104,27 +104,66 @@ State :: struct {
new_file_buffer_iter_from_beginning :: proc(file_buffer: ^FileBuffer) -> FileBufferIter { new_file_buffer_iter_from_beginning :: proc(file_buffer: ^FileBuffer) -> FileBufferIter {
return FileBufferIter { buffer = file_buffer }; return FileBufferIter { buffer = file_buffer };
} }
new_file_buffer_iter_with_index :: proc(file_buffer: ^FileBuffer, index: FileBufferIndex) -> FileBufferIter { new_file_buffer_iter_with_cursor :: proc(file_buffer: ^FileBuffer, cursor: Cursor) -> FileBufferIter {
return FileBufferIter { buffer = file_buffer, index = index }; return FileBufferIter { buffer = file_buffer, cursor = cursor };
} }
new_file_buffer_iter :: proc{new_file_buffer_iter_from_beginning, new_file_buffer_iter_with_index}; new_file_buffer_iter :: proc{new_file_buffer_iter_from_beginning, new_file_buffer_iter_with_cursor};
iterate_file_buffer :: proc(it: ^FileBufferIter) -> (character: u8, idx: FileBufferIndex, cond: bool) { iterate_file_buffer :: proc(it: ^FileBufferIter) -> (character: u8, idx: FileBufferIndex, cond: bool) {
if it.index.slice_index >= len(it.buffer.content_slices) || it.index.content_index >= len(it.buffer.content_slices[it.index.slice_index]) { if it.cursor.index.slice_index >= len(it.buffer.content_slices) || it.cursor.index.content_index >= len(it.buffer.content_slices[it.cursor.index.slice_index]) {
return; return;
} }
cond = true; cond = true;
character = it.buffer.content_slices[it.index.slice_index][it.index.content_index]; character = it.buffer.content_slices[it.cursor.index.slice_index][it.cursor.index.content_index];
if character == '\n' {
it.cursor.col = 0;
it.cursor.line += 1;
} else {
it.cursor.col += 1;
}
it.index.content_index += 1; it.cursor.index.content_index += 1;
if it.index.content_index >= len(it.buffer.content_slices[it.index.slice_index]) { if it.cursor.index.content_index >= len(it.buffer.content_slices[it.cursor.index.slice_index]) {
it.index.content_index = 0; it.cursor.index.content_index = 0;
it.index.slice_index += 1; it.cursor.index.slice_index += 1;
} }
return; return;
} }
iterate_file_buffer_reverse_mangle_cursor :: proc(it: ^FileBufferIter) -> (character: u8, idx: FileBufferIndex, cond: bool) {
if it.cursor.index.content_index == 0 {
if it.cursor.index.slice_index > 0 {
it.cursor.index.slice_index -= 1;
it.cursor.index.content_index = len(it.buffer.content_slices[it.cursor.index.slice_index])-1;
} else {
return 0, it.cursor.index, false;
}
} else {
it.cursor.index.content_index -= 1;
}
return it.buffer.content_slices[it.cursor.index.slice_index][it.cursor.index.content_index], it.cursor.index, true;
}
iterate_file_buffer_reverse :: proc(it: ^FileBufferIter) -> (character: u8, idx: FileBufferIndex, cond: bool) {
if character, idx, cond = iterate_file_buffer_reverse_mangle_cursor(it); cond {
if character == '\n' {
if it.cursor.line > 0 {
line_length := file_buffer_line_length(it.buffer, it.cursor.index);
if line_length < 0 { line_length = 0; }
it.cursor.line -= 1;
it.cursor.col = line_length;
} else {
return 0, it.cursor.index, false;
}
} else {
it.cursor.col -= 1;
}
}
return character, it.cursor.index, cond;
}
update_file_buffer_index_from_cursor :: proc(buffer: ^FileBuffer) { update_file_buffer_index_from_cursor :: proc(buffer: ^FileBuffer) {
it := new_file_buffer_iter(buffer); it := new_file_buffer_iter(buffer);
@ -148,31 +187,29 @@ update_file_buffer_index_from_cursor :: proc(buffer: ^FileBuffer) {
before_it = it; before_it = it;
} }
buffer.cursor.buffer_index = before_it.index; // FIXME: just swap cursors
buffer.cursor.index = before_it.cursor.index;
} }
file_buffer_line_length :: proc(buffer: ^FileBuffer) -> int { file_buffer_line_length :: proc(buffer: ^FileBuffer, index: FileBufferIndex) -> int {
line_length := 0; line_length := 0;
rendered_line := 0;
for i in 0..<len(buffer.content_slices) { left_it := new_file_buffer_iter_with_cursor(buffer, Cursor { index = index });
content := buffer.content_slices[i]; for character in iterate_file_buffer_reverse_mangle_cursor(&left_it) {
if character == '\n' {
for c in content { break;
if c == '\n' {
rendered_line += 1;
if rendered_line > buffer.cursor.line {
return line_length;
} }
continue;
}
if rendered_line == buffer.cursor.line {
line_length += 1; line_length += 1;
} }
right_it := new_file_buffer_iter_with_cursor(buffer, Cursor { index = index });
for character in iterate_file_buffer(&right_it) {
if character == '\n' {
break;
} }
line_length += 1;
} }
return line_length; return line_length;
@ -180,34 +217,59 @@ file_buffer_line_length :: proc(buffer: ^FileBuffer) -> int {
move_cursor_up :: proc(buffer: ^FileBuffer) { move_cursor_up :: proc(buffer: ^FileBuffer) {
if buffer.cursor.line > 0 { if buffer.cursor.line > 0 {
buffer.cursor.line -= 1; current_line := buffer.cursor.line;
current_col := buffer.cursor.col;
it := new_file_buffer_iter_with_cursor(buffer, buffer.cursor);
for _ in iterate_file_buffer_reverse(&it) {
if it.cursor.line <= current_line-1 {
break;
}
}
// the `it.cursor.col > 0` is here because after the above loop, the
// iterator is left on the new line instead of the last character of the line
if it.cursor.col > current_col || it.cursor.col > 0 {
for _ in iterate_file_buffer_reverse(&it) {
if it.cursor.col <= current_col {
break;
}
}
}
buffer.cursor = it.cursor;
if buffer.cursor.line < buffer.top_line + 5 && buffer.cursor.line >= 4 { if buffer.cursor.line < buffer.top_line + 5 && buffer.cursor.line >= 4 {
buffer.top_line = buffer.cursor.line - 4; buffer.top_line = buffer.cursor.line - 4;
} }
line_length := file_buffer_line_length(buffer);
if buffer.cursor.col >= line_length {
buffer.cursor.col = line_length < 1 ? 0 : line_length - 1;
}
update_file_buffer_index_from_cursor(buffer);
} }
} }
move_cursor_down :: proc(buffer: ^FileBuffer) { move_cursor_down :: proc(buffer: ^FileBuffer) {
buffer.cursor.line += 1; current_line := buffer.cursor.line;
current_col := buffer.cursor.col;
it := new_file_buffer_iter_with_cursor(buffer, buffer.cursor);
for _ in iterate_file_buffer(&it) {
if it.cursor.line >= current_line+1 {
break;
}
}
line_length := file_buffer_line_length(buffer, it.cursor.index);
if it.cursor.col < line_length && it.cursor.col < current_col {
for _ in iterate_file_buffer(&it) {
if it.cursor.col >= line_length-1 || it.cursor.col >= current_col {
break;
}
}
}
buffer.cursor = it.cursor;
if buffer.cursor.line > buffer.top_line + (buffer.glyph_buffer_height - 5) { if buffer.cursor.line > buffer.top_line + (buffer.glyph_buffer_height - 5) {
buffer.top_line = buffer.cursor.line - (buffer.glyph_buffer_height - 5); buffer.top_line = buffer.cursor.line - (buffer.glyph_buffer_height - 5);
} }
line_length := file_buffer_line_length(buffer);
if buffer.cursor.col >= line_length {
buffer.cursor.col = line_length < 1 ? 0 : line_length - 1;
}
update_file_buffer_index_from_cursor(buffer);
} }
move_cursor_left :: proc(buffer: ^FileBuffer) { move_cursor_left :: proc(buffer: ^FileBuffer) {
@ -218,7 +280,7 @@ move_cursor_left :: proc(buffer: ^FileBuffer) {
} }
move_cursor_right :: proc(buffer: ^FileBuffer) { move_cursor_right :: proc(buffer: ^FileBuffer) {
line_length := file_buffer_line_length(buffer); line_length := file_buffer_line_length(buffer, buffer.cursor.index);
if line_length > 0 && buffer.cursor.col < line_length-1 { if line_length > 0 && buffer.cursor.col < line_length-1 {
buffer.cursor.col += 1; buffer.cursor.col += 1;
@ -410,30 +472,78 @@ insert_content :: proc(buffer: ^FileBuffer, to_be_inserted: []u8) {
// TODO: is this even needed? would mean that the cursor isn't always in a valid state. // TODO: is this even needed? would mean that the cursor isn't always in a valid state.
update_file_buffer_index_from_cursor(buffer); update_file_buffer_index_from_cursor(buffer);
before_it := new_file_buffer_iter(buffer, buffer.cursor.buffer_index); it := new_file_buffer_iter(buffer, buffer.cursor);
length := append(&buffer.added_content, ..to_be_inserted); length := append(&buffer.added_content, ..to_be_inserted);
inserted_slice: []u8 = buffer.added_content[len(buffer.added_content)-length:]; inserted_slice: []u8 = buffer.added_content[len(buffer.added_content)-length:];
if before_it.index.content_index == 0 { if it.cursor.index.content_index == 0 {
// insertion happening in beginning of content slice // insertion happening in beginning of content slice
inject_at(&buffer.content_slices, 0, inserted_slice); inject_at(&buffer.content_slices, buffer.cursor.index.slice_index, inserted_slice);
} }
else { else {
// insertion is happening in middle of content slice // insertion is happening in middle of content slice
// cut current slice // cut current slice
end_slice := buffer.content_slices[before_it.index.slice_index][before_it.index.content_index:]; end_slice := buffer.content_slices[it.cursor.index.slice_index][it.cursor.index.content_index:];
buffer.content_slices[before_it.index.slice_index] = buffer.content_slices[before_it.index.slice_index][:before_it.index.content_index]; buffer.content_slices[it.cursor.index.slice_index] = buffer.content_slices[it.cursor.index.slice_index][:it.cursor.index.content_index];
inject_at(&buffer.content_slices, before_it.index.slice_index+1, inserted_slice); inject_at(&buffer.content_slices, it.cursor.index.slice_index+1, inserted_slice);
inject_at(&buffer.content_slices, before_it.index.slice_index+2, end_slice); inject_at(&buffer.content_slices, it.cursor.index.slice_index+2, end_slice);
} }
update_file_buffer_index_from_cursor(buffer); update_file_buffer_index_from_cursor(buffer);
} }
// TODO: potentially add FileBufferIndex as parameter
split_content_slice :: proc(buffer: ^FileBuffer) {
if buffer.cursor.index.content_index == 0 {
return;
}
end_slice := buffer.content_slices[buffer.cursor.index.slice_index][buffer.cursor.index.content_index:];
buffer.content_slices[buffer.cursor.index.slice_index] = buffer.content_slices[buffer.cursor.index.slice_index][:buffer.cursor.index.content_index];
inject_at(&buffer.content_slices, buffer.cursor.index.slice_index+1, end_slice);
// TODO: maybe move this out of this function
buffer.cursor.index.slice_index += 1;
buffer.cursor.index.content_index = 0;
}
delete_content :: proc(buffer: ^FileBuffer, amount: int) {
if amount <= len(buffer.input_buffer) {
runtime.resize(&buffer.input_buffer, len(buffer.input_buffer)-amount);
} else {
amount := amount - len(buffer.input_buffer);
runtime.clear(&buffer.input_buffer);
split_content_slice(buffer);
it := new_file_buffer_iter_with_cursor(buffer, buffer.cursor);
// go back one (to be at the end of the content slice)
iterate_file_buffer_reverse(&it);
for i in 0..<amount {
content_slice_ptr := &buffer.content_slices[it.cursor.index.slice_index];
if len(content_slice_ptr^) == 1 {
// move cursor to previous content_slice so we can delete the current one
iterate_file_buffer_reverse(&it);
runtime.ordered_remove(&buffer.content_slices, it.cursor.index.slice_index+1);
} else {
iterate_file_buffer_reverse(&it);
content_slice_ptr^ = content_slice_ptr^[:len(content_slice_ptr^)-1];
}
}
iterate_file_buffer(&it);
buffer.cursor = it.cursor;
}
}
// TODO: use buffer list in state // TODO: use buffer list in state
do_insert_mode :: proc(state: ^State, buffer: ^FileBuffer) { do_insert_mode :: proc(state: ^State, buffer: ^FileBuffer) {
key := raylib.GetCharPressed(); key := raylib.GetCharPressed();
@ -457,6 +567,10 @@ do_insert_mode :: proc(state: ^State, buffer: ^FileBuffer) {
runtime.clear(&buffer.input_buffer); runtime.clear(&buffer.input_buffer);
return; return;
} }
if raylib.IsKeyPressed(.BACKSPACE) {
delete_content(buffer, 1);
}
} }
main :: proc() { main :: proc() {
@ -481,7 +595,7 @@ main :: proc() {
raylib.ClearBackground(raylib.GetColor(0x232136ff)); raylib.ClearBackground(raylib.GetColor(0x232136ff));
draw_file_buffer(&state, &buffer, 0, 32, font); draw_file_buffer(&state, &buffer, 0, 32, font);
raylib.DrawTextEx(font, raylib.TextFormat("Line: %d, Col: %d", buffer.cursor.line + 1, buffer.cursor.col + 1), raylib.Vector2 { 0, 0 }, source_font_height, 0, raylib.DARKGRAY); raylib.DrawTextEx(font, raylib.TextFormat("Line: %d, Col: %d --- Slice Index: %d, Content Index: %d", buffer.cursor.line + 1, buffer.cursor.col + 1, buffer.cursor.index.slice_index, buffer.cursor.index.content_index), raylib.Vector2 { 0, 0 }, source_font_height, 0, raylib.DARKGRAY);
} }
switch state.mode { switch state.mode {