Prechádzať zdrojové kódy

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. :^)
Andreas Kling 6 rokov pred
rodič
commit
ae3ec3fc37

+ 12 - 41
Applications/TextEditor/main.cpp

@@ -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(redo_action.copy_ref());
+    edit_menu->add_action(text_editor->undo_action());
+    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(copy_action.copy_ref());
-    edit_menu->add_action(paste_action.copy_ref());
-    edit_menu->add_action(delete_action.copy_ref());
+    edit_menu->add_action(text_editor->cut_action());
+    edit_menu->add_action(text_editor->copy_action());
+    edit_menu->add_action(text_editor->paste_action());
+    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(copy_action.copy_ref());
-    toolbar->add_action(move(paste_action));
-    toolbar->add_action(delete_action.copy_ref());
+    toolbar->add_action(text_editor->cut_action());
+    toolbar->add_action(text_editor->copy_action());
+    toolbar->add_action(text_editor->paste_action());
+    toolbar->add_action(text_editor->delete_action());
 
 
     toolbar->add_separator();
     toolbar->add_separator();
 
 
-    toolbar->add_action(move(undo_action));
-    toolbar->add_action(move(redo_action));
-
-    text_editor->on_selection_change = [&] {
-        cut_action->set_enabled(text_editor->has_selection());
-        copy_action->set_enabled(text_editor->has_selection());
-    };
+    toolbar->add_action(text_editor->undo_action());
+    toolbar->add_action(text_editor->redo_action());
 
 
     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()));

+ 1 - 1
LibGUI/GMenu.cpp

@@ -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)));
 }
 }

+ 4 - 1
LibGUI/GMenu.h

@@ -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();

+ 0 - 1
LibGUI/GMenuBar.h

@@ -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;

+ 48 - 1
LibGUI/GTextEditor.cpp

@@ -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());
+}

+ 18 - 0
LibGUI/GTextEditor.h

@@ -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;
 };
 };