2024-02-11 16:10:17 +00:00
|
|
|
/**
|
|
|
|
* @file EditorWidget.cpp
|
|
|
|
* @author apio (cloudapio.eu)
|
|
|
|
* @brief Multiline text editing widget.
|
|
|
|
*
|
|
|
|
* @copyright Copyright (c) 2024, the Luna authors.
|
|
|
|
*
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include "EditorWidget.h"
|
|
|
|
#include <ctype.h>
|
2024-03-29 19:36:44 +00:00
|
|
|
#include <luna/PathParser.h>
|
2024-09-19 16:27:16 +00:00
|
|
|
#include <luna/RefString.h>
|
2024-02-11 16:10:17 +00:00
|
|
|
#include <luna/Utf8.h>
|
|
|
|
#include <os/File.h>
|
2024-03-14 12:09:03 +00:00
|
|
|
#include <os/FileSystem.h>
|
2024-02-11 16:10:17 +00:00
|
|
|
#include <ui/App.h>
|
2024-09-19 16:27:16 +00:00
|
|
|
#include <ui/Dialog.h>
|
2024-02-11 16:10:17 +00:00
|
|
|
|
2024-03-29 19:17:12 +00:00
|
|
|
EditorWidget::EditorWidget(SharedPtr<ui::Font> font) : ui::TextInput(), m_font(font)
|
2024-02-11 16:10:17 +00:00
|
|
|
{
|
|
|
|
recalculate_lines();
|
|
|
|
}
|
|
|
|
|
|
|
|
Result<void> EditorWidget::load_file(const os::Path& path)
|
|
|
|
{
|
2024-03-14 12:09:03 +00:00
|
|
|
struct stat st;
|
2024-03-29 19:41:16 +00:00
|
|
|
auto rc = os::FileSystem::stat(path, st, true);
|
2024-03-14 12:09:03 +00:00
|
|
|
|
2024-03-29 19:41:16 +00:00
|
|
|
if (!rc.has_error() && !S_ISREG(st.st_mode))
|
2024-03-14 12:09:03 +00:00
|
|
|
{
|
2024-09-19 16:27:16 +00:00
|
|
|
auto message = TRY(RefString::format("%s is not a regular file", path.name().chars()));
|
|
|
|
ui::Dialog::show_message("Error", message.view());
|
2024-03-14 12:09:03 +00:00
|
|
|
return {};
|
|
|
|
}
|
|
|
|
|
2024-02-11 16:10:17 +00:00
|
|
|
os::eprintln("Loading file: %s", path.name().chars());
|
|
|
|
|
|
|
|
auto file = TRY(os::File::open_or_create(path, os::File::ReadOnly));
|
|
|
|
|
|
|
|
m_data = TRY(file->read_all());
|
|
|
|
|
|
|
|
os::eprintln("Read %zu bytes.", m_data.size());
|
|
|
|
|
|
|
|
m_cursor = m_data.size();
|
|
|
|
|
2024-09-19 16:27:16 +00:00
|
|
|
m_path = TRY(String::from_string_view(path.name()));
|
2024-02-27 19:11:14 +00:00
|
|
|
|
2024-09-19 16:27:16 +00:00
|
|
|
auto basename = TRY(PathParser::basename(m_path.view()));
|
2024-03-29 19:36:44 +00:00
|
|
|
|
|
|
|
String title = TRY(String::format("Text Editor - %s"_sv, basename.chars()));
|
2024-03-29 19:17:12 +00:00
|
|
|
window()->set_title(title.view());
|
2024-02-27 19:11:14 +00:00
|
|
|
|
2024-02-11 16:10:17 +00:00
|
|
|
TRY(recalculate_lines());
|
|
|
|
|
|
|
|
return {};
|
|
|
|
}
|
|
|
|
|
|
|
|
Result<ui::EventResult> EditorWidget::handle_key_event(const ui::KeyEventRequest& request)
|
|
|
|
{
|
|
|
|
// Avoid handling "key released" events
|
|
|
|
if (!request.pressed) return ui::EventResult::DidNotHandle;
|
|
|
|
|
2024-03-06 21:41:21 +00:00
|
|
|
if (request.code == moon::K_UpArrow)
|
2024-02-11 16:10:17 +00:00
|
|
|
{
|
|
|
|
if (m_cursor_position.y > 0) m_cursor_position.y--;
|
|
|
|
else
|
|
|
|
return ui::EventResult::DidNotHandle;
|
|
|
|
recalculate_cursor_index();
|
2024-03-29 19:17:12 +00:00
|
|
|
update_cursor();
|
2024-02-11 16:10:17 +00:00
|
|
|
return ui::EventResult::DidHandle;
|
|
|
|
}
|
2024-03-06 21:41:21 +00:00
|
|
|
|
|
|
|
if (request.code == moon::K_DownArrow)
|
|
|
|
{
|
2024-02-11 16:10:17 +00:00
|
|
|
if (m_cursor_position.y + 1 < (int)m_lines.size()) m_cursor_position.y++;
|
|
|
|
else
|
|
|
|
return ui::EventResult::DidNotHandle;
|
|
|
|
recalculate_cursor_index();
|
2024-03-29 19:17:12 +00:00
|
|
|
update_cursor();
|
2024-02-11 16:10:17 +00:00
|
|
|
return ui::EventResult::DidHandle;
|
|
|
|
}
|
2024-03-06 21:41:21 +00:00
|
|
|
|
|
|
|
if (request.code == moon::K_LeftArrow)
|
|
|
|
{
|
2024-02-11 16:10:17 +00:00
|
|
|
if (m_cursor > 0) m_cursor--;
|
|
|
|
else
|
|
|
|
return ui::EventResult::DidNotHandle;
|
|
|
|
recalculate_cursor_position();
|
2024-03-29 19:17:12 +00:00
|
|
|
update_cursor();
|
2024-02-11 16:10:17 +00:00
|
|
|
return ui::EventResult::DidHandle;
|
|
|
|
}
|
2024-03-06 21:41:21 +00:00
|
|
|
|
|
|
|
if (request.code == moon::K_RightArrow)
|
|
|
|
{
|
2024-02-11 16:10:17 +00:00
|
|
|
if (m_cursor < m_data.size()) m_cursor++;
|
|
|
|
else
|
|
|
|
return ui::EventResult::DidNotHandle;
|
|
|
|
recalculate_cursor_position();
|
2024-03-29 19:17:12 +00:00
|
|
|
update_cursor();
|
2024-02-11 16:10:17 +00:00
|
|
|
return ui::EventResult::DidHandle;
|
|
|
|
}
|
2024-03-06 21:41:21 +00:00
|
|
|
|
|
|
|
if (request.code == moon::K_Backspace)
|
|
|
|
{
|
|
|
|
if (m_cursor == 0) return ui::EventResult::DidNotHandle;
|
|
|
|
m_cursor--;
|
|
|
|
|
2024-03-29 19:17:12 +00:00
|
|
|
delete_current_character();
|
2024-03-06 21:41:21 +00:00
|
|
|
|
|
|
|
TRY(recalculate_lines());
|
|
|
|
|
2024-03-29 19:17:12 +00:00
|
|
|
update_cursor();
|
2024-03-06 21:41:21 +00:00
|
|
|
|
|
|
|
return ui::EventResult::DidHandle;
|
2024-02-27 19:11:14 +00:00
|
|
|
}
|
2024-03-06 21:41:21 +00:00
|
|
|
|
|
|
|
if (request.letter != '\n' && iscntrl(request.letter)) return ui::EventResult::DidNotHandle;
|
|
|
|
|
|
|
|
if (m_cursor == m_data.size()) TRY(m_data.append_data((const u8*)&request.letter, 1));
|
|
|
|
else
|
2024-03-29 19:17:12 +00:00
|
|
|
TRY(insert_character(request.letter));
|
2024-03-06 21:41:21 +00:00
|
|
|
|
|
|
|
m_cursor++;
|
|
|
|
TRY(recalculate_lines());
|
|
|
|
|
2024-03-29 19:17:12 +00:00
|
|
|
update_cursor();
|
2024-03-06 21:41:21 +00:00
|
|
|
|
|
|
|
return ui::EventResult::DidHandle;
|
2024-02-11 16:10:17 +00:00
|
|
|
}
|
|
|
|
|
2024-09-19 16:27:16 +00:00
|
|
|
Result<void> EditorWidget::save_file_as()
|
|
|
|
{
|
|
|
|
ui::Dialog::show_input_dialog(
|
|
|
|
"Save file as...", "Please enter the path to save this file to:", [this](StringView path) {
|
|
|
|
m_path = String::from_string_view(path).release_value();
|
|
|
|
auto rc = save_file();
|
|
|
|
if (rc.has_error())
|
|
|
|
{
|
|
|
|
os::eprintln("Failed to save file %s: %s", m_path.chars(), rc.error_string());
|
|
|
|
ui::Dialog::show_message("Error", "Failed to save file");
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
auto basename = PathParser::basename(m_path.view()).release_value();
|
|
|
|
|
|
|
|
String title = String::format("Text Editor - %s"_sv, basename.chars()).release_value();
|
|
|
|
window()->set_title(title.view());
|
|
|
|
}
|
|
|
|
});
|
|
|
|
|
|
|
|
return {};
|
|
|
|
}
|
|
|
|
|
2024-02-27 19:11:14 +00:00
|
|
|
Result<void> EditorWidget::save_file()
|
|
|
|
{
|
2024-09-19 16:27:16 +00:00
|
|
|
if (m_path.is_empty())
|
2024-02-27 19:11:14 +00:00
|
|
|
{
|
2024-09-19 16:27:16 +00:00
|
|
|
TRY(save_file_as());
|
2024-10-26 12:00:57 +00:00
|
|
|
return {};
|
2024-02-27 19:11:14 +00:00
|
|
|
}
|
|
|
|
|
2024-09-19 16:27:16 +00:00
|
|
|
auto file = TRY(os::File::open_or_create(m_path.view(), os::File::WriteOnly));
|
2024-02-27 19:11:14 +00:00
|
|
|
return file->write(m_data);
|
|
|
|
}
|
|
|
|
|
2024-02-11 16:10:17 +00:00
|
|
|
Result<void> EditorWidget::draw(ui::Canvas& canvas)
|
|
|
|
{
|
|
|
|
int visible_lines = canvas.height / m_font->height();
|
|
|
|
int visible_columns = canvas.width / m_font->width();
|
|
|
|
|
|
|
|
if ((usize)visible_lines > m_lines.size()) visible_lines = static_cast<int>(m_lines.size());
|
|
|
|
|
|
|
|
for (int i = 0; i < visible_lines; i++)
|
|
|
|
{
|
|
|
|
auto line = m_lines[i];
|
|
|
|
if (line.begin == line.end) continue;
|
|
|
|
|
|
|
|
auto slice = TRY(m_data.slice(line.begin, line.end - line.begin));
|
|
|
|
auto string = TRY(
|
|
|
|
String::from_string_view(StringView::from_fixed_size_cstring((const char*)slice, line.end - line.begin)));
|
|
|
|
|
|
|
|
Utf8StringDecoder decoder(string.chars());
|
|
|
|
wchar_t buf[4096];
|
|
|
|
decoder.decode(buf, sizeof(buf)).release_value();
|
|
|
|
|
|
|
|
int characters_to_render = (int)wcslen(buf);
|
|
|
|
|
|
|
|
for (int j = 0; j < visible_columns && j < characters_to_render; j++)
|
|
|
|
{
|
|
|
|
auto subcanvas =
|
|
|
|
canvas.subcanvas({ j * m_font->width(), i * m_font->height(), m_font->width(), m_font->height() });
|
|
|
|
m_font->render(buf[j], ui::WHITE, subcanvas);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Draw the cursor
|
|
|
|
if (m_cursor_position.x < visible_columns && m_cursor_position.y < visible_lines && m_cursor_activated)
|
|
|
|
{
|
|
|
|
canvas
|
|
|
|
.subcanvas(
|
|
|
|
{ m_cursor_position.x * m_font->width(), m_cursor_position.y * m_font->height(), 1, m_font->height() })
|
|
|
|
.fill(ui::WHITE);
|
|
|
|
}
|
|
|
|
|
|
|
|
return {};
|
|
|
|
}
|
|
|
|
|
|
|
|
Result<void> EditorWidget::recalculate_lines()
|
|
|
|
{
|
|
|
|
m_lines.clear();
|
|
|
|
|
|
|
|
Line l;
|
|
|
|
l.begin = 0;
|
|
|
|
for (usize i = 0; i < m_data.size(); i++)
|
|
|
|
{
|
|
|
|
if (m_data.data()[i] == '\n')
|
|
|
|
{
|
|
|
|
l.end = i;
|
|
|
|
TRY(m_lines.try_append(l));
|
|
|
|
l.begin = i + 1;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
l.end = m_data.size();
|
|
|
|
TRY(m_lines.try_append(l));
|
|
|
|
|
|
|
|
recalculate_cursor_position();
|
|
|
|
|
|
|
|
return {};
|
|
|
|
}
|
|
|
|
|
|
|
|
void EditorWidget::recalculate_cursor_position()
|
|
|
|
{
|
|
|
|
if (m_cursor == 0) m_cursor_position = { 0, 0 };
|
|
|
|
|
|
|
|
for (int i = 0; i < (int)m_lines.size(); i++)
|
|
|
|
{
|
|
|
|
auto line = m_lines[i];
|
|
|
|
if (m_cursor >= line.begin && m_cursor <= line.end)
|
|
|
|
{
|
|
|
|
m_cursor_position.x = (int)(m_cursor - line.begin);
|
|
|
|
m_cursor_position.y = i;
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
unreachable();
|
|
|
|
}
|
|
|
|
|
|
|
|
void EditorWidget::recalculate_cursor_index()
|
|
|
|
{
|
|
|
|
m_cursor = m_lines[m_cursor_position.y].begin + m_cursor_position.x;
|
|
|
|
if (m_cursor > m_lines[m_cursor_position.y].end)
|
|
|
|
{
|
|
|
|
m_cursor = m_lines[m_cursor_position.y].end;
|
|
|
|
recalculate_cursor_position();
|
|
|
|
}
|
|
|
|
}
|