mirror of
https://github.com/RGBCube/serenity
synced 2025-10-24 11:32:08 +00:00
161 lines
5.4 KiB
C++
161 lines
5.4 KiB
C++
/*
|
|
* Copyright (c) 2021, Itamar S. <itamar8910@gmail.com>
|
|
*
|
|
* SPDX-License-Identifier: BSD-2-Clause
|
|
*/
|
|
|
|
#include "FileDB.h"
|
|
|
|
#include <AK/Debug.h>
|
|
#include <AK/LexicalPath.h>
|
|
#include <AK/NonnullRefPtr.h>
|
|
#include <LibCore/File.h>
|
|
|
|
namespace LanguageServers {
|
|
|
|
RefPtr<const GUI::TextDocument> FileDB::get_document(DeprecatedString const& filename) const
|
|
{
|
|
auto absolute_path = to_absolute_path(filename);
|
|
auto document_optional = m_open_files.get(absolute_path);
|
|
if (!document_optional.has_value())
|
|
return nullptr;
|
|
|
|
return *document_optional.value();
|
|
}
|
|
|
|
RefPtr<GUI::TextDocument> FileDB::get_document(DeprecatedString const& filename)
|
|
{
|
|
auto document = reinterpret_cast<FileDB const*>(this)->get_document(filename);
|
|
if (document.is_null())
|
|
return nullptr;
|
|
return adopt_ref(*const_cast<GUI::TextDocument*>(document.leak_ref()));
|
|
}
|
|
|
|
Optional<DeprecatedString> FileDB::get_or_read_from_filesystem(StringView filename) const
|
|
{
|
|
auto absolute_path = to_absolute_path(filename);
|
|
auto document = get_document(absolute_path);
|
|
if (document)
|
|
return document->text();
|
|
|
|
auto document_or_error = create_from_filesystem(absolute_path);
|
|
if (document_or_error.is_error()) {
|
|
dbgln("Failed to create document '{}': {}", absolute_path, document_or_error.error());
|
|
return {};
|
|
}
|
|
return document_or_error.value()->text();
|
|
}
|
|
|
|
bool FileDB::is_open(DeprecatedString const& filename) const
|
|
{
|
|
return m_open_files.contains(to_absolute_path(filename));
|
|
}
|
|
|
|
bool FileDB::add(DeprecatedString const& filename, int fd)
|
|
{
|
|
auto document_or_error = create_from_fd(fd);
|
|
if (document_or_error.is_error()) {
|
|
dbgln("Failed to create document: {}", document_or_error.error());
|
|
return false;
|
|
}
|
|
|
|
m_open_files.set(to_absolute_path(filename), document_or_error.release_value());
|
|
return true;
|
|
}
|
|
|
|
DeprecatedString FileDB::to_absolute_path(DeprecatedString const& filename) const
|
|
{
|
|
if (LexicalPath { filename }.is_absolute()) {
|
|
return filename;
|
|
}
|
|
if (m_project_root.is_null())
|
|
return filename;
|
|
return LexicalPath { DeprecatedString::formatted("{}/{}", m_project_root, filename) }.string();
|
|
}
|
|
|
|
ErrorOr<NonnullRefPtr<GUI::TextDocument>> FileDB::create_from_filesystem(DeprecatedString const& filename) const
|
|
{
|
|
auto file = TRY(Core::File::open(to_absolute_path(filename), Core::File::OpenMode::Read));
|
|
return create_from_file(move(file));
|
|
}
|
|
|
|
ErrorOr<NonnullRefPtr<GUI::TextDocument>> FileDB::create_from_fd(int fd) const
|
|
{
|
|
auto file = TRY(Core::File::adopt_fd(fd, Core::File::OpenMode::Read));
|
|
return create_from_file(move(file));
|
|
}
|
|
|
|
class DefaultDocumentClient final : public GUI::TextDocument::Client {
|
|
public:
|
|
virtual ~DefaultDocumentClient() override = default;
|
|
virtual void document_did_append_line() override {};
|
|
virtual void document_did_insert_line(size_t) override {};
|
|
virtual void document_did_remove_line(size_t) override {};
|
|
virtual void document_did_remove_all_lines() override {};
|
|
virtual void document_did_change(GUI::AllowCallback) override {};
|
|
virtual void document_did_set_text(GUI::AllowCallback) override {};
|
|
virtual void document_did_set_cursor(const GUI::TextPosition&) override {};
|
|
virtual void document_did_update_undo_stack() override { }
|
|
|
|
virtual bool is_automatic_indentation_enabled() const override { return false; }
|
|
virtual int soft_tab_width() const override { return 4; }
|
|
};
|
|
static DefaultDocumentClient s_default_document_client;
|
|
|
|
ErrorOr<NonnullRefPtr<GUI::TextDocument>> FileDB::create_from_file(NonnullOwnPtr<Core::File> file) const
|
|
{
|
|
auto content = TRY(file->read_until_eof());
|
|
auto document = GUI::TextDocument::create(&s_default_document_client);
|
|
document->set_text(content);
|
|
return document;
|
|
}
|
|
|
|
void FileDB::on_file_edit_insert_text(DeprecatedString const& filename, DeprecatedString const& inserted_text, size_t start_line, size_t start_column)
|
|
{
|
|
VERIFY(is_open(filename));
|
|
auto document = get_document(filename);
|
|
VERIFY(document);
|
|
GUI::TextPosition start_position { start_line, start_column };
|
|
document->insert_at(start_position, inserted_text, &s_default_document_client);
|
|
|
|
dbgln_if(FILE_CONTENT_DEBUG, "{}", document->text());
|
|
}
|
|
|
|
void FileDB::on_file_edit_remove_text(DeprecatedString const& filename, size_t start_line, size_t start_column, size_t end_line, size_t end_column)
|
|
{
|
|
// TODO: If file is not open - need to get its contents
|
|
// Otherwise- somehow verify that respawned language server is synced with all file contents
|
|
VERIFY(is_open(filename));
|
|
auto document = get_document(filename);
|
|
VERIFY(document);
|
|
GUI::TextPosition start_position { start_line, start_column };
|
|
GUI::TextRange range {
|
|
GUI::TextPosition { start_line, start_column },
|
|
GUI::TextPosition { end_line, end_column }
|
|
};
|
|
|
|
document->remove(range);
|
|
dbgln_if(FILE_CONTENT_DEBUG, "{}", document->text());
|
|
}
|
|
|
|
RefPtr<GUI::TextDocument> FileDB::create_with_content(DeprecatedString const& content)
|
|
{
|
|
StringView content_view(content);
|
|
auto document = GUI::TextDocument::create(&s_default_document_client);
|
|
document->set_text(content_view);
|
|
return document;
|
|
}
|
|
|
|
bool FileDB::add(DeprecatedString const& filename, DeprecatedString const& content)
|
|
{
|
|
auto document = create_with_content(content);
|
|
if (!document) {
|
|
VERIFY_NOT_REACHED();
|
|
return false;
|
|
}
|
|
|
|
m_open_files.set(to_absolute_path(filename), document.release_nonnull());
|
|
return true;
|
|
}
|
|
|
|
}
|