mirror of
https://github.com/RGBCube/serenity
synced 2025-05-25 22:35:07 +00:00
HackStudio: Start adding a "find in files" function
Projects now contain a set of TextDocument objects. Each TextDocument represents a member file in the project. TextDocuments may not have their file contents loaded at all times, but they will be loaded on demand when calling TextDocument::contents(). "Find in files" works by iterating over the documents in the project and calling find(needle) on each one. The return value from find() is a vector of line numbers where the needle was found. This is obviously going to need a bunch more work. :^)
This commit is contained in:
parent
41289e652f
commit
d3e81d2ba8
6 changed files with 209 additions and 38 deletions
|
@ -2,6 +2,7 @@ include ../../Makefile.common
|
|||
|
||||
OBJS = \
|
||||
Project.o \
|
||||
TextDocument.o \
|
||||
TerminalWrapper.o \
|
||||
main.o
|
||||
|
||||
|
|
|
@ -14,11 +14,11 @@ public:
|
|||
{
|
||||
int row = index.row();
|
||||
if (role == Role::Display) {
|
||||
return m_project.m_files.at(row);
|
||||
return m_project.m_files.at(row).name();
|
||||
}
|
||||
if (role == Role::Font) {
|
||||
extern String g_currently_open_file;
|
||||
if (m_project.m_files.at(row) == g_currently_open_file)
|
||||
if (m_project.m_files.at(row).name() == g_currently_open_file)
|
||||
return Font::default_bold_font();
|
||||
return {};
|
||||
}
|
||||
|
@ -30,9 +30,11 @@ private:
|
|||
Project& m_project;
|
||||
};
|
||||
|
||||
Project::Project(Vector<String>&& files)
|
||||
: m_files(move(files))
|
||||
Project::Project(Vector<String>&& filenames)
|
||||
{
|
||||
for (auto& filename : filenames) {
|
||||
m_files.append(TextDocument::construct_with_name(filename));
|
||||
}
|
||||
m_model = adopt(*new ProjectModel(*this));
|
||||
}
|
||||
|
||||
|
|
|
@ -1,6 +1,8 @@
|
|||
#pragma once
|
||||
|
||||
#include "TextDocument.h"
|
||||
#include <AK/Noncopyable.h>
|
||||
#include <AK/NonnullRefPtrVector.h>
|
||||
#include <AK/OwnPtr.h>
|
||||
#include <LibGUI/GModel.h>
|
||||
|
||||
|
@ -12,10 +14,18 @@ public:
|
|||
|
||||
GModel& model() { return *m_model; }
|
||||
|
||||
template<typename Callback>
|
||||
void for_each_text_file(Callback callback) const
|
||||
{
|
||||
for (auto& file : m_files) {
|
||||
callback(file);
|
||||
}
|
||||
}
|
||||
|
||||
private:
|
||||
friend class ProjectModel;
|
||||
explicit Project(Vector<String>&& files);
|
||||
|
||||
RefPtr<GModel> m_model;
|
||||
Vector<String> m_files;
|
||||
NonnullRefPtrVector<TextDocument> m_files;
|
||||
};
|
||||
|
|
41
DevTools/HackStudio/TextDocument.cpp
Normal file
41
DevTools/HackStudio/TextDocument.cpp
Normal file
|
@ -0,0 +1,41 @@
|
|||
#include "TextDocument.h"
|
||||
#include <LibCore/CFile.h>
|
||||
#include <string.h>
|
||||
|
||||
const ByteBuffer& TextDocument::contents() const
|
||||
{
|
||||
if (m_contents.is_null()) {
|
||||
auto file = CFile::construct(m_name);
|
||||
if (file->open(CFile::ReadOnly))
|
||||
m_contents = file->read_all();
|
||||
}
|
||||
return m_contents;
|
||||
}
|
||||
|
||||
Vector<int> TextDocument::find(const StringView& needle) const
|
||||
{
|
||||
// NOTE: This forces us to load the contents if we hadn't already.
|
||||
contents();
|
||||
|
||||
Vector<int> matching_line_numbers;
|
||||
|
||||
String needle_as_string(needle);
|
||||
|
||||
int line_index = 0;
|
||||
int start_of_line = 0;
|
||||
for (int i = 0; i < m_contents.size(); ++i) {
|
||||
char ch = m_contents[i];
|
||||
if (ch == '\n') {
|
||||
// FIXME: Please come back here and do this the good boy way.
|
||||
String line(StringView(m_contents.data() + start_of_line, i - start_of_line));
|
||||
auto* found = strstr(line.characters(), needle_as_string.characters());
|
||||
if (found)
|
||||
matching_line_numbers.append(line_index + 1);
|
||||
++line_index;
|
||||
start_of_line = i + 1;
|
||||
continue;
|
||||
}
|
||||
}
|
||||
|
||||
return matching_line_numbers;
|
||||
}
|
29
DevTools/HackStudio/TextDocument.h
Normal file
29
DevTools/HackStudio/TextDocument.h
Normal file
|
@ -0,0 +1,29 @@
|
|||
#pragma once
|
||||
|
||||
#include <AK/ByteBuffer.h>
|
||||
#include <AK/NonnullRefPtr.h>
|
||||
#include <AK/RefCounted.h>
|
||||
#include <AK/String.h>
|
||||
|
||||
class TextDocument : public RefCounted<TextDocument> {
|
||||
public:
|
||||
static NonnullRefPtr<TextDocument> construct_with_name(const String& name)
|
||||
{
|
||||
return adopt(*new TextDocument(name));
|
||||
}
|
||||
|
||||
const String& name() const { return m_name; }
|
||||
|
||||
const ByteBuffer& contents() const;
|
||||
|
||||
Vector<int> find(const StringView&) const;
|
||||
|
||||
private:
|
||||
explicit TextDocument(const String& name)
|
||||
: m_name(name)
|
||||
{
|
||||
}
|
||||
|
||||
String m_name;
|
||||
mutable ByteBuffer m_contents;
|
||||
};
|
|
@ -4,6 +4,7 @@
|
|||
#include <LibGUI/GAboutDialog.h>
|
||||
#include <LibGUI/GAction.h>
|
||||
#include <LibGUI/GApplication.h>
|
||||
#include <LibGUI/GButton.h>
|
||||
#include <LibGUI/GBoxLayout.h>
|
||||
#include <LibGUI/GInputBox.h>
|
||||
#include <LibGUI/GListView.h>
|
||||
|
@ -13,6 +14,7 @@
|
|||
#include <LibGUI/GSplitter.h>
|
||||
#include <LibGUI/GStatusBar.h>
|
||||
#include <LibGUI/GTabWidget.h>
|
||||
#include <LibGUI/GTextBox.h>
|
||||
#include <LibGUI/GTextEditor.h>
|
||||
#include <LibGUI/GToolBar.h>
|
||||
#include <LibGUI/GWidget.h>
|
||||
|
@ -21,20 +23,26 @@
|
|||
#include <unistd.h>
|
||||
|
||||
String g_currently_open_file;
|
||||
OwnPtr<Project> g_project;
|
||||
RefPtr<GWindow> g_window;
|
||||
RefPtr<GListView> g_project_list_view;
|
||||
RefPtr<GTextEditor> g_text_editor;
|
||||
|
||||
static void build(TerminalWrapper&);
|
||||
static void run(TerminalWrapper&);
|
||||
static NonnullRefPtr<GWidget> build_find_in_files_widget();
|
||||
static void open_file(const String&);
|
||||
|
||||
int main(int argc, char** argv)
|
||||
{
|
||||
GApplication app(argc, argv);
|
||||
|
||||
auto window = GWindow::construct();
|
||||
window->set_rect(100, 100, 800, 600);
|
||||
window->set_title("HackStudio");
|
||||
g_window = GWindow::construct();
|
||||
g_window->set_rect(100, 100, 800, 600);
|
||||
g_window->set_title("HackStudio");
|
||||
|
||||
auto widget = GWidget::construct();
|
||||
window->set_main_widget(widget);
|
||||
g_window->set_main_widget(widget);
|
||||
|
||||
widget->set_fill_with_background_color(true);
|
||||
widget->set_layout(make<GBoxLayout>(Orientation::Vertical));
|
||||
|
@ -44,65 +52,59 @@ int main(int argc, char** argv)
|
|||
perror("chdir");
|
||||
return 1;
|
||||
}
|
||||
auto project = Project::load_from_file("little.files");
|
||||
ASSERT(project);
|
||||
g_project = Project::load_from_file("little.files");
|
||||
ASSERT(g_project);
|
||||
|
||||
auto toolbar = GToolBar::construct(widget);
|
||||
|
||||
auto outer_splitter = GSplitter::construct(Orientation::Horizontal, widget);
|
||||
auto project_list_view = GListView::construct(outer_splitter);
|
||||
project_list_view->set_model(project->model());
|
||||
project_list_view->set_size_policy(SizePolicy::Fixed, SizePolicy::Fill);
|
||||
project_list_view->set_preferred_size(200, 0);
|
||||
g_project_list_view = GListView::construct(outer_splitter);
|
||||
g_project_list_view->set_model(g_project->model());
|
||||
g_project_list_view->set_size_policy(SizePolicy::Fixed, SizePolicy::Fill);
|
||||
g_project_list_view->set_preferred_size(200, 0);
|
||||
|
||||
auto inner_splitter = GSplitter::construct(Orientation::Vertical, outer_splitter);
|
||||
auto text_editor = GTextEditor::construct(GTextEditor::MultiLine, inner_splitter);
|
||||
text_editor->set_ruler_visible(true);
|
||||
g_text_editor = GTextEditor::construct(GTextEditor::MultiLine, inner_splitter);
|
||||
g_text_editor->set_ruler_visible(true);
|
||||
|
||||
project_list_view->on_activation = [&](auto& index) {
|
||||
auto filename = project_list_view->model()->data(index).to_string();
|
||||
auto file = CFile::construct(filename);
|
||||
if (!file->open(CFile::ReadOnly)) {
|
||||
GMessageBox::show("Could not open!", "Error", GMessageBox::Type::Error, GMessageBox::InputType::OK, window);
|
||||
return;
|
||||
}
|
||||
text_editor->set_text(file->read_all());
|
||||
g_currently_open_file = filename;
|
||||
window->set_title(String::format("%s - HackStudio", g_currently_open_file.characters()));
|
||||
project_list_view->update();
|
||||
g_project_list_view->on_activation = [&](auto& index) {
|
||||
auto filename = g_project_list_view->model()->data(index).to_string();
|
||||
open_file(filename);
|
||||
};
|
||||
|
||||
auto tab_widget = GTabWidget::construct(inner_splitter);
|
||||
|
||||
tab_widget->add_widget("Find in files", build_find_in_files_widget());
|
||||
|
||||
auto terminal_wrapper = TerminalWrapper::construct(nullptr);
|
||||
tab_widget->add_widget("Console", terminal_wrapper);
|
||||
|
||||
auto statusbar = GStatusBar::construct(widget);
|
||||
|
||||
text_editor->on_cursor_change = [&] {
|
||||
statusbar->set_text(String::format("Line: %d, Column: %d", text_editor->cursor().line(), text_editor->cursor().column()));
|
||||
g_text_editor->on_cursor_change = [&] {
|
||||
statusbar->set_text(String::format("Line: %d, Column: %d", g_text_editor->cursor().line(), g_text_editor->cursor().column()));
|
||||
};
|
||||
|
||||
text_editor->add_custom_context_menu_action(GAction::create(
|
||||
g_text_editor->add_custom_context_menu_action(GAction::create(
|
||||
"Go to line...", { Mod_Ctrl, Key_L }, GraphicsBitmap::load_from_file("/res/icons/16x16/go-forward.png"), [&](auto&) {
|
||||
auto input_box = GInputBox::construct("Line:", "Go to line", window);
|
||||
auto input_box = GInputBox::construct("Line:", "Go to line", g_window);
|
||||
auto result = input_box->exec();
|
||||
if (result == GInputBox::ExecOK) {
|
||||
bool ok;
|
||||
auto line_number = input_box->text_value().to_uint(ok);
|
||||
if (ok) {
|
||||
text_editor->set_cursor(line_number - 1, 0);
|
||||
g_text_editor->set_cursor(line_number - 1, 0);
|
||||
}
|
||||
}
|
||||
},
|
||||
text_editor));
|
||||
g_text_editor));
|
||||
|
||||
auto menubar = make<GMenuBar>();
|
||||
auto app_menu = make<GMenu>("HackStudio");
|
||||
app_menu->add_action(GAction::create("Save", { Mod_Ctrl, Key_S }, GraphicsBitmap::load_from_file("/res/icons/16x16/save.png"), [&](auto&) {
|
||||
if (g_currently_open_file.is_empty())
|
||||
return;
|
||||
text_editor->write_to_file(g_currently_open_file);
|
||||
g_text_editor->write_to_file(g_currently_open_file);
|
||||
}));
|
||||
app_menu->add_action(GCommonActions::make_quit_action([&](auto&) {
|
||||
app.quit();
|
||||
|
@ -122,15 +124,15 @@ int main(int argc, char** argv)
|
|||
|
||||
auto help_menu = make<GMenu>("Help");
|
||||
help_menu->add_action(GAction::create("About", [&](auto&) {
|
||||
GAboutDialog::show("HackStudio", small_icon, window);
|
||||
GAboutDialog::show("HackStudio", small_icon, g_window);
|
||||
}));
|
||||
menubar->add_menu(move(help_menu));
|
||||
|
||||
app.set_menubar(move(menubar));
|
||||
|
||||
window->set_icon(small_icon);
|
||||
g_window->set_icon(small_icon);
|
||||
|
||||
window->show();
|
||||
g_window->show();
|
||||
return app.exec();
|
||||
}
|
||||
|
||||
|
@ -143,3 +145,89 @@ void run(TerminalWrapper& wrapper)
|
|||
{
|
||||
wrapper.run_command("make run");
|
||||
}
|
||||
|
||||
struct FilenameAndLineNumber {
|
||||
String filename;
|
||||
int line_number { -1 };
|
||||
};
|
||||
|
||||
class SearchResultsModel final : public GModel {
|
||||
public:
|
||||
explicit SearchResultsModel(const Vector<FilenameAndLineNumber>&& matches)
|
||||
: m_matches(move(matches))
|
||||
{
|
||||
}
|
||||
|
||||
virtual int row_count(const GModelIndex& = GModelIndex()) const override { return m_matches.size(); }
|
||||
virtual int column_count(const GModelIndex& = GModelIndex()) const override { return 1; }
|
||||
virtual GVariant data(const GModelIndex& index, Role role = Role::Display) const override
|
||||
{
|
||||
if (role == Role::Display) {
|
||||
auto& match = m_matches.at(index.row());
|
||||
return String::format("%s:%d", match.filename.characters(), match.line_number);
|
||||
}
|
||||
return {};
|
||||
}
|
||||
virtual void update() override {}
|
||||
|
||||
private:
|
||||
Vector<FilenameAndLineNumber> m_matches;
|
||||
};
|
||||
|
||||
static RefPtr<SearchResultsModel> find_in_files(const StringView& text)
|
||||
{
|
||||
Vector<FilenameAndLineNumber> matches;
|
||||
g_project->for_each_text_file([&](auto& file) {
|
||||
auto matches_in_file = file.find(text);
|
||||
for (int match : matches_in_file) {
|
||||
matches.append({ file.name(), match });
|
||||
}
|
||||
});
|
||||
|
||||
return adopt(*new SearchResultsModel(move(matches)));
|
||||
}
|
||||
|
||||
NonnullRefPtr<GWidget> build_find_in_files_widget()
|
||||
{
|
||||
auto widget = GWidget::construct();
|
||||
widget->set_layout(make<GBoxLayout>(Orientation::Vertical));
|
||||
auto textbox = GTextBox::construct(widget);
|
||||
textbox->set_size_policy(SizePolicy::Fill, SizePolicy::Fixed);
|
||||
textbox->set_preferred_size(0, 20);
|
||||
auto button = GButton::construct("Find in files", widget);
|
||||
button->set_size_policy(SizePolicy::Fill, SizePolicy::Fixed);
|
||||
button->set_preferred_size(0, 20);
|
||||
|
||||
auto result_view = GListView::construct(widget);
|
||||
|
||||
result_view->on_activation = [result_view](auto& index) {
|
||||
auto match_string = result_view->model()->data(index).to_string();
|
||||
auto parts = match_string.split(':');
|
||||
ASSERT(parts.size() == 2);
|
||||
bool ok;
|
||||
int line_number = parts[1].to_int(ok);
|
||||
ASSERT(ok);
|
||||
open_file(parts[0]);
|
||||
g_text_editor->set_cursor(line_number - 1, 0);
|
||||
g_text_editor->set_focus(true);
|
||||
};
|
||||
|
||||
button->on_click = [textbox, result_view = result_view.ptr()](auto&) {
|
||||
auto results_model = find_in_files(textbox->text());
|
||||
result_view->set_model(results_model);
|
||||
};
|
||||
return widget;
|
||||
}
|
||||
|
||||
void open_file(const String& filename)
|
||||
{
|
||||
auto file = CFile::construct(filename);
|
||||
if (!file->open(CFile::ReadOnly)) {
|
||||
GMessageBox::show("Could not open!", "Error", GMessageBox::Type::Error, GMessageBox::InputType::OK, g_window);
|
||||
return;
|
||||
}
|
||||
g_text_editor->set_text(file->read_all());
|
||||
g_currently_open_file = filename;
|
||||
g_window->set_title(String::format("%s - HackStudio", g_currently_open_file.characters()));
|
||||
g_project_list_view->update();
|
||||
}
|
||||
|
|
Loading…
Add table
Add a link
Reference in a new issue