mirror of https://github.com/GNOME/gimp.git
243 lines
7.5 KiB
C
243 lines
7.5 KiB
C
/* GIMP - The GNU Image Manipulation Program
|
|
* Copyright (C) 1995 Spencer Kimball and Peter Mattis
|
|
*
|
|
* gimpmenubar.c
|
|
* Copyright (C) 2023 Jehan
|
|
*
|
|
* This program 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.
|
|
*
|
|
* This program 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 this program. If not, see <https://www.gnu.org/licenses/>.
|
|
*/
|
|
|
|
#include "config.h"
|
|
|
|
#include <gegl.h>
|
|
#include <gtk/gtk.h>
|
|
|
|
#include "libgimpbase/gimpbase.h"
|
|
|
|
#include "widgets-types.h"
|
|
|
|
#include "core/gimp.h"
|
|
|
|
#include "gimpaction.h"
|
|
#include "gimpmenu.h"
|
|
#include "gimpmenubar.h"
|
|
#include "gimpmenumodel.h"
|
|
#include "gimpmenushell.h"
|
|
#include "gimpradioaction.h"
|
|
#include "gimpuimanager.h"
|
|
#include "gimpwidgets-utils.h"
|
|
|
|
|
|
/**
|
|
* GimpMenuBar:
|
|
*
|
|
* Our own menu bar widget.
|
|
*
|
|
* We cannot use the simpler gtk_application_set_menubar() because it lacks
|
|
* tooltip support and unfortunately GTK does not plan to implement this:
|
|
* https://gitlab.gnome.org/GNOME/gtk/-/issues/785
|
|
* This is why we need to implement our own GimpMenuBar subclass.
|
|
*/
|
|
|
|
struct _GimpMenuBarPrivate
|
|
{
|
|
GTree *menus;
|
|
};
|
|
|
|
|
|
/* local function prototypes */
|
|
|
|
static void gimp_menu_bar_iface_init (GimpMenuShellInterface *iface);
|
|
|
|
static void gimp_menu_bar_dispose (GObject *object);
|
|
|
|
static void gimp_menu_bar_append (GimpMenuShell *shell,
|
|
GimpMenuModel *model);
|
|
static void gimp_menu_bar_add_ui (GimpMenuShell *shell,
|
|
const gchar **paths,
|
|
const gchar *action_name,
|
|
gboolean top);
|
|
|
|
|
|
G_DEFINE_TYPE_WITH_CODE (GimpMenuBar, gimp_menu_bar, GTK_TYPE_MENU_BAR,
|
|
G_ADD_PRIVATE (GimpMenuBar)
|
|
G_IMPLEMENT_INTERFACE (GIMP_TYPE_MENU_SHELL, gimp_menu_bar_iface_init))
|
|
|
|
#define parent_class gimp_menu_bar_parent_class
|
|
|
|
|
|
/* Class functions */
|
|
|
|
static void
|
|
gimp_menu_bar_class_init (GimpMenuBarClass *klass)
|
|
{
|
|
GObjectClass *object_class = G_OBJECT_CLASS (klass);
|
|
|
|
object_class->dispose = gimp_menu_bar_dispose;
|
|
object_class->get_property = gimp_menu_shell_get_property;
|
|
object_class->set_property = gimp_menu_shell_set_property;
|
|
|
|
gimp_menu_shell_install_properties (object_class);
|
|
}
|
|
|
|
static void
|
|
gimp_menu_bar_iface_init (GimpMenuShellInterface *iface)
|
|
{
|
|
iface->append = gimp_menu_bar_append;
|
|
iface->add_ui = gimp_menu_bar_add_ui;
|
|
}
|
|
|
|
static void
|
|
gimp_menu_bar_init (GimpMenuBar *bar)
|
|
{
|
|
bar->priv = gimp_menu_bar_get_instance_private (bar);
|
|
|
|
bar->priv->menus = g_tree_new_full ((GCompareDataFunc) g_strcmp0, NULL,
|
|
g_free, NULL);
|
|
|
|
gimp_menu_shell_init (GIMP_MENU_SHELL (bar));
|
|
}
|
|
|
|
static void
|
|
gimp_menu_bar_dispose (GObject *object)
|
|
{
|
|
GimpMenuBar *bar = GIMP_MENU_BAR (object);
|
|
|
|
g_clear_pointer (&bar->priv->menus, g_tree_unref);
|
|
|
|
G_OBJECT_CLASS (parent_class)->dispose (object);
|
|
}
|
|
|
|
static void
|
|
gimp_menu_bar_append (GimpMenuShell *shell,
|
|
GimpMenuModel *model)
|
|
{
|
|
GimpMenuBar *bar = GIMP_MENU_BAR (shell);
|
|
GimpUIManager *manager = gimp_menu_shell_get_manager (GIMP_MENU_SHELL (shell));
|
|
gint n_items;
|
|
|
|
g_return_if_fail (GTK_IS_CONTAINER (shell));
|
|
|
|
n_items = g_menu_model_get_n_items (G_MENU_MODEL (model));
|
|
for (gint i = 0; i < n_items; i++)
|
|
{
|
|
GMenuModel *subsection;
|
|
GMenuModel *submenu;
|
|
gchar *label = NULL;
|
|
gchar *action_name = NULL;
|
|
|
|
subsection = g_menu_model_get_item_link (G_MENU_MODEL (model), i, G_MENU_LINK_SECTION);
|
|
submenu = g_menu_model_get_item_link (G_MENU_MODEL (model), i, G_MENU_LINK_SUBMENU);
|
|
g_menu_model_get_item_attribute (G_MENU_MODEL (model), i, G_MENU_ATTRIBUTE_LABEL, "s", &label);
|
|
g_menu_model_get_item_attribute (G_MENU_MODEL (model), i, G_MENU_ATTRIBUTE_ACTION, "s", &action_name);
|
|
|
|
if (submenu != NULL)
|
|
{
|
|
GtkWidget *subcontainer;
|
|
GtkWidget *item;
|
|
|
|
g_return_if_fail (label != NULL);
|
|
|
|
/* I don't show the item on purpose because
|
|
* gimp_menu_append() will show the parent item if any of
|
|
* the added actions are visible.
|
|
*/
|
|
item = gtk_menu_item_new_with_mnemonic (label);
|
|
gtk_container_add (GTK_CONTAINER (shell), item);
|
|
|
|
subcontainer = gimp_menu_new (manager);
|
|
gtk_menu_item_set_submenu (GTK_MENU_ITEM (item), subcontainer);
|
|
gimp_menu_shell_append (GIMP_MENU_SHELL (subcontainer), GIMP_MENU_MODEL (submenu));
|
|
gtk_widget_show (subcontainer);
|
|
|
|
g_tree_insert (bar->priv->menus,
|
|
gimp_utils_make_canonical_menu_label (label),
|
|
subcontainer);
|
|
}
|
|
else
|
|
{
|
|
if (subsection != NULL)
|
|
g_warning ("%s: unexpected subsection. Only menus are allowed in menu bar's root.",
|
|
G_STRFUNC);
|
|
if (action_name == NULL)
|
|
g_warning ("%s: missing action name. Only menus are allowed in menu bar's root.",
|
|
G_STRFUNC);
|
|
else
|
|
g_warning ("%s: unexpected action '%s'. Only menus are allowed in menu bar's root.",
|
|
G_STRFUNC, action_name);
|
|
}
|
|
g_free (label);
|
|
g_free (action_name);
|
|
g_clear_object (&submenu);
|
|
g_clear_object (&subsection);
|
|
}
|
|
}
|
|
|
|
static void
|
|
gimp_menu_bar_add_ui (GimpMenuShell *shell,
|
|
const gchar **paths,
|
|
const gchar *action_name,
|
|
gboolean top)
|
|
{
|
|
GimpMenuBar *bar = GIMP_MENU_BAR (shell);
|
|
GimpUIManager *manager = gimp_menu_shell_get_manager (GIMP_MENU_SHELL (shell));
|
|
|
|
g_return_if_fail (paths != NULL);
|
|
|
|
if (paths[0] == NULL)
|
|
{
|
|
g_warning ("%s: unexpected action '%s'. Only menus are allowed in menu bar's root.",
|
|
G_STRFUNC, action_name);
|
|
}
|
|
else
|
|
{
|
|
GtkWidget *menu = NULL;
|
|
|
|
menu = g_tree_lookup (bar->priv->menus, paths[0]);
|
|
|
|
if (menu == NULL)
|
|
{
|
|
GtkWidget *item;
|
|
|
|
item = gtk_menu_item_new_with_mnemonic (paths[0]);
|
|
gtk_container_add (GTK_CONTAINER (shell), item);
|
|
|
|
menu = gimp_menu_new (manager);
|
|
gtk_menu_item_set_submenu (GTK_MENU_ITEM (item), menu);
|
|
gtk_widget_show (menu);
|
|
|
|
g_tree_insert (bar->priv->menus, g_strdup (paths[0]), menu);
|
|
}
|
|
|
|
GIMP_MENU_SHELL_GET_INTERFACE (menu)->add_ui (GIMP_MENU_SHELL (menu),
|
|
paths + 1, action_name, top);
|
|
}
|
|
}
|
|
|
|
|
|
/* Public functions */
|
|
|
|
GtkWidget *
|
|
gimp_menu_bar_new (GimpMenuModel *model,
|
|
GimpUIManager *manager)
|
|
{
|
|
g_return_val_if_fail (GIMP_IS_UI_MANAGER (manager) &&
|
|
G_IS_MENU_MODEL (model), NULL);
|
|
|
|
return g_object_new (GIMP_TYPE_MENU_BAR,
|
|
"model", model,
|
|
"manager", manager,
|
|
NULL);
|
|
}
|