[epiphany/pgriffis/browser-action-abstraction] WebExtensions: Redesign Browser Actions to have a flexible presentation
- From: Patrick Griffis <pgriffis src gnome org>
- To: commits-list gnome org
- Cc:
- Subject: [epiphany/pgriffis/browser-action-abstraction] WebExtensions: Redesign Browser Actions to have a flexible presentation
- Date: Wed, 12 Oct 2022 21:36:15 +0000 (UTC)
commit 3b4a01503d330e60fa913d666994c78bd33ccadb
Author: Patrick Griffis <pgriffis igalia com>
Date: Mon Oct 10 20:01:00 2022 -0500
WebExtensions: Redesign Browser Actions to have a flexible presentation
Previously the WebExtensionManager would directly poke around inserting
widgets into the HeaderBar. This falls apart with adaptive views so
switching to mobile would lose access to extensions.
This changes the design where WebExtensionsManager exposes a list of
EphyBrowserAction objects and the UI tracks this list and decides
how to show them in any mode or in multiple windows.
As well as this change the UI was changed so now all browser actions
are shown in a popover and the popover doubles as the browser actions
popup UI when clicked. This matches Firefox's UI (but is even smoother!).
The ability to show all actions as a dedicated button in the headerbar
was lost for now. I think this is ok for now but ideally we implement
a solution similar to other browsers where the user can select which
ones to be dedicated buttons and which ones to go into the dropdown.
meson.build | 6 +-
src/ephy-action-bar-end.c | 182 ++++++++++++++++++++++--
src/ephy-action-bar-end.h | 5 +-
src/ephy-header-bar.c | 7 -
src/ephy-header-bar.h | 3 -
src/resources/epiphany.gresource.xml | 1 +
src/resources/gtk/action-bar-end.ui | 92 ++++++++++++-
src/resources/gtk/browser-action-row.ui | 32 +++++
src/resources/themes/_shared-base.scss | 4 +
src/webextension/api/commands.c | 3 +-
src/webextension/api/menus.c | 4 +-
src/webextension/ephy-browser-action-row.c | 147 ++++++++++++++++++++
src/webextension/ephy-browser-action-row.h | 37 +++++
src/webextension/ephy-browser-action.c | 150 ++++++++++++++++++++
src/webextension/ephy-browser-action.h | 39 ++++++
src/webextension/ephy-web-extension-manager.c | 191 +++++++++++---------------
src/webextension/ephy-web-extension-manager.h | 14 +-
src/webextension/meson.build | 2 +
18 files changed, 767 insertions(+), 152 deletions(-)
---
diff --git a/meson.build b/meson.build
index 0270ed4fd..30fb154c4 100644
--- a/meson.build
+++ b/meson.build
@@ -73,13 +73,13 @@ gsb_api_key = get_option('gsb_api_key')
conf.set_quoted('GSB_API_KEY', gsb_api_key)
conf.set10('ENABLE_GSB', gsb_api_key != '')
-glib_requirement = '>= 2.67.4'
+glib_requirement = '>= 2.70.0'
gtk_requirement = '>= 3.24.0'
nettle_requirement = '>= 3.4'
webkitgtk_requirement = '>= 2.37.90'
-conf.set('GLIB_VERSION_MIN_REQUIRED', 'GLIB_VERSION_2_68')
-conf.set('GLIB_VERSION_MAX_ALLOWED', 'GLIB_VERSION_2_68')
+conf.set('GLIB_VERSION_MIN_REQUIRED', 'GLIB_VERSION_2_70')
+conf.set('GLIB_VERSION_MAX_ALLOWED', 'GLIB_VERSION_2_70')
cairo_dep = dependency('cairo', version: '>= 1.2')
gcr_dep = dependency('gcr-3', version: '>= 3.5.5')
diff --git a/src/ephy-action-bar-end.c b/src/ephy-action-bar-end.c
index c6e0cd154..c93c3d78a 100644
--- a/src/ephy-action-bar-end.c
+++ b/src/ephy-action-bar-end.c
@@ -21,10 +21,13 @@
#include "ephy-action-bar-end.h"
#include "ephy-add-bookmark-popover.h"
+#include "ephy-browser-action.h"
+#include "ephy-browser-action-row.h"
#include "ephy-desktop-utils.h"
#include "ephy-downloads-popover.h"
#include "ephy-location-entry.h"
#include "ephy-shell.h"
+#include "ephy-web-extension-manager.h"
#include "ephy-window.h"
#define NEEDS_ATTENTION_ANIMATION_TIMEOUT 2000 /*ms */
@@ -42,7 +45,13 @@ struct _EphyActionBarEnd {
GtkWidget *downloads_popover;
GtkWidget *downloads_icon;
GtkWidget *downloads_progress;
- GtkWidget *browser_action_box;
+ GtkWidget *browser_actions_button;
+ GtkWidget *browser_actions_popover;
+ GtkWidget *browser_actions_scrolled_window;
+ GtkWidget *browser_actions_listbox;
+ GtkWidget *browser_actions_stack;
+ GtkWidget *browser_actions_popup_view_box;
+ GtkWidget *browser_actions_popup_view_label;
guint downloads_button_attention_timeout_id;
};
@@ -50,6 +59,8 @@ struct _EphyActionBarEnd {
G_DEFINE_TYPE (EphyActionBarEnd, ephy_action_bar_end, GTK_TYPE_BOX)
static void begin_complete_theatrics (EphyActionBarEnd *self);
+static void set_browser_actions (EphyActionBarEnd *action_bar_end,
+ GListStore *browser_actions);
static void
remove_downloads_button_attention_style (EphyActionBarEnd *self)
@@ -223,6 +234,93 @@ show_downloads_cb (EphyDownloadsManager *manager,
gtk_toggle_button_set_active (GTK_TOGGLE_BUTTON (action_bar_end->downloads_button), TRUE);
}
+static void
+remove_popup_webview (EphyActionBarEnd *action_bar_end)
+{
+ GList *children = gtk_container_get_children (GTK_CONTAINER
(action_bar_end->browser_actions_popup_view_box));
+ GtkWidget *last_child = GTK_WIDGET (g_list_last (children)->data);
+
+ if (WEBKIT_IS_WEB_VIEW (last_child))
+ gtk_widget_destroy (last_child);
+
+ g_list_free (children);
+}
+
+static void
+show_browser_action_popup (EphyActionBarEnd *action_bar_end,
+ EphyBrowserAction *action)
+{
+ EphyWebExtensionManager *manager = ephy_web_extension_manager_get_default ();
+ EphyWebExtension *web_extension = ephy_browser_action_get_web_extension (action);
+ GtkWidget *popup_view = ephy_web_extension_manager_create_browser_popup (manager, web_extension);
+
+ gtk_container_add (GTK_CONTAINER (action_bar_end->browser_actions_popup_view_box), popup_view);
+
+ gtk_label_set_text (GTK_LABEL (action_bar_end->browser_actions_popup_view_label),
+ ephy_browser_action_get_title (action));
+
+ gtk_stack_set_visible_child (GTK_STACK (action_bar_end->browser_actions_stack),
+ action_bar_end->browser_actions_popup_view_box);
+}
+
+static void
+browser_actions_popup_view_back_clicked_cb (GtkButton *button,
+ EphyActionBarEnd *action_bar_end)
+{
+ gtk_stack_set_visible_child (GTK_STACK (action_bar_end->browser_actions_stack),
+ action_bar_end->browser_actions_scrolled_window);
+ remove_popup_webview (action_bar_end);
+}
+
+static void
+show_browser_action_cb (EphyWebExtensionManager *manager,
+ EphyBrowserAction *action,
+ EphyActionBarEnd *action_bar_end)
+{
+ GtkWindow *parent_window = GTK_WINDOW (gtk_widget_get_toplevel (GTK_WIDGET (action_bar_end)));
+ GtkWindow *active_window = gtk_application_get_active_window (GTK_APPLICATION (g_application_get_default
()));
+
+ /* There may be multiple action bars that exist. We only want to show the popup in the visible bar the
active window. */
+ if (parent_window != active_window)
+ return;
+ if (!gtk_widget_is_visible (action_bar_end->browser_actions_button))
+ return;
+
+ remove_popup_webview (action_bar_end);
+ gtk_popover_popup (GTK_POPOVER (action_bar_end->browser_actions_popover));
+ show_browser_action_popup (action_bar_end, action);
+}
+
+static void
+browser_actions_row_activated_cb (GtkListBox *listbox,
+ EphyBrowserActionRow *row,
+ EphyActionBarEnd *action_bar_end)
+{
+ EphyBrowserAction *action = ephy_browser_action_row_get_browser_action (row);
+
+ /* If it was handled we are done, otherwise we have to show a popup. */
+ if (ephy_browser_action_activate (action)) {
+ gtk_popover_popdown (GTK_POPOVER (action_bar_end->browser_actions_popover));
+ return;
+ }
+
+ show_browser_action_popup (action_bar_end, action);
+}
+
+static void
+browser_action_popover_visible_changed_cb (GtkWidget *popover,
+ GParamSpec *pspec,
+ EphyActionBarEnd *action_bar_end)
+{
+ if (!gtk_widget_get_visible (popover)) {
+ GtkStack *stack = GTK_STACK (action_bar_end->browser_actions_stack);
+
+ /* Reset to default state and destroy any open webview. */
+ gtk_stack_set_visible_child (stack, action_bar_end->browser_actions_scrolled_window);
+ remove_popup_webview (action_bar_end);
+ }
+}
+
static void
ephy_action_bar_end_class_init (EphyActionBarEndClass *klass)
{
@@ -254,7 +352,30 @@ ephy_action_bar_end_class_init (EphyActionBarEndClass *klass)
downloads_progress);
gtk_widget_class_bind_template_child (widget_class,
EphyActionBarEnd,
- browser_action_box);
+ browser_actions_button);
+ gtk_widget_class_bind_template_child (widget_class,
+ EphyActionBarEnd,
+ browser_actions_popover);
+ gtk_widget_class_bind_template_child (widget_class,
+ EphyActionBarEnd,
+ browser_actions_scrolled_window);
+ gtk_widget_class_bind_template_child (widget_class,
+ EphyActionBarEnd,
+ browser_actions_listbox);
+ gtk_widget_class_bind_template_child (widget_class,
+ EphyActionBarEnd,
+ browser_actions_stack);
+ gtk_widget_class_bind_template_child (widget_class,
+ EphyActionBarEnd,
+ browser_actions_popup_view_box);
+ gtk_widget_class_bind_template_child (widget_class,
+ EphyActionBarEnd,
+ browser_actions_popup_view_label);
+
+ gtk_widget_class_bind_template_callback (widget_class,
+ browser_actions_popup_view_back_clicked_cb);
+ gtk_widget_class_bind_template_callback (widget_class,
+ browser_actions_row_activated_cb);
}
static void
@@ -263,6 +384,7 @@ ephy_action_bar_end_init (EphyActionBarEnd *action_bar_end)
GObject *object = G_OBJECT (action_bar_end);
EphyDownloadsManager *downloads_manager;
GtkWidget *popover;
+ EphyWebExtensionManager *extension_manager;
gtk_widget_init_template (GTK_WIDGET (action_bar_end));
@@ -307,6 +429,17 @@ ephy_action_bar_end_init (EphyActionBarEnd *action_bar_end)
popover = ephy_add_bookmark_popover_new ();
gtk_menu_button_set_popover (GTK_MENU_BUTTON (action_bar_end->bookmark_button), popover);
+
+ extension_manager = ephy_web_extension_manager_get_default ();
+ g_signal_connect_object (extension_manager, "show-browser-action",
+ G_CALLBACK (show_browser_action_cb),
+ object, 0);
+
+ set_browser_actions (action_bar_end, ephy_web_extension_manager_get_browser_actions (extension_manager));
+
+ g_signal_connect (action_bar_end->browser_actions_popover, "notify::visible",
+ G_CALLBACK (browser_action_popover_visible_changed_cb),
+ action_bar_end);
}
EphyActionBarEnd *
@@ -349,13 +482,6 @@ ephy_action_bar_end_get_downloads_revealer (EphyActionBarEnd *action_bar_end)
return action_bar_end->downloads_revealer;
}
-void
-ephy_action_bar_end_add_browser_action (EphyActionBarEnd *action_bar_end,
- GtkWidget *action)
-{
- gtk_container_add (GTK_CONTAINER (action_bar_end->browser_action_box), action);
-}
-
void
ephy_action_bar_end_set_show_bookmark_button (EphyActionBarEnd *action_bar_end,
gboolean show)
@@ -390,3 +516,41 @@ ephy_action_bar_end_set_bookmark_icon_state (EphyActionBarEnd *action_bar_e
g_assert_not_reached ();
}
}
+
+GtkWidget *
+create_browser_action_item_widget (EphyBrowserAction *action,
+ gpointer user_data)
+{
+ return ephy_browser_action_row_new (action);
+}
+
+static void
+browser_actions_items_changed_cb (GListModel *list,
+ guint position,
+ guint removed,
+ guint added,
+ EphyActionBarEnd *action_bar_end)
+{
+ gtk_widget_set_visible (action_bar_end->browser_actions_button, g_list_model_get_n_items (list) != 0);
+
+ /* This handles an edge-case where if an extension is disabled while its popover is open the webview
should be destroyed.
+ * However in normal usage this shouldn't happen and with the GTK4 port the extension dialog is also modal.
+ * So we just always manually close it instead of trying to track which extension popup is open. */
+ if (removed)
+ gtk_popover_popdown (GTK_POPOVER (action_bar_end->browser_actions_popover));
+}
+
+static void
+set_browser_actions (EphyActionBarEnd *action_bar_end,
+ GListStore *browser_actions)
+{
+ gtk_list_box_bind_model (GTK_LIST_BOX (action_bar_end->browser_actions_listbox),
+ G_LIST_MODEL (browser_actions),
+ (GtkListBoxCreateWidgetFunc)create_browser_action_item_widget,
+ NULL, NULL);
+
+ g_signal_connect_object (browser_actions, "items-changed", G_CALLBACK (browser_actions_items_changed_cb),
+ action_bar_end, 0);
+
+ browser_actions_items_changed_cb (G_LIST_MODEL (browser_actions), 0, 0, 0, action_bar_end);
+}
diff --git a/src/ephy-action-bar-end.h b/src/ephy-action-bar-end.h
index ffecf9596..3872aed46 100644
--- a/src/ephy-action-bar-end.h
+++ b/src/ephy-action-bar-end.h
@@ -38,13 +38,12 @@ void ephy_action_bar_end_show_downloads (EphyActionBarEn
void ephy_action_bar_end_show_bookmarks (EphyActionBarEnd *action_bar_end);
GtkWidget *ephy_action_bar_end_get_downloads_revealer (EphyActionBarEnd *action_bar_end);
-void ephy_action_bar_end_add_browser_action (EphyActionBarEnd *action_bar_end,
- GtkWidget *action);
+void ephy_action_bar_end_set_browser_actions (EphyActionBarEnd *action_bar_end,
+ GListStore *browser_actions);
void ephy_action_bar_end_set_show_bookmark_button (EphyActionBarEnd *action_bar_end,
gboolean show);
void ephy_action_bar_end_set_bookmark_icon_state (EphyActionBarEnd *action_bar_end,
EphyBookmarkIconState state);
-
G_END_DECLS
diff --git a/src/ephy-header-bar.c b/src/ephy-header-bar.c
index d6132ba8b..8a08d3fa1 100644
--- a/src/ephy-header-bar.c
+++ b/src/ephy-header-bar.c
@@ -468,10 +468,3 @@ ephy_header_bar_set_zoom_level (EphyHeaderBar *header_bar,
gtk_label_set_label (GTK_LABEL (header_bar->zoom_level_label), zoom_level);
}
-
-void
-ephy_header_bar_add_browser_action (EphyHeaderBar *header_bar,
- GtkWidget *action)
-{
- ephy_action_bar_end_add_browser_action (header_bar->action_bar_end, action);
-}
diff --git a/src/ephy-header-bar.h b/src/ephy-header-bar.h
index f066de16d..56fde0d3d 100644
--- a/src/ephy-header-bar.h
+++ b/src/ephy-header-bar.h
@@ -49,7 +49,4 @@ void ephy_header_bar_start_change_combined_stop_reload_state (Eph
void ephy_header_bar_set_zoom_level (EphyHeaderBar *header_bar,
gdouble zoom);
-void ephy_header_bar_add_browser_action (EphyHeaderBar *header_bar,
- GtkWidget *action);
-
G_END_DECLS
diff --git a/src/resources/epiphany.gresource.xml b/src/resources/epiphany.gresource.xml
index d962bccd1..d9491d6cf 100644
--- a/src/resources/epiphany.gresource.xml
+++ b/src/resources/epiphany.gresource.xml
@@ -8,6 +8,7 @@
<file preprocess="xml-stripblanks" compressed="true">gtk/bookmark-properties.ui</file>
<file preprocess="xml-stripblanks" compressed="true">gtk/bookmark-row.ui</file>
<file preprocess="xml-stripblanks" compressed="true">gtk/bookmarks-popover.ui</file>
+ <file preprocess="xml-stripblanks" compressed="true">gtk/browser-action-row.ui</file>
<file preprocess="xml-stripblanks" compressed="true">gtk/clear-data-view.ui</file>
<file preprocess="xml-stripblanks" compressed="true">gtk/data-view.ui</file>
<file preprocess="xml-stripblanks" compressed="true">gtk/encoding-dialog.ui</file>
diff --git a/src/resources/gtk/action-bar-end.ui b/src/resources/gtk/action-bar-end.ui
index fc86d7b69..a2401ef7e 100644
--- a/src/resources/gtk/action-bar-end.ui
+++ b/src/resources/gtk/action-bar-end.ui
@@ -2,12 +2,6 @@
<interface>
<template class="EphyActionBarEnd" parent="GtkBox">
<property name="spacing">6</property>
- <child>
- <object class="GtkBox" id="browser_action_box">
- <property name="visible">True</property>
- <property name="spacing">6</property>
- </object>
- </child>
<child>
<object class="GtkRevealer" id="downloads_revealer">
<property name="visible">True</property>
@@ -49,6 +43,21 @@
</child>
</object>
</child>
+ <child>
+ <object class="GtkMenuButton" id="browser_actions_button">
+ <!-- Translators: tooltip for the webextension actions button -->
+ <property name="tooltip_text" translatable="yes">View extension actions</property>
+ <property name="visible">False</property>
+ <property name="receives_default">True</property>
+ <property name="popover">browser_actions_popover</property>
+ <child>
+ <object class="GtkImage">
+ <property name="visible">True</property>
+ <property name="icon-name">application-x-addon-symbolic</property>
+ </object>
+ </child>
+ </object>
+ </child>
<child>
<object class="GtkMenuButton" id="bookmark_button">
<property name="visible">True</property>
@@ -88,4 +97,75 @@
<object class="EphyBookmarksPopover" id="bookmarks_popover">
<property name="visible">True</property>
</object>
+ <object class="GtkPopover" id="browser_actions_popover">
+ <property name="width-request">250</property>
+ <child>
+ <object class="GtkStack" id="browser_actions_stack">
+ <property name="visible">True</property>
+ <property name="transition-type">slide-left-right</property>
+ <property name="interpolate-size">True</property>
+ <property name="margin-start">6</property>
+ <property name="margin-end">6</property>
+ <property name="margin-top">6</property>
+ <property name="margin-bottom">6</property>
+ <child>
+ <object class="GtkScrolledWindow" id="browser_actions_scrolled_window">
+ <property name="visible">True</property>
+ <property name="hscrollbar-policy">never</property>
+ <property name="max-content-height">360</property>
+ <property name="propagate-natural-height">True</property>
+ <child>
+ <object class="GtkListBox" id="browser_actions_listbox">
+ <property name="visible">True</property>
+ <property name="selection-mode">none</property>
+ <signal name="row-activated" handler="browser_actions_row_activated_cb"/>
+ <style>
+ <class name="browser-actions-listbox"/>\
+ <class name="background"/>
+ </style>
+ </object>
+ </child>
+ </object>
+ </child>
+ <child>
+ <object class="GtkBox" id="browser_actions_popup_view_box">
+ <property name="visible">True</property>
+ <property name="orientation">vertical</property>
+ <property name="spacing">6</property>
+ <child>
+ <object class="GtkBox">
+ <property name="visible">True</property>
+ <property name="orientation">horizontal</property>
+ <property name="spacing">6</property>
+ <child>
+ <object class="GtkButton">
+ <property name="visible">True</property>
+ <signal name="clicked" handler="browser_actions_popup_view_back_clicked_cb"/>
+ <style>
+ <class name="flat"/>
+ </style>
+ <child>
+ <object class="GtkImage">
+ <property name="visible">True</property>
+ <property name="icon-name">go-previous-symbolic</property>
+ </object>
+ </child>
+ </object>
+ </child>
+ <child>
+ <object class="GtkLabel" id="browser_actions_popup_view_label">
+ <property name="visible">True</property>
+ <property name="hexpand">True</property>
+ <style>
+ <class name="heading"/>
+ </style>
+ </object>
+ </child>
+ </object>
+ </child>
+ </object>
+ </child>
+ </object>
+ </child>
+ </object>
</interface>
diff --git a/src/resources/gtk/browser-action-row.ui b/src/resources/gtk/browser-action-row.ui
new file mode 100644
index 000000000..0ba4b0d01
--- /dev/null
+++ b/src/resources/gtk/browser-action-row.ui
@@ -0,0 +1,32 @@
+<?xml version="1.0" encoding="UTF-8"?>
+<interface>
+ <requires lib="gtk+" version="3.18"/>
+ <template class="EphyBrowserActionRow" parent="GtkListBoxRow">
+ <property name="visible">True</property>
+ <child>
+ <object class="GtkBox">
+ <property name="visible">True</property>
+ <property name="spacing">6</property>
+ <property name="margin-start">6</property>
+ <property name="margin-end">6</property>
+ <property name="margin-top">6</property>
+ <property name="margin-bottom">6</property>
+ <child>
+ <object class="GtkImage" id="browser_action_image">
+ <property name="visible">True</property>
+ <property name="pixel-size">16</property>
+ </object>
+ </child>
+ <child>
+ <object class="GtkLabel" id="title_label">
+ <property name="visible">True</property>
+ <property name="hexpand">True</property>
+ <property name="ellipsize">end</property>
+ <property name="max_width_chars">40</property>
+ <property name="xalign">0</property>
+ </object>
+ </child>
+ </object>
+ </child>
+ </template>
+</interface>
diff --git a/src/resources/themes/_shared-base.scss b/src/resources/themes/_shared-base.scss
index d1628d4e1..7c0a6bc71 100644
--- a/src/resources/themes/_shared-base.scss
+++ b/src/resources/themes/_shared-base.scss
@@ -253,3 +253,7 @@ fullscreenbox > flap {
background: linear-gradient(to bottom, gtkalpha(black, .1), gtkalpha(black, .0));
}
}
+
+.browser-actions-listbox {
+ padding: 6px;
+}
\ No newline at end of file
diff --git a/src/webextension/api/commands.c b/src/webextension/api/commands.c
index 31e3c4a2e..8f4c3218b 100644
--- a/src/webextension/api/commands.c
+++ b/src/webextension/api/commands.c
@@ -33,12 +33,11 @@ on_command_activated (GAction *action,
EphyWebExtensionManager *manager = ephy_web_extension_manager_get_default ();
EphyShell *shell = ephy_shell_get_default ();
EphyWebView *view = EPHY_WEB_VIEW (ephy_shell_get_active_web_view (shell));
- EphyWindow *window = EPHY_WINDOW (gtk_application_get_active_window (GTK_APPLICATION (shell)));
GtkWidget *button;
const char *command_name = g_object_get_data (G_OBJECT (action), "command-name-json");
if (strcmp (command_name, "\"_execute_browser_action\"") == 0) {
- ephy_web_extension_manager_activate_browser_action (manager, self, window);
+ ephy_web_extension_manager_show_browser_action (manager, self);
return;
} else if (strcmp (command_name, "\"_execute_page_action\"") == 0) {
button = ephy_web_extension_manager_get_page_action (manager, self, view);
diff --git a/src/webextension/api/menus.c b/src/webextension/api/menus.c
index 674f39a75..1cc6f0cfb 100644
--- a/src/webextension/api/menus.c
+++ b/src/webextension/api/menus.c
@@ -558,10 +558,8 @@ menu_activate_browser_action (gpointer user_data)
{
EphyWebExtension *web_extension = user_data;
EphyWebExtensionManager *manager = ephy_web_extension_manager_get_default ();
- EphyShell *shell = ephy_shell_get_default ();
- EphyWindow *window = EPHY_WINDOW (gtk_application_get_active_window (GTK_APPLICATION (shell)));
- ephy_web_extension_manager_activate_browser_action (manager, web_extension, window);
+ ephy_web_extension_manager_show_browser_action (manager, web_extension);
return G_SOURCE_REMOVE;
}
diff --git a/src/webextension/ephy-browser-action-row.c b/src/webextension/ephy-browser-action-row.c
new file mode 100644
index 000000000..164645902
--- /dev/null
+++ b/src/webextension/ephy-browser-action-row.c
@@ -0,0 +1,147 @@
+/* -*- Mode: C; tab-width: 2; indent-tabs-mode: nil; c-basic-offset: 2 -*- */
+/*
+ * Copyright © 2016 Iulian-Gabriel Radu <iulian radu67 gmail com>
+ * Copyright 2022 Igalia S.L.
+ *
+ * This file is part of Epiphany.
+ *
+ * Epiphany is free software: you can redistribute it and/or modify
+ * it under the terms of the GNU General Public License as published by
+ * the Free Software Foundation, either version 3 of the License, or
+ * (at your option) any later version.
+ *
+ * Epiphany is distributed in the hope that it will be useful,
+ * but WITHOUT ANY WARRANTY; without even the implied warranty of
+ * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
+ * GNU General Public License for more details.
+ *
+ * You should have received a copy of the GNU General Public License
+ * along with Epiphany. If not, see <http://www.gnu.org/licenses/>.
+ */
+
+#include "config.h"
+
+#include "ephy-browser-action-row.h"
+
+struct _EphyBrowserActionRow {
+ GtkListBoxRow parent_instance;
+
+ EphyBrowserAction *browser_action;
+
+ GtkWidget *browser_action_image;
+ GtkWidget *title_label;
+};
+
+G_DEFINE_FINAL_TYPE (EphyBrowserActionRow, ephy_browser_action_row, GTK_TYPE_LIST_BOX_ROW)
+
+enum {
+ PROP_0,
+ PROP_BROWSER_ACTION,
+ LAST_PROP
+};
+
+static GParamSpec *obj_properties[LAST_PROP];
+
+static void
+ephy_browser_action_row_set_property (GObject *object,
+ guint prop_id,
+ const GValue *value,
+ GParamSpec *pspec)
+{
+ EphyBrowserActionRow *self = EPHY_BROWSER_ACTION_ROW (object);
+
+ switch (prop_id) {
+ case PROP_BROWSER_ACTION:
+ self->browser_action = g_value_dup_object (value);
+ break;
+ default:
+ G_OBJECT_WARN_INVALID_PROPERTY_ID (object, prop_id, pspec);
+ }
+}
+
+static void
+ephy_browser_action_row_get_property (GObject *object,
+ guint prop_id,
+ GValue *value,
+ GParamSpec *pspec)
+{
+ EphyBrowserActionRow *self = EPHY_BROWSER_ACTION_ROW (object);
+
+ switch (prop_id) {
+ case PROP_BROWSER_ACTION:
+ g_value_set_object (value, ephy_browser_action_row_get_browser_action (self));
+ break;
+ default:
+ G_OBJECT_WARN_INVALID_PROPERTY_ID (object, prop_id, pspec);
+ }
+}
+
+static void
+ephy_browser_action_row_constructed (GObject *object)
+{
+ EphyBrowserActionRow *self = EPHY_BROWSER_ACTION_ROW (object);
+
+ gtk_label_set_label (GTK_LABEL (self->title_label),
+ ephy_browser_action_get_title (self->browser_action));
+ gtk_image_set_from_pixbuf (GTK_IMAGE (self->browser_action_image),
+ ephy_browser_action_get_pixbuf (self->browser_action, 16));
+
+ G_OBJECT_CLASS (ephy_browser_action_row_parent_class)->constructed (object);
+}
+
+static void
+ephy_browser_action_row_dispose (GObject *object)
+{
+ EphyBrowserActionRow *self = EPHY_BROWSER_ACTION_ROW (object);
+
+ g_clear_object (&self->browser_action);
+
+ G_OBJECT_CLASS (ephy_browser_action_row_parent_class)->dispose (object);
+}
+
+static void
+ephy_browser_action_row_class_init (EphyBrowserActionRowClass *klass)
+{
+ GObjectClass *object_class = G_OBJECT_CLASS (klass);
+ GtkWidgetClass *widget_class = GTK_WIDGET_CLASS (klass);
+
+ object_class->set_property = ephy_browser_action_row_set_property;
+ object_class->get_property = ephy_browser_action_row_get_property;
+ object_class->dispose = ephy_browser_action_row_dispose;
+ object_class->constructed = ephy_browser_action_row_constructed;
+
+ obj_properties[PROP_BROWSER_ACTION] =
+ g_param_spec_object ("browser-action",
+ "",
+ "",
+ EPHY_TYPE_BROWSER_ACTION,
+ G_PARAM_READWRITE | G_PARAM_CONSTRUCT_ONLY | G_PARAM_STATIC_STRINGS);
+
+ g_object_class_install_properties (object_class, LAST_PROP, obj_properties);
+
+ gtk_widget_class_set_template_from_resource (widget_class,
"/org/gnome/epiphany/gtk/browser-action-row.ui");
+ gtk_widget_class_bind_template_child (widget_class, EphyBrowserActionRow, browser_action_image);
+ gtk_widget_class_bind_template_child (widget_class, EphyBrowserActionRow, title_label);
+}
+
+static void
+ephy_browser_action_row_init (EphyBrowserActionRow *self)
+{
+ gtk_widget_init_template (GTK_WIDGET (self));
+}
+
+GtkWidget *
+ephy_browser_action_row_new (EphyBrowserAction *browser_action)
+{
+ return g_object_new (EPHY_TYPE_BROWSER_ACTION_ROW,
+ "browser-action", browser_action,
+ NULL);
+}
+
+EphyBrowserAction *
+ephy_browser_action_row_get_browser_action (EphyBrowserActionRow *self)
+{
+ g_assert (EPHY_IS_BROWSER_ACTION_ROW (self));
+
+ return self->browser_action;
+}
diff --git a/src/webextension/ephy-browser-action-row.h b/src/webextension/ephy-browser-action-row.h
new file mode 100644
index 000000000..b309940be
--- /dev/null
+++ b/src/webextension/ephy-browser-action-row.h
@@ -0,0 +1,37 @@
+/* -*- Mode: C; tab-width: 2; indent-tabs-mode: nil; c-basic-offset: 2 -*- */
+/*
+ * Copyright 2022 Igalia S.L.
+ *
+ * This file is part of Epiphany.
+ *
+ * Epiphany is free software: you can redistribute it and/or modify
+ * it under the terms of the GNU General Public License as published by
+ * the Free Software Foundation, either version 3 of the License, or
+ * (at your option) any later version.
+ *
+ * Epiphany is distributed in the hope that it will be useful,
+ * but WITHOUT ANY WARRANTY; without even the implied warranty of
+ * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
+ * GNU General Public License for more details.
+ *
+ * You should have received a copy of the GNU General Public License
+ * along with Epiphany. If not, see <http://www.gnu.org/licenses/>.
+ */
+
+#pragma once
+
+#include "ephy-browser-action.h"
+
+#include <gtk/gtk.h>
+
+G_BEGIN_DECLS
+
+#define EPHY_TYPE_BROWSER_ACTION_ROW (ephy_browser_action_row_get_type ())
+
+G_DECLARE_FINAL_TYPE (EphyBrowserActionRow, ephy_browser_action_row, EPHY, BROWSER_ACTION_ROW, GtkListBoxRow)
+
+GtkWidget *ephy_browser_action_row_new (EphyBrowserAction *browser_action);
+
+EphyBrowserAction *ephy_browser_action_row_get_browser_action (EphyBrowserActionRow *self);
+
+G_END_DECLS
diff --git a/src/webextension/ephy-browser-action.c b/src/webextension/ephy-browser-action.c
new file mode 100644
index 000000000..e55b3e670
--- /dev/null
+++ b/src/webextension/ephy-browser-action.c
@@ -0,0 +1,150 @@
+/* -*- Mode: C; tab-width: 2; indent-tabs-mode: nil; c-basic-offset: 2 -*- */
+/*
+ * Copyright © 2022 Igalia S.L.
+ *
+ * This file is part of Epiphany.
+ *
+ * Epiphany is free software: you can redistribute it and/or modify
+ * it under the terms of the GNU General Public License as published by
+ * the Free Software Foundation, either version 3 of the License, or
+ * (at your option) any later version.
+ *
+ * Epiphany is distributed in the hope that it will be useful,
+ * but WITHOUT ANY WARRANTY; without even the implied warranty of
+ * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
+ * GNU General Public License for more details.
+ *
+ * You should have received a copy of the GNU General Public License
+ * along with Epiphany. If not, see <http://www.gnu.org/licenses/>.
+ */
+
+#include "config.h"
+
+#include "ephy-browser-action.h"
+#include "ephy-web-extension-manager.h"
+
+struct _EphyBrowserAction {
+ GObject parent_instance;
+ EphyWebExtension *web_extension;
+};
+
+G_DEFINE_FINAL_TYPE (EphyBrowserAction, ephy_browser_action, G_TYPE_OBJECT)
+
+enum {
+ PROP_0,
+ PROP_WEB_EXTENSION,
+ N_PROPS
+};
+
+static GParamSpec *properties[N_PROPS];
+
+EphyBrowserAction *
+ephy_browser_action_new (EphyWebExtension *web_extension)
+{
+ return g_object_new (EPHY_TYPE_BROWSER_ACTION,
+ "web-extension", web_extension,
+ NULL);
+}
+
+const char *
+ephy_browser_action_get_title (EphyBrowserAction *self)
+{
+ const char *short_name = ephy_web_extension_get_short_name (self->web_extension);
+
+ return short_name && *short_name ? short_name : ephy_web_extension_get_name (self->web_extension);
+}
+
+GdkPixbuf *
+ephy_browser_action_get_pixbuf (EphyBrowserAction *self,
+ gint64 size)
+{
+ return ephy_web_extension_get_icon (self->web_extension, size);
+}
+
+EphyWebExtension *
+ephy_browser_action_get_web_extension (EphyBrowserAction *self)
+{
+ return self->web_extension;
+}
+
+gboolean
+ephy_browser_action_activate (EphyBrowserAction *self)
+{
+ EphyWebExtensionManager *manager = ephy_web_extension_manager_get_default ();
+
+ /* If it has no popup clicking just emits this event and is handled already. */
+ if (ephy_web_extension_get_browser_popup (self->web_extension) == NULL) {
+ ephy_web_extension_manager_emit_in_background_view (manager, self->web_extension,
"browserAction.onClicked", "");
+ return TRUE;
+ }
+
+ return FALSE;
+}
+
+static void
+ephy_browser_action_finalize (GObject *object)
+{
+ EphyBrowserAction *self = (EphyBrowserAction *)object;
+
+ g_clear_object (&self->web_extension);
+
+ G_OBJECT_CLASS (ephy_browser_action_parent_class)->finalize (object);
+}
+
+static void
+ephy_browser_action_get_property (GObject *object,
+ guint prop_id,
+ GValue *value,
+ GParamSpec *pspec)
+{
+ EphyBrowserAction *self = EPHY_BROWSER_ACTION (object);
+
+ switch (prop_id) {
+ case PROP_WEB_EXTENSION:
+ g_value_set_object (value, self->web_extension);
+ break;
+ default:
+ G_OBJECT_WARN_INVALID_PROPERTY_ID (object, prop_id, pspec);
+ }
+}
+
+static void
+ephy_browser_action_set_property (GObject *object,
+ guint prop_id,
+ const GValue *value,
+ GParamSpec *pspec)
+{
+ EphyBrowserAction *self = EPHY_BROWSER_ACTION (object);
+
+ switch (prop_id) {
+ case PROP_WEB_EXTENSION:
+ g_set_object (&self->web_extension, g_value_dup_object (value));
+ break;
+ default:
+ G_OBJECT_WARN_INVALID_PROPERTY_ID (object, prop_id, pspec);
+ }
+}
+
+static void
+ephy_browser_action_class_init (EphyBrowserActionClass *klass)
+{
+ GObjectClass *object_class = G_OBJECT_CLASS (klass);
+
+ object_class->finalize = ephy_browser_action_finalize;
+ object_class->get_property = ephy_browser_action_get_property;
+ object_class->set_property = ephy_browser_action_set_property;
+
+ properties[PROP_WEB_EXTENSION] =
+ g_param_spec_object ("web-extension",
+ "",
+ "",
+ EPHY_TYPE_WEB_EXTENSION,
+ G_PARAM_READWRITE | G_PARAM_CONSTRUCT_ONLY | G_PARAM_STATIC_STRINGS);
+
+ g_object_class_install_properties (object_class, N_PROPS, properties);
+}
+
+static void
+ephy_browser_action_init (EphyBrowserAction *self)
+{
+}
diff --git a/src/webextension/ephy-browser-action.h b/src/webextension/ephy-browser-action.h
new file mode 100644
index 000000000..96d2fbd27
--- /dev/null
+++ b/src/webextension/ephy-browser-action.h
@@ -0,0 +1,39 @@
+/* -*- Mode: C; tab-width: 2; indent-tabs-mode: nil; c-basic-offset: 2 -*- */
+/*
+ * Copyright © 2022 Igalia S.L.
+ *
+ * This file is part of Epiphany.
+ *
+ * Epiphany is free software: you can redistribute it and/or modify
+ * it under the terms of the GNU General Public License as published by
+ * the Free Software Foundation, either version 3 of the License, or
+ * (at your option) any later version.
+ *
+ * Epiphany is distributed in the hope that it will be useful,
+ * but WITHOUT ANY WARRANTY; without even the implied warranty of
+ * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
+ * GNU General Public License for more details.
+ *
+ * You should have received a copy of the GNU General Public License
+ * along with Epiphany. If not, see <http://www.gnu.org/licenses/>.
+ */
+
+#pragma once
+
+#include <gio/gio.h>
+
+#include "ephy-web-extension.h"
+
+G_BEGIN_DECLS
+
+#define EPHY_TYPE_BROWSER_ACTION (ephy_browser_action_get_type())
+
+G_DECLARE_FINAL_TYPE (EphyBrowserAction, ephy_browser_action, EPHY, BROWSER_ACTION, GObject)
+
+EphyBrowserAction *ephy_browser_action_new (EphyWebExtension *web_extension);
+const char *ephy_browser_action_get_title (EphyBrowserAction *self);
+GdkPixbuf *ephy_browser_action_get_pixbuf (EphyBrowserAction *self, gint64 size);
+EphyWebExtension *ephy_browser_action_get_web_extension (EphyBrowserAction *self);
+gboolean ephy_browser_action_activate (EphyBrowserAction *self);
+
+G_END_DECLS
diff --git a/src/webextension/ephy-web-extension-manager.c b/src/webextension/ephy-web-extension-manager.c
index 2cb6b9340..2adaab326 100644
--- a/src/webextension/ephy-web-extension-manager.c
+++ b/src/webextension/ephy-web-extension-manager.c
@@ -20,6 +20,7 @@
#include "config.h"
+#include "ephy-browser-action.h"
#include "ephy-debug.h"
#include "ephy-embed-shell.h"
#include "ephy-embed-prefs.h"
@@ -58,7 +59,9 @@ struct _EphyWebExtensionManager {
GCancellable *cancellable;
GPtrArray *web_extensions;
GHashTable *page_action_map;
+
GHashTable *browser_action_map;
+ GListStore *browser_actions;
GHashTable *user_agent_overrides;
@@ -87,6 +90,7 @@ EphyWebExtensionApiHandler api_handlers[] = {
enum {
CHANGED,
+ SHOW_BROWSER_ACTION,
LAST_SIGNAL
};
@@ -206,12 +210,6 @@ ephy_web_extension_manager_scan_directory_async (EphyWebExtensionManager *self,
self);
}
-static void
-destroy_widget_list (GSList *widget_list)
-{
- g_slist_free_full (widget_list, (GDestroyNotify)gtk_widget_destroy);
-}
-
static void
ephy_webextension_scheme_cb (WebKitURISchemeRequest *request,
gpointer user_data)
@@ -269,7 +267,8 @@ ephy_web_extension_manager_constructed (GObject *object)
self->background_web_views = g_hash_table_new_full (g_direct_hash, g_direct_equal, NULL,
(GDestroyNotify)gtk_widget_destroy);
self->popup_web_views = g_hash_table_new_full (g_direct_hash, g_direct_equal, NULL,
(GDestroyNotify)g_ptr_array_free);
self->page_action_map = g_hash_table_new_full (NULL, NULL, NULL, (GDestroyNotify)g_hash_table_destroy);
- self->browser_action_map = g_hash_table_new_full (NULL, NULL, NULL, (GDestroyNotify)destroy_widget_list);
+ self->browser_action_map = g_hash_table_new_full (NULL, NULL, NULL, (GDestroyNotify)g_object_unref);
+ self->browser_actions = g_list_store_new (EPHY_TYPE_BROWSER_ACTION);
self->pending_messages = g_hash_table_new_full (g_direct_hash, g_direct_equal, NULL,
(GDestroyNotify)g_hash_table_destroy);
self->web_extensions = g_ptr_array_new_full (0, g_object_unref);
self->user_agent_overrides = create_user_agent_overrides ();
@@ -284,8 +283,12 @@ ephy_web_extension_manager_dispose (GObject *object)
ephy_web_extension_api_downloads_dispose (self);
+ g_list_store_remove_all (self->browser_actions);
+
g_clear_pointer (&self->background_web_views, g_hash_table_destroy);
g_clear_pointer (&self->popup_web_views, g_hash_table_destroy);
+ g_clear_object (&self->browser_actions);
+ g_clear_pointer (&self->browser_action_map, g_hash_table_destroy);
g_clear_pointer (&self->page_action_map, g_hash_table_destroy);
g_clear_pointer (&self->pending_messages, g_hash_table_destroy);
g_clear_pointer (&self->web_extensions, g_ptr_array_unref);
@@ -306,6 +309,14 @@ ephy_web_extension_manager_class_init (EphyWebExtensionManagerClass *klass)
G_SIGNAL_RUN_FIRST | G_SIGNAL_ACTION,
0, NULL, NULL, NULL,
G_TYPE_NONE, 0);
+
+ signals[SHOW_BROWSER_ACTION] =
+ g_signal_new ("show-browser-action",
+ G_OBJECT_CLASS_TYPE (object_class),
+ G_SIGNAL_RUN_FIRST,
+ 0, NULL, NULL, NULL,
+ G_TYPE_NONE,
+ 1, EPHY_TYPE_BROWSER_ACTION);
}
static void
@@ -1066,17 +1077,20 @@ ephy_web_extension_manager_register_popup_view (EphyWebExtensionManager *manager
g_signal_connect (web_view, "destroy", G_CALLBACK (on_popup_view_destroyed), web_extension);
}
-static GtkWidget *
-create_browser_popup (EphyWebExtension *web_extension)
+GtkWidget *
+ephy_web_extension_manager_create_browser_popup (EphyWebExtensionManager *self,
+ EphyWebExtension *web_extension)
{
- EphyWebExtensionManager *manager = ephy_web_extension_manager_get_default ();
GtkWidget *web_view;
g_autofree char *popup_uri = NULL;
const char *popup;
web_view = ephy_web_extensions_manager_create_web_extensions_webview (web_extension);
+ gtk_widget_set_hexpand (web_view, TRUE);
+ gtk_widget_set_vexpand (web_view, TRUE);
gtk_widget_hide (web_view); /* Shown in on_popup_load_changed. */
- ephy_web_extension_manager_register_popup_view (manager, web_extension, web_view);
+
+ ephy_web_extension_manager_register_popup_view (self, web_extension, web_view);
popup = ephy_web_extension_get_browser_popup (web_extension);
popup_uri = g_strdup_printf ("ephy-webextension://%s/%s", ephy_web_extension_get_guid (web_extension),
popup);
@@ -1086,75 +1100,6 @@ create_browser_popup (EphyWebExtension *web_extension)
return web_view;
}
-static gboolean
-on_browser_action_clicked (GtkWidget *event_box,
- gpointer user_data)
-{
- EphyWebExtension *web_extension = EPHY_WEB_EXTENSION (user_data);
- EphyWebExtensionManager *self = ephy_web_extension_manager_get_default ();
- g_autofree char *script = NULL;
- WebKitWebView *web_view = ephy_web_extension_manager_get_background_web_view (self, web_extension);
-
- script = g_strdup_printf ("window.browser.browserAction.onClicked._emit();");
-
- webkit_web_view_run_javascript (web_view,
- script,
- NULL,
- NULL,
- NULL);
-
- return GDK_EVENT_STOP;
-}
-
-static void
-on_browser_action_visible_changed (GtkWidget *popover,
- GParamSpec *pspec,
- gpointer user_data)
-{
- EphyWebExtension *web_extension = EPHY_WEB_EXTENSION (user_data);
- GtkWidget *child;
-
- if (gtk_widget_get_visible (popover)) {
- child = create_browser_popup (web_extension);
- gtk_container_add (GTK_CONTAINER (popover), child);
- } else {
- child = gtk_bin_get_child (GTK_BIN (popover));
- gtk_container_remove (GTK_CONTAINER (popover), child);
- }
-}
-
-GtkWidget *
-create_browser_action (EphyWebExtension *web_extension,
- EphyWindow *window)
-{
- GtkWidget *button;
- GtkWidget *image;
- GtkWidget *popover;
- GdkPixbuf *pixbuf;
-
- pixbuf = ephy_web_extension_browser_action_get_icon (web_extension, 16);
- if (pixbuf)
- image = gtk_image_new_from_pixbuf (pixbuf);
- else
- image = gtk_image_new_from_icon_name ("application-x-addon-symbolic", GTK_ICON_SIZE_BUTTON);
-
- if (ephy_web_extension_get_browser_popup (web_extension)) {
- button = gtk_menu_button_new ();
- popover = gtk_popover_new (NULL);
- g_signal_connect (popover, "notify::visible", G_CALLBACK (on_browser_action_visible_changed),
web_extension);
- gtk_menu_button_set_popover (GTK_MENU_BUTTON (button), popover);
- gtk_button_set_image (GTK_BUTTON (button), image);
- } else {
- button = gtk_button_new ();
- g_signal_connect_object (button, "clicked", G_CALLBACK (on_browser_action_clicked), web_extension, 0);
- gtk_button_set_image (GTK_BUTTON (button), image);
- }
-
- gtk_widget_set_visible (button, TRUE);
-
- return button;
-}
-
void
ephy_web_extension_manager_add_web_extension_to_window (EphyWebExtensionManager *self,
EphyWebExtension *web_extension,
@@ -1174,16 +1119,6 @@ ephy_web_extension_manager_add_web_extension_to_window (EphyWebExtensionManager
ephy_web_extension_manager_add_web_extension_to_webview (self, web_extension, window, web_view);
}
- if (ephy_web_extension_has_browser_action (web_extension)) {
- GtkWidget *browser_action_widget = create_browser_action (web_extension, window);
- GSList *widget_list = g_hash_table_lookup (self->browser_action_map, web_extension);
-
- ephy_header_bar_add_browser_action (EPHY_HEADER_BAR (ephy_window_get_header_bar (window)),
browser_action_widget);
-
- g_hash_table_steal (self->browser_action_map, web_extension); /* Avoid freeing list. */
- g_hash_table_insert (self->browser_action_map, web_extension, g_slist_append (widget_list,
browser_action_widget));
- }
-
ephy_web_extension_manager_update_location_entry (self, window);
g_signal_connect_object (view, "page-attached", G_CALLBACK (page_attached_cb), web_extension, 0);
}
@@ -1356,6 +1291,23 @@ application_window_removed_cb (EphyShell *shell,
ephy_web_extension_manager_emit_in_extension_views (manager, web_extension, "windows.onRemoved",
window_json);
}
+static void
+remove_browser_action (EphyWebExtensionManager *self,
+ EphyWebExtension *web_extension)
+{
+ EphyBrowserAction *action;
+ guint position;
+
+ action = g_hash_table_lookup (self->browser_action_map, web_extension);
+ if (!action)
+ return;
+
+ g_assert (g_list_store_find (self->browser_actions, action, &position));
+ g_list_store_remove (self->browser_actions, position);
+
+ g_hash_table_remove (self->browser_action_map, web_extension);
+}
+
void
ephy_web_extension_manager_set_active (EphyWebExtensionManager *self,
EphyWebExtension *web_extension,
@@ -1401,42 +1353,33 @@ ephy_web_extension_manager_set_active (EphyWebExtensionManager *self,
if (ephy_web_extension_has_background_web_view (web_extension))
run_background_script (self, web_extension);
+ if (ephy_web_extension_has_browser_action (web_extension)) {
+ EphyBrowserAction *action = ephy_browser_action_new (web_extension);
+ g_list_store_append (self->browser_actions, action);
+ g_hash_table_insert (self->browser_action_map, web_extension, g_steal_pointer (&action));
+ }
+
ephy_web_extension_api_commands_init (web_extension);
} else {
g_signal_handlers_disconnect_by_data (shell, web_extension);
- g_hash_table_remove (self->browser_action_map, web_extension);
+ remove_browser_action (self, web_extension);
g_hash_table_remove (self->background_web_views, web_extension);
g_object_set_data (G_OBJECT (web_extension), "alarms", NULL); /* Set in alarms.c */
ephy_web_extension_api_commands_dispose (web_extension);
}
}
-gint
-get_browser_action_for_window (GtkWidget *widget,
- GtkWidget *window)
-{
- return gtk_widget_get_toplevel (widget) != window;
-}
-
-
void
-ephy_web_extension_manager_activate_browser_action (EphyWebExtensionManager *self,
- EphyWebExtension *web_extension,
- EphyWindow *window)
+ephy_web_extension_manager_show_browser_action (EphyWebExtensionManager *self,
+ EphyWebExtension *web_extension)
{
- GSList *table, *l;
- GtkWidget *button;
+ EphyBrowserAction *action = EPHY_BROWSER_ACTION (g_hash_table_lookup (self->browser_action_map,
web_extension));
- table = g_hash_table_lookup (self->browser_action_map, web_extension);
- if (table) {
- l = g_slist_find_custom (table, window, (GCompareFunc)get_browser_action_for_window);
-
- g_assert (l && l->data);
+ if (!action || ephy_browser_action_activate (action))
+ return;
- button = l->data;
- gtk_widget_mnemonic_activate (button, FALSE);
- }
+ g_signal_emit (self, signals[SHOW_BROWSER_ACTION], 0, action);
}
GtkWidget *
@@ -1454,6 +1397,12 @@ ephy_web_extension_manager_get_page_action (EphyWebExtensionManager *self,
return ret;
}
+GListStore *
+ephy_web_extension_manager_get_browser_actions (EphyWebExtensionManager *self)
+{
+ return self->browser_actions;
+}
+
void
ephy_web_extension_manager_handle_notifications_action (EphyWebExtensionManager *self,
GVariant *params)
@@ -1653,6 +1602,24 @@ on_extension_emit_ready (GObject *source,
g_warning ("Emitting in view errored: %s", error->message);
}
+void
+ephy_web_extension_manager_emit_in_background_view (EphyWebExtensionManager *self,
+ EphyWebExtension *web_extension,
+ const char *name,
+ const char *json)
+{
+ g_autofree char *script = NULL;
+ WebKitWebView *web_view = ephy_web_extension_manager_get_background_web_view (self, web_extension);
+
+ script = g_strdup_printf ("window.browser.%s._emit(%s);", name, json);
+
+ webkit_web_view_run_javascript (web_view,
+ script,
+ NULL,
+ NULL,
+ NULL);
+}
+
static void
ephy_web_extension_manager_emit_in_extension_views_internal (EphyWebExtensionManager *self,
EphyWebExtension *web_extension,
diff --git a/src/webextension/ephy-web-extension-manager.h b/src/webextension/ephy-web-extension-manager.h
index 13f68b22d..89901bd75 100644
--- a/src/webextension/ephy-web-extension-manager.h
+++ b/src/webextension/ephy-web-extension-manager.h
@@ -64,17 +64,19 @@ void ephy_web_extension_manager_set_active
EphyWebExtension
*web_extension,
gboolean
active);
-void ephy_web_extension_manager_activate_browser_action (EphyWebExtensionManager
*self,
- EphyWebExtension
*web_extension,
- EphyWindow
*window);
+void ephy_web_extension_manager_show_browser_action (EphyWebExtensionManager
*self,
+ EphyWebExtension
*web_extension);
GtkWidget *ephy_web_extension_manager_get_page_action (EphyWebExtensionManager
*self,
EphyWebExtension
*web_extension,
EphyWebView
*web_view);
+GListStore *ephy_web_extension_manager_get_browser_actions (EphyWebExtensionManager
*self);
+
WebKitWebView *ephy_web_extension_manager_get_background_web_view (EphyWebExtensionManager
*self,
EphyWebExtension
*web_extension);
-
+GtkWidget *ephy_web_extension_manager_create_browser_popup (EphyWebExtensionManager
*self,
+ EphyWebExtension
*web_extension);
void ephy_web_extension_manager_handle_notifications_action (EphyWebExtensionManager
*self,
GVariant
*params);
@@ -101,6 +103,10 @@ void ephy_web_extension_manager_emit_in_tab_with_reply
WebKitWebView
*target_web_view,
const char
*sender_json,
GTask
*reply_task);
+void ephy_web_extension_manager_emit_in_background_view (EphyWebExtensionManager
*self,
+ EphyWebExtension
*web_extension,
+ const char
*name,
+ const char
*json);
GtkWidget *ephy_web_extensions_manager_create_web_extensions_webview (EphyWebExtension
*web_extension);
diff --git a/src/webextension/meson.build b/src/webextension/meson.build
index c4ad72c50..32a76c5af 100644
--- a/src/webextension/meson.build
+++ b/src/webextension/meson.build
@@ -10,6 +10,8 @@ ephywebextension_src = [
'webextension/api/storage.c',
'webextension/api/tabs.c',
'webextension/api/windows.c',
+ 'webextension/ephy-browser-action.c',
+ 'webextension/ephy-browser-action-row.c',
'webextension/ephy-json-utils.c',
'webextension/ephy-web-extension-manager.c',
'webextension/ephy-web-extension.c',
[
Date Prev][
Date Next] [
Thread Prev][
Thread Next]
[
Thread Index]
[
Date Index]
[
Author Index]