mirror of
https://github.com/SerenityOS/serenity.git
synced 2025-01-23 09:51:57 -05:00
LibGUI: Give GTextEditor a context menu.
Now GTextEditor manages its own editing actions (cut/copy/paste/etc) and will show a context menu containing them when requested. Apps that want to put a GTextEditor's actions in its menu can get to the actions via public getters. :^)
This commit is contained in:
parent
67d7fc94fc
commit
ae3ec3fc37
6 changed files with 83 additions and 45 deletions
|
@ -57,30 +57,6 @@ int main(int argc, char** argv)
|
||||||
text_editor->write_to_file(path);
|
text_editor->write_to_file(path);
|
||||||
});
|
});
|
||||||
|
|
||||||
auto undo_action = GAction::create("Undo", { Mod_Ctrl, Key_Z }, GraphicsBitmap::load_from_file("/res/icons/16x16/undo.png"), [&] (const GAction&) {
|
|
||||||
// FIXME: Undo
|
|
||||||
});
|
|
||||||
|
|
||||||
auto redo_action = GAction::create("Redo", { Mod_Ctrl, Key_Y }, GraphicsBitmap::load_from_file("/res/icons/16x16/redo.png"), [&] (const GAction&) {
|
|
||||||
// FIXME: Redo
|
|
||||||
});
|
|
||||||
|
|
||||||
auto cut_action = GAction::create("Cut", { Mod_Ctrl, Key_X }, GraphicsBitmap::load_from_file("/res/icons/cut16.png"), [&] (const GAction&) {
|
|
||||||
text_editor->cut();
|
|
||||||
});
|
|
||||||
|
|
||||||
auto copy_action = GAction::create("Copy", { Mod_Ctrl, Key_C }, GraphicsBitmap::load_from_file("/res/icons/16x16/edit-copy.png"), [&] (const GAction&) {
|
|
||||||
text_editor->copy();
|
|
||||||
});
|
|
||||||
|
|
||||||
auto paste_action = GAction::create("Paste", { Mod_Ctrl, Key_V }, GraphicsBitmap::load_from_file("/res/icons/paste16.png"), [&] (const GAction&) {
|
|
||||||
text_editor->paste();
|
|
||||||
});
|
|
||||||
|
|
||||||
auto delete_action = GAction::create("Delete", { 0, Key_Delete }, GraphicsBitmap::load_from_file("/res/icons/16x16/delete.png"), [&] (const GAction&) {
|
|
||||||
text_editor->do_delete();
|
|
||||||
});
|
|
||||||
|
|
||||||
auto menubar = make<GMenuBar>();
|
auto menubar = make<GMenuBar>();
|
||||||
auto app_menu = make<GMenu>("TextEditor");
|
auto app_menu = make<GMenu>("TextEditor");
|
||||||
app_menu->add_action(GAction::create("Quit", { Mod_Alt, Key_F4 }, [] (const GAction&) {
|
app_menu->add_action(GAction::create("Quit", { Mod_Alt, Key_F4 }, [] (const GAction&) {
|
||||||
|
@ -96,13 +72,13 @@ int main(int argc, char** argv)
|
||||||
menubar->add_menu(move(file_menu));
|
menubar->add_menu(move(file_menu));
|
||||||
|
|
||||||
auto edit_menu = make<GMenu>("Edit");
|
auto edit_menu = make<GMenu>("Edit");
|
||||||
edit_menu->add_action(undo_action.copy_ref());
|
edit_menu->add_action(text_editor->undo_action());
|
||||||
edit_menu->add_action(redo_action.copy_ref());
|
edit_menu->add_action(text_editor->redo_action());
|
||||||
edit_menu->add_separator();
|
edit_menu->add_separator();
|
||||||
edit_menu->add_action(cut_action.copy_ref());
|
edit_menu->add_action(text_editor->cut_action());
|
||||||
edit_menu->add_action(copy_action.copy_ref());
|
edit_menu->add_action(text_editor->copy_action());
|
||||||
edit_menu->add_action(paste_action.copy_ref());
|
edit_menu->add_action(text_editor->paste_action());
|
||||||
edit_menu->add_action(delete_action.copy_ref());
|
edit_menu->add_action(text_editor->delete_action());
|
||||||
menubar->add_menu(move(edit_menu));
|
menubar->add_menu(move(edit_menu));
|
||||||
|
|
||||||
auto font_menu = make<GMenu>("Font");
|
auto font_menu = make<GMenu>("Font");
|
||||||
|
@ -128,20 +104,15 @@ int main(int argc, char** argv)
|
||||||
|
|
||||||
toolbar->add_separator();
|
toolbar->add_separator();
|
||||||
|
|
||||||
toolbar->add_action(cut_action.copy_ref());
|
toolbar->add_action(text_editor->cut_action());
|
||||||
toolbar->add_action(copy_action.copy_ref());
|
toolbar->add_action(text_editor->copy_action());
|
||||||
toolbar->add_action(move(paste_action));
|
toolbar->add_action(text_editor->paste_action());
|
||||||
toolbar->add_action(delete_action.copy_ref());
|
toolbar->add_action(text_editor->delete_action());
|
||||||
|
|
||||||
toolbar->add_separator();
|
toolbar->add_separator();
|
||||||
|
|
||||||
toolbar->add_action(move(undo_action));
|
toolbar->add_action(text_editor->undo_action());
|
||||||
toolbar->add_action(move(redo_action));
|
toolbar->add_action(text_editor->redo_action());
|
||||||
|
|
||||||
text_editor->on_selection_change = [&] {
|
|
||||||
cut_action->set_enabled(text_editor->has_selection());
|
|
||||||
copy_action->set_enabled(text_editor->has_selection());
|
|
||||||
};
|
|
||||||
|
|
||||||
auto* window = new GWindow;
|
auto* window = new GWindow;
|
||||||
window->set_title(String::format("TextEditor: %s", path.characters()));
|
window->set_title(String::format("TextEditor: %s", path.characters()));
|
||||||
|
|
|
@ -29,7 +29,7 @@ GMenu::~GMenu()
|
||||||
unrealize_menu();
|
unrealize_menu();
|
||||||
}
|
}
|
||||||
|
|
||||||
void GMenu::add_action(Retained<GAction>&& action)
|
void GMenu::add_action(Retained<GAction> action)
|
||||||
{
|
{
|
||||||
m_items.append(make<GMenuItem>(m_menu_id, move(action)));
|
m_items.append(make<GMenuItem>(m_menu_id, move(action)));
|
||||||
}
|
}
|
||||||
|
|
|
@ -2,6 +2,8 @@
|
||||||
|
|
||||||
#include <LibGUI/GMenuItem.h>
|
#include <LibGUI/GMenuItem.h>
|
||||||
#include <AK/Function.h>
|
#include <AK/Function.h>
|
||||||
|
#include <AK/Retainable.h>
|
||||||
|
#include <AK/Retained.h>
|
||||||
#include <AK/Vector.h>
|
#include <AK/Vector.h>
|
||||||
|
|
||||||
class GAction;
|
class GAction;
|
||||||
|
@ -16,7 +18,7 @@ public:
|
||||||
|
|
||||||
GAction* action_at(int);
|
GAction* action_at(int);
|
||||||
|
|
||||||
void add_action(Retained<GAction>&&);
|
void add_action(Retained<GAction>);
|
||||||
void add_separator();
|
void add_separator();
|
||||||
|
|
||||||
void popup(const Point& screen_position);
|
void popup(const Point& screen_position);
|
||||||
|
@ -26,6 +28,7 @@ public:
|
||||||
|
|
||||||
private:
|
private:
|
||||||
friend class GMenuBar;
|
friend class GMenuBar;
|
||||||
|
|
||||||
int menu_id() const { return m_menu_id; }
|
int menu_id() const { return m_menu_id; }
|
||||||
int realize_menu();
|
int realize_menu();
|
||||||
void unrealize_menu();
|
void unrealize_menu();
|
||||||
|
|
|
@ -2,7 +2,6 @@
|
||||||
|
|
||||||
#include <LibGUI/GMenu.h>
|
#include <LibGUI/GMenu.h>
|
||||||
#include <AK/Badge.h>
|
#include <AK/Badge.h>
|
||||||
#include <AK/OwnPtr.h>
|
|
||||||
#include <AK/Vector.h>
|
#include <AK/Vector.h>
|
||||||
|
|
||||||
class GApplication;
|
class GApplication;
|
||||||
|
|
|
@ -1,9 +1,11 @@
|
||||||
#include <LibGUI/GTextEditor.h>
|
#include <LibGUI/GTextEditor.h>
|
||||||
#include <LibGUI/GScrollBar.h>
|
#include <LibGUI/GScrollBar.h>
|
||||||
#include <LibGUI/GFontDatabase.h>
|
#include <LibGUI/GFontDatabase.h>
|
||||||
#include <LibGUI/GClipboard.h>
|
#include <LibGUI/GClipboard.h>
|
||||||
#include <LibGUI/GPainter.h>
|
#include <LibGUI/GPainter.h>
|
||||||
#include <LibGUI/GWindow.h>
|
#include <LibGUI/GWindow.h>
|
||||||
|
#include <LibGUI/GMenu.h>
|
||||||
|
#include <LibGUI/GAction.h>
|
||||||
#include <Kernel/KeyCode.h>
|
#include <Kernel/KeyCode.h>
|
||||||
#include <AK/StringBuilder.h>
|
#include <AK/StringBuilder.h>
|
||||||
#include <unistd.h>
|
#include <unistd.h>
|
||||||
|
@ -22,12 +24,40 @@ GTextEditor::GTextEditor(Type type, GWidget* parent)
|
||||||
set_font(GFontDatabase::the().get_by_name("Csilla Thin"));
|
set_font(GFontDatabase::the().get_by_name("Csilla Thin"));
|
||||||
m_lines.append(make<Line>());
|
m_lines.append(make<Line>());
|
||||||
m_cursor = { 0, 0 };
|
m_cursor = { 0, 0 };
|
||||||
|
create_actions();
|
||||||
}
|
}
|
||||||
|
|
||||||
GTextEditor::~GTextEditor()
|
GTextEditor::~GTextEditor()
|
||||||
{
|
{
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void GTextEditor::create_actions()
|
||||||
|
{
|
||||||
|
m_undo_action = GAction::create("Undo", { Mod_Ctrl, Key_Z }, GraphicsBitmap::load_from_file("/res/icons/16x16/undo.png"), [&] (const GAction&) {
|
||||||
|
// FIXME: Undo
|
||||||
|
});
|
||||||
|
|
||||||
|
m_redo_action = GAction::create("Redo", { Mod_Ctrl, Key_Y }, GraphicsBitmap::load_from_file("/res/icons/16x16/redo.png"), [&] (const GAction&) {
|
||||||
|
// FIXME: Redo
|
||||||
|
});
|
||||||
|
|
||||||
|
m_cut_action = GAction::create("Cut", { Mod_Ctrl, Key_X }, GraphicsBitmap::load_from_file("/res/icons/cut16.png"), [&] (const GAction&) {
|
||||||
|
cut();
|
||||||
|
});
|
||||||
|
|
||||||
|
m_copy_action = GAction::create("Copy", { Mod_Ctrl, Key_C }, GraphicsBitmap::load_from_file("/res/icons/16x16/edit-copy.png"), [&] (const GAction&) {
|
||||||
|
copy();
|
||||||
|
});
|
||||||
|
|
||||||
|
m_paste_action = GAction::create("Paste", { Mod_Ctrl, Key_V }, GraphicsBitmap::load_from_file("/res/icons/paste16.png"), [&] (const GAction&) {
|
||||||
|
paste();
|
||||||
|
});
|
||||||
|
|
||||||
|
m_delete_action = GAction::create("Delete", { 0, Key_Delete }, GraphicsBitmap::load_from_file("/res/icons/16x16/delete.png"), [&] (const GAction&) {
|
||||||
|
do_delete();
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
void GTextEditor::set_text(const String& text)
|
void GTextEditor::set_text(const String& text)
|
||||||
{
|
{
|
||||||
if (is_single_line() && text.length() == m_lines[0]->length() && !memcmp(text.characters(), m_lines[0]->characters(), text.length()))
|
if (is_single_line() && text.length() == m_lines[0]->length() && !memcmp(text.characters(), m_lines[0]->characters(), text.length()))
|
||||||
|
@ -865,6 +895,23 @@ void GTextEditor::did_change()
|
||||||
|
|
||||||
void GTextEditor::did_update_selection()
|
void GTextEditor::did_update_selection()
|
||||||
{
|
{
|
||||||
|
m_cut_action->set_enabled(has_selection());
|
||||||
|
m_copy_action->set_enabled(has_selection());
|
||||||
if (on_selection_change)
|
if (on_selection_change)
|
||||||
on_selection_change();
|
on_selection_change();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void GTextEditor::context_menu_event(GContextMenuEvent& event)
|
||||||
|
{
|
||||||
|
if (!m_context_menu) {
|
||||||
|
m_context_menu = make<GMenu>("GTextEditor context menu");
|
||||||
|
m_context_menu->add_action(undo_action());
|
||||||
|
m_context_menu->add_action(redo_action());
|
||||||
|
m_context_menu->add_separator();
|
||||||
|
m_context_menu->add_action(cut_action());
|
||||||
|
m_context_menu->add_action(copy_action());
|
||||||
|
m_context_menu->add_action(paste_action());
|
||||||
|
m_context_menu->add_action(delete_action());
|
||||||
|
}
|
||||||
|
m_context_menu->popup(event.screen_position());
|
||||||
|
}
|
||||||
|
|
|
@ -4,6 +4,8 @@
|
||||||
#include <AK/Function.h>
|
#include <AK/Function.h>
|
||||||
#include <AK/HashMap.h>
|
#include <AK/HashMap.h>
|
||||||
|
|
||||||
|
class GAction;
|
||||||
|
class GMenu;
|
||||||
class GScrollBar;
|
class GScrollBar;
|
||||||
class Painter;
|
class Painter;
|
||||||
|
|
||||||
|
@ -106,6 +108,13 @@ public:
|
||||||
|
|
||||||
virtual const char* class_name() const override { return "GTextEditor"; }
|
virtual const char* class_name() const override { return "GTextEditor"; }
|
||||||
|
|
||||||
|
GAction& undo_action() { return *m_undo_action; }
|
||||||
|
GAction& redo_action() { return *m_redo_action; }
|
||||||
|
GAction& cut_action() { return *m_cut_action; }
|
||||||
|
GAction& copy_action() { return *m_copy_action; }
|
||||||
|
GAction& paste_action() { return *m_paste_action; }
|
||||||
|
GAction& delete_action() { return *m_delete_action; }
|
||||||
|
|
||||||
private:
|
private:
|
||||||
virtual void paint_event(GPaintEvent&) override;
|
virtual void paint_event(GPaintEvent&) override;
|
||||||
virtual void mousedown_event(GMouseEvent&) override;
|
virtual void mousedown_event(GMouseEvent&) override;
|
||||||
|
@ -118,7 +127,9 @@ private:
|
||||||
virtual bool accepts_focus() const override { return true; }
|
virtual bool accepts_focus() const override { return true; }
|
||||||
virtual void enter_event(CEvent&) override;
|
virtual void enter_event(CEvent&) override;
|
||||||
virtual void leave_event(CEvent&) override;
|
virtual void leave_event(CEvent&) override;
|
||||||
|
virtual void context_menu_event(GContextMenuEvent&) override;
|
||||||
|
|
||||||
|
void create_actions();
|
||||||
void paint_ruler(Painter&);
|
void paint_ruler(Painter&);
|
||||||
void update_content_size();
|
void update_content_size();
|
||||||
void did_change();
|
void did_change();
|
||||||
|
@ -175,4 +186,11 @@ private:
|
||||||
int m_soft_tab_width { 4 };
|
int m_soft_tab_width { 4 };
|
||||||
int m_horizontal_content_padding { 2 };
|
int m_horizontal_content_padding { 2 };
|
||||||
GTextRange m_selection;
|
GTextRange m_selection;
|
||||||
|
OwnPtr<GMenu> m_context_menu;
|
||||||
|
RetainPtr<GAction> m_undo_action;
|
||||||
|
RetainPtr<GAction> m_redo_action;
|
||||||
|
RetainPtr<GAction> m_cut_action;
|
||||||
|
RetainPtr<GAction> m_copy_action;
|
||||||
|
RetainPtr<GAction> m_paste_action;
|
||||||
|
RetainPtr<GAction> m_delete_action;
|
||||||
};
|
};
|
||||||
|
|
Loading…
Add table
Reference in a new issue