gimp_container_box_finalize() /o\... it's a minor miracle that it just warned and leaked instead of exploding completely. Fixes #14596.
386 lines
13 KiB
C
386 lines
13 KiB
C
/* GIMP - The GNU Image Manipulation Program
|
|
* Copyright (C) 1995 Spencer Kimball and Peter Mattis
|
|
*
|
|
* gimpcontainerbox.c
|
|
* Copyright (C) 2004-2025 Michael Natterer <mitch@gimp.org>
|
|
*
|
|
* 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 <string.h>
|
|
|
|
#include <gegl.h>
|
|
#include <gtk/gtk.h>
|
|
|
|
#include "libgimpwidgets/gimpwidgets.h"
|
|
|
|
#include "widgets-types.h"
|
|
|
|
#include "core/gimpcontainer.h"
|
|
#include "core/gimpcontext.h"
|
|
#include "core/gimp-utils.h"
|
|
|
|
#include "gimpcontainerbox.h"
|
|
#include "gimpcontainerview.h"
|
|
#include "gimpdnd.h"
|
|
#include "gimpdocked.h"
|
|
#include "gimppropwidgets.h"
|
|
#include "gimpview.h"
|
|
#include "gimpviewrenderer.h"
|
|
|
|
|
|
typedef struct _GimpContainerBoxPrivate GimpContainerBoxPrivate;
|
|
|
|
struct _GimpContainerBoxPrivate
|
|
{
|
|
GtkGesture *zoom_gesture;
|
|
gdouble zoom_last_value;
|
|
gdouble zoom_accum_scroll_delta;
|
|
};
|
|
|
|
#define GET_PRIVATE(obj) \
|
|
((GimpContainerBoxPrivate *) gimp_container_box_get_instance_private ((GimpContainerBox *) (obj)))
|
|
|
|
|
|
static void gimp_container_box_view_iface_init (GimpContainerViewInterface *iface);
|
|
static void gimp_container_box_docked_iface_init (GimpDockedInterface *iface);
|
|
|
|
static void gimp_container_box_constructed (GObject *object);
|
|
static void gimp_container_box_finalize (GObject *object);
|
|
|
|
static void gimp_container_box_set_context (GimpDocked *docked,
|
|
GimpContext *context);
|
|
static GtkWidget *
|
|
gimp_container_box_get_preview (GimpDocked *docked,
|
|
GimpContext *context,
|
|
GtkIconSize size);
|
|
|
|
static gboolean
|
|
gimp_container_box_scroll_to_zoom (GtkWidget *widget,
|
|
GdkEventScroll *event,
|
|
GimpContainerBox *box);
|
|
static void gimp_container_box_zoom_gesture_begin (GtkGestureZoom *gesture,
|
|
GdkEventSequence *sequence,
|
|
GimpContainerBox *box);
|
|
static void gimp_container_box_zoom_gesture_update (GtkGestureZoom *gesture,
|
|
GdkEventSequence *sequence,
|
|
GimpContainerBox *box);
|
|
|
|
|
|
G_DEFINE_TYPE_WITH_CODE (GimpContainerBox, gimp_container_box,
|
|
GIMP_TYPE_EDITOR,
|
|
G_ADD_PRIVATE (GimpContainerBox)
|
|
G_IMPLEMENT_INTERFACE (GIMP_TYPE_CONTAINER_VIEW,
|
|
gimp_container_box_view_iface_init)
|
|
G_IMPLEMENT_INTERFACE (GIMP_TYPE_DOCKED,
|
|
gimp_container_box_docked_iface_init))
|
|
|
|
#define parent_class gimp_container_box_parent_class
|
|
|
|
|
|
static void
|
|
gimp_container_box_class_init (GimpContainerBoxClass *klass)
|
|
{
|
|
GObjectClass *object_class = G_OBJECT_CLASS (klass);
|
|
|
|
object_class->constructed = gimp_container_box_constructed;
|
|
object_class->finalize = gimp_container_box_finalize;
|
|
object_class->set_property = _gimp_container_view_set_property;
|
|
object_class->get_property = _gimp_container_view_get_property;
|
|
|
|
_gimp_container_view_install_properties (object_class);
|
|
}
|
|
|
|
static void
|
|
gimp_container_box_init (GimpContainerBox *box)
|
|
{
|
|
GimpContainerBoxPrivate *priv = GET_PRIVATE (box);
|
|
GtkWidget *sb;
|
|
|
|
box->scrolled_win = gtk_scrolled_window_new (NULL, NULL);
|
|
gtk_box_pack_start (GTK_BOX (box), box->scrolled_win, TRUE, TRUE, 0);
|
|
gtk_widget_show (box->scrolled_win);
|
|
|
|
gtk_widget_add_events (box->scrolled_win, GDK_TOUCHPAD_GESTURE_MASK);
|
|
|
|
g_signal_connect (box->scrolled_win, "scroll-event",
|
|
G_CALLBACK (gimp_container_box_scroll_to_zoom),
|
|
box);
|
|
|
|
sb = gtk_scrolled_window_get_vscrollbar (GTK_SCROLLED_WINDOW (box->scrolled_win));
|
|
gtk_widget_set_can_focus (sb, FALSE);
|
|
|
|
priv->zoom_gesture = gtk_gesture_zoom_new (GTK_WIDGET (box->scrolled_win));
|
|
gtk_event_controller_set_propagation_phase (GTK_EVENT_CONTROLLER (priv->zoom_gesture),
|
|
GTK_PHASE_CAPTURE);
|
|
|
|
/* The default signal handler needs to run first to setup scale delta */
|
|
g_signal_connect_after (priv->zoom_gesture, "begin",
|
|
G_CALLBACK (gimp_container_box_zoom_gesture_begin),
|
|
box);
|
|
g_signal_connect_after (priv->zoom_gesture, "update",
|
|
G_CALLBACK (gimp_container_box_zoom_gesture_update),
|
|
box);
|
|
}
|
|
|
|
static void
|
|
gimp_container_box_view_iface_init (GimpContainerViewInterface *iface)
|
|
{
|
|
}
|
|
|
|
static void
|
|
gimp_container_box_docked_iface_init (GimpDockedInterface *iface)
|
|
{
|
|
iface->get_preview = gimp_container_box_get_preview;
|
|
iface->set_context = gimp_container_box_set_context;
|
|
}
|
|
|
|
static void
|
|
gimp_container_box_constructed (GObject *object)
|
|
{
|
|
GimpContainerBox *box = GIMP_CONTAINER_BOX (object);
|
|
|
|
/* This is evil: the hash table of "insert_data" is created on
|
|
* demand when GimpContainerView API is used, using a
|
|
* value_free_func that is set in the interface_init functions of
|
|
* its implementors. Therefore, no GimpContainerView API must be
|
|
* called from any init() function, because the interface_init()
|
|
* function of a subclass that sets the right value_free_func might
|
|
* not have been called yet, leaving the insert_data hash table
|
|
* without memory management.
|
|
*
|
|
* Call GimpContainerView API from GObject::constructed() instead,
|
|
* which runs after everything is set up correctly.
|
|
*/
|
|
gimp_container_view_set_dnd_widget (GIMP_CONTAINER_VIEW (box),
|
|
box->scrolled_win);
|
|
|
|
G_OBJECT_CLASS (parent_class)->constructed (object);
|
|
}
|
|
|
|
static void
|
|
gimp_container_box_finalize (GObject *object)
|
|
{
|
|
GimpContainerBoxPrivate *priv = GET_PRIVATE (object);
|
|
|
|
g_clear_object (&priv->zoom_gesture);
|
|
|
|
G_OBJECT_CLASS (parent_class)->finalize (object);
|
|
}
|
|
|
|
void
|
|
gimp_container_box_set_size_request (GimpContainerBox *box,
|
|
gint width,
|
|
gint height)
|
|
{
|
|
GimpContainerView *view;
|
|
GtkScrolledWindowClass *sw_class;
|
|
GtkStyleContext *sw_style;
|
|
GtkWidget *sb;
|
|
GtkRequisition req;
|
|
GtkBorder border;
|
|
gint view_size;
|
|
gint scrollbar_width;
|
|
gint border_x;
|
|
gint border_y;
|
|
|
|
g_return_if_fail (GIMP_IS_CONTAINER_BOX (box));
|
|
|
|
view = GIMP_CONTAINER_VIEW (box);
|
|
|
|
view_size = gimp_container_view_get_view_size (view, NULL);
|
|
|
|
g_return_if_fail (width <= 0 || width >= view_size);
|
|
g_return_if_fail (height <= 0 || height >= view_size);
|
|
|
|
sw_class = GTK_SCROLLED_WINDOW_GET_CLASS (box->scrolled_win);
|
|
|
|
if (sw_class->scrollbar_spacing >= 0)
|
|
scrollbar_width = sw_class->scrollbar_spacing;
|
|
else
|
|
gtk_widget_style_get (GTK_WIDGET (box->scrolled_win),
|
|
"scrollbar-spacing", &scrollbar_width,
|
|
NULL);
|
|
|
|
sb = gtk_scrolled_window_get_vscrollbar (GTK_SCROLLED_WINDOW (box->scrolled_win));
|
|
|
|
gtk_widget_get_preferred_size (sb, &req, NULL);
|
|
scrollbar_width += req.width;
|
|
|
|
border_x = border_y = gtk_container_get_border_width (GTK_CONTAINER (box));
|
|
|
|
sw_style = gtk_widget_get_style_context (box->scrolled_win);
|
|
|
|
gtk_style_context_get_border (sw_style, gtk_widget_get_state_flags (box->scrolled_win), &border);
|
|
|
|
border_x += border.left + border.right + scrollbar_width;
|
|
border_y += border.top + border.bottom;
|
|
|
|
gtk_widget_set_size_request (box->scrolled_win,
|
|
width > 0 ? width + border_x : -1,
|
|
height > 0 ? height + border_y : -1);
|
|
}
|
|
|
|
static void
|
|
gimp_container_box_set_context (GimpDocked *docked,
|
|
GimpContext *context)
|
|
{
|
|
gimp_container_view_set_context (GIMP_CONTAINER_VIEW (docked), context);
|
|
}
|
|
|
|
static GtkWidget *
|
|
gimp_container_box_get_preview (GimpDocked *docked,
|
|
GimpContext *context,
|
|
GtkIconSize size)
|
|
{
|
|
GimpContainerView *view = GIMP_CONTAINER_VIEW (docked);
|
|
GimpContainer *container;
|
|
GtkWidget *preview;
|
|
gint width;
|
|
gint height;
|
|
gint border_width = 1;
|
|
const gchar *prop_name;
|
|
|
|
container = gimp_container_view_get_container (view);
|
|
|
|
g_return_val_if_fail (container != NULL, NULL);
|
|
|
|
gtk_icon_size_lookup (size, &width, &height);
|
|
|
|
prop_name =
|
|
gimp_context_type_to_prop_name (gimp_container_get_child_type (container));
|
|
|
|
preview = gimp_prop_view_new (G_OBJECT (context), prop_name,
|
|
context, height);
|
|
GIMP_VIEW (preview)->renderer->size = -1;
|
|
|
|
gimp_container_view_get_view_size (view, &border_width);
|
|
|
|
border_width = MIN (1, border_width);
|
|
|
|
gimp_view_renderer_set_size_full (GIMP_VIEW (preview)->renderer,
|
|
width, height, border_width);
|
|
|
|
return preview;
|
|
}
|
|
|
|
/* We want to zoom on each 1/4 scroll events to roughly match zooming
|
|
* behavior of the main canvas. Each step of GIMP_VIEW_SIZE_* is
|
|
* approximately of sqrt(2) = 1.4 relative change. The main canvas
|
|
* on the other hand has zoom step equal to ZOOM_MIN_STEP=1.1
|
|
* (see gimp_zoom_model_zoom_step).
|
|
*/
|
|
#define SCROLL_ZOOM_STEP_SIZE 0.25
|
|
|
|
static gboolean
|
|
gimp_container_box_scroll_to_zoom (GtkWidget *widget,
|
|
GdkEventScroll *event,
|
|
GimpContainerBox *box)
|
|
{
|
|
GimpContainerBoxPrivate *priv = GET_PRIVATE (box);
|
|
GimpContainerView *view = GIMP_CONTAINER_VIEW (box);
|
|
gint view_border_width;
|
|
gint view_size;
|
|
|
|
if ((event->state & GDK_CONTROL_MASK) == 0)
|
|
return FALSE;
|
|
|
|
if (event->direction == GDK_SCROLL_UP)
|
|
priv->zoom_accum_scroll_delta -= SCROLL_ZOOM_STEP_SIZE;
|
|
else if (event->direction == GDK_SCROLL_DOWN)
|
|
priv->zoom_accum_scroll_delta += SCROLL_ZOOM_STEP_SIZE;
|
|
else if (event->direction == GDK_SCROLL_SMOOTH)
|
|
priv->zoom_accum_scroll_delta += event->delta_y * SCROLL_ZOOM_STEP_SIZE;
|
|
else
|
|
return FALSE;
|
|
|
|
view_size = gimp_container_view_get_view_size (view, &view_border_width);
|
|
|
|
if (priv->zoom_accum_scroll_delta > 1)
|
|
{
|
|
priv->zoom_accum_scroll_delta -= 1;
|
|
|
|
view_size = gimp_view_size_get_smaller (view_size);
|
|
}
|
|
else if (priv->zoom_accum_scroll_delta < -1)
|
|
{
|
|
priv->zoom_accum_scroll_delta += 1;
|
|
|
|
view_size = gimp_view_size_get_larger (view_size);
|
|
}
|
|
else
|
|
{
|
|
return TRUE;
|
|
}
|
|
|
|
gimp_container_view_set_view_size (view, view_size, view_border_width);
|
|
|
|
return TRUE;
|
|
}
|
|
|
|
#define ZOOM_GESTURE_STEP_SIZE 1.2
|
|
|
|
static void
|
|
gimp_container_box_zoom_gesture_begin (GtkGestureZoom *gesture,
|
|
GdkEventSequence *sequence,
|
|
GimpContainerBox *box)
|
|
{
|
|
GimpContainerBoxPrivate *priv = GET_PRIVATE (box);
|
|
|
|
priv->zoom_last_value = gtk_gesture_zoom_get_scale_delta (gesture);
|
|
}
|
|
|
|
static void
|
|
gimp_container_box_zoom_gesture_update (GtkGestureZoom *gesture,
|
|
GdkEventSequence *sequence,
|
|
GimpContainerBox *box)
|
|
{
|
|
GimpContainerBoxPrivate *priv = GET_PRIVATE (box);
|
|
GimpContainerView *view = GIMP_CONTAINER_VIEW (box);
|
|
gdouble current_scale;
|
|
gdouble last_value;
|
|
gdouble min_increase;
|
|
gdouble max_decrease;
|
|
gint view_border_width;
|
|
gint view_size;
|
|
|
|
last_value = priv->zoom_last_value;
|
|
min_increase = last_value * ZOOM_GESTURE_STEP_SIZE;
|
|
max_decrease = last_value / ZOOM_GESTURE_STEP_SIZE;
|
|
|
|
view_size = gimp_container_view_get_view_size (view, &view_border_width);
|
|
|
|
current_scale = gtk_gesture_zoom_get_scale_delta (gesture);
|
|
|
|
if (current_scale > min_increase)
|
|
{
|
|
last_value = min_increase;
|
|
view_size = gimp_view_size_get_larger (view_size);
|
|
}
|
|
else if (current_scale < max_decrease)
|
|
{
|
|
last_value = max_decrease;
|
|
view_size = gimp_view_size_get_smaller (view_size);
|
|
}
|
|
else
|
|
{
|
|
return;
|
|
}
|
|
|
|
priv->zoom_last_value = last_value;
|
|
|
|
gimp_container_view_set_view_size (view, view_size, view_border_width);
|
|
}
|