PIKApp/app/widgets/pikamessagebox.c

550 lines
18 KiB
C

/* PIKA - Photo and Image Kooker Application
* a rebranding of The GNU Image Manipulation Program (created with heckimp)
* A derived work which may be trivial. However, any changes may be (C)2023 by Aldercone Studio
*
* Original copyright, applying to most contents (license remains unchanged):
* Copyright (C) 1995 Spencer Kimball and Peter Mattis
*
* pikamessagebox.c
* Copyright (C) 2004 Sven Neumann <sven@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 <gegl.h>
#include <gtk/gtk.h>
#include "libpikabase/pikabase.h"
#include "libpikawidgets/pikawidgets.h"
#include "widgets-types.h"
#include "pikamessagebox.h"
#include "pika-intl.h"
#define PIKA_MESSAGE_BOX_SPACING 12
enum
{
PROP_0,
PROP_ICON_NAME
};
static void pika_message_box_constructed (GObject *object);
static void pika_message_box_dispose (GObject *object);
static void pika_message_box_finalize (GObject *object);
static void pika_message_box_set_property (GObject *object,
guint property_id,
const GValue *value,
GParamSpec *pspec);
static void pika_message_box_get_property (GObject *object,
guint property_id,
GValue *value,
GParamSpec *pspec);
static void pika_message_box_get_preferred_width (GtkWidget *widget,
gint *minimum_width,
gint *natural_width);
static void pika_message_box_get_preferred_height (GtkWidget *widget,
gint *minimum_height,
gint *natural_height);
static void pika_message_box_get_preferred_width_for_height
(GtkWidget *widget,
gint height,
gint *minimum_width,
gint *natural_width);
static void pika_message_box_get_preferred_height_for_width
(GtkWidget *widget,
gint width,
gint *minimum_height,
gint *natural_height);
static void pika_message_box_size_allocate (GtkWidget *widget,
GtkAllocation *allocation);
static void pika_message_box_forall (GtkContainer *container,
gboolean include_internals,
GtkCallback callback,
gpointer callback_data);
static void pika_message_box_set_label_text (PikaMessageBox *box,
gint n,
const gchar *format,
va_list args) G_GNUC_PRINTF (3, 0);
static void pika_message_box_set_label_markup (PikaMessageBox *box,
gint n,
const gchar *format,
va_list args) G_GNUC_PRINTF (3, 0);
static gboolean pika_message_box_update (gpointer data);
G_DEFINE_TYPE (PikaMessageBox, pika_message_box, GTK_TYPE_BOX)
#define parent_class pika_message_box_parent_class
static void
pika_message_box_class_init (PikaMessageBoxClass *klass)
{
GObjectClass *object_class = G_OBJECT_CLASS (klass);
GtkWidgetClass *widget_class = GTK_WIDGET_CLASS (klass);
GtkContainerClass *container_class = GTK_CONTAINER_CLASS (klass);
object_class->constructed = pika_message_box_constructed;
object_class->dispose = pika_message_box_dispose;
object_class->finalize = pika_message_box_finalize;
object_class->set_property = pika_message_box_set_property;
object_class->get_property = pika_message_box_get_property;
widget_class->get_preferred_width = pika_message_box_get_preferred_width;
widget_class->get_preferred_height = pika_message_box_get_preferred_height;
widget_class->get_preferred_width_for_height = pika_message_box_get_preferred_width_for_height;
widget_class->get_preferred_height_for_width = pika_message_box_get_preferred_height_for_width;
widget_class->size_allocate = pika_message_box_size_allocate;
container_class->forall = pika_message_box_forall;
gtk_container_class_handle_border_width (container_class);
g_object_class_install_property (object_class, PROP_ICON_NAME,
g_param_spec_string ("icon-name", NULL, NULL,
NULL,
PIKA_PARAM_READWRITE |
G_PARAM_CONSTRUCT_ONLY));
}
static void
pika_message_box_init (PikaMessageBox *box)
{
gint i;
gtk_orientable_set_orientation (GTK_ORIENTABLE (box),
GTK_ORIENTATION_VERTICAL);
gtk_box_set_spacing (GTK_BOX (box), 12);
gtk_container_set_border_width (GTK_CONTAINER (box), 12);
for (i = 0; i < 2; i++)
{
GtkWidget *label = g_object_new (GTK_TYPE_LABEL,
"wrap", TRUE,
"wrap-mode", PANGO_WRAP_WORD_CHAR,
"max-width-chars", 80,
"selectable", TRUE,
"xalign", 0.0,
"yalign", 0.5,
NULL);
if (i == 0)
pika_label_set_attributes (GTK_LABEL (label),
PANGO_ATTR_SCALE, PANGO_SCALE_LARGE,
PANGO_ATTR_WEIGHT, PANGO_WEIGHT_BOLD,
-1);
gtk_box_pack_start (GTK_BOX (box), label, FALSE, FALSE, 0);
box->label[i] = label;
}
box->repeat = 0;
box->label[2] = NULL;
box->idle_id = 0;
}
static void
pika_message_box_constructed (GObject *object)
{
PikaMessageBox *box = PIKA_MESSAGE_BOX (object);
G_OBJECT_CLASS (parent_class)->constructed (object);
if (box->icon_name)
{
box->image = gtk_image_new_from_icon_name (box->icon_name,
GTK_ICON_SIZE_DIALOG);
gtk_widget_set_halign (box->image, GTK_ALIGN_START);
gtk_widget_set_valign (box->image, GTK_ALIGN_START);
gtk_widget_set_parent (box->image, GTK_WIDGET (box));
gtk_widget_show (box->image);
}
}
static void
pika_message_box_dispose (GObject *object)
{
PikaMessageBox *box = PIKA_MESSAGE_BOX (object);
if (box->image)
{
gtk_widget_unparent (box->image);
box->image = NULL;
}
G_OBJECT_CLASS (parent_class)->dispose (object);
}
static void
pika_message_box_finalize (GObject *object)
{
PikaMessageBox *box = PIKA_MESSAGE_BOX (object);
if (box->idle_id)
{
g_source_remove (box->idle_id);
box->idle_id = 0;
}
g_clear_pointer (&box->icon_name, g_free);
G_OBJECT_CLASS (parent_class)->finalize (object);
}
static void
pika_message_box_set_property (GObject *object,
guint property_id,
const GValue *value,
GParamSpec *pspec)
{
PikaMessageBox *box = PIKA_MESSAGE_BOX (object);
switch (property_id)
{
case PROP_ICON_NAME:
box->icon_name = g_value_dup_string (value);
break;
default:
G_OBJECT_WARN_INVALID_PROPERTY_ID (object, property_id, pspec);
break;
}
}
static void
pika_message_box_get_property (GObject *object,
guint property_id,
GValue *value,
GParamSpec *pspec)
{
PikaMessageBox *box = PIKA_MESSAGE_BOX (object);
switch (property_id)
{
case PROP_ICON_NAME:
g_value_set_string (value, box->icon_name);
break;
default:
G_OBJECT_WARN_INVALID_PROPERTY_ID (object, property_id, pspec);
break;
}
}
static void
pika_message_box_get_preferred_width (GtkWidget *widget,
gint *minimum_width,
gint *natural_width)
{
PikaMessageBox *box = PIKA_MESSAGE_BOX (widget);
GTK_WIDGET_CLASS (parent_class)->get_preferred_width (widget,
minimum_width,
natural_width);
if (box->image && gtk_widget_get_visible (box->image))
{
gint image_minimum;
gint image_natural;
gtk_widget_get_preferred_width (box->image,
&image_minimum, &image_natural);
*minimum_width += image_minimum + PIKA_MESSAGE_BOX_SPACING;
*natural_width += image_natural + PIKA_MESSAGE_BOX_SPACING;
}
}
static void
pika_message_box_get_preferred_height (GtkWidget *widget,
gint *minimum_height,
gint *natural_height)
{
PikaMessageBox *box = PIKA_MESSAGE_BOX (widget);
GTK_WIDGET_CLASS (parent_class)->get_preferred_height (widget,
minimum_height,
natural_height);
if (box->image && gtk_widget_get_visible (box->image))
{
gint image_minimum;
gint image_natural;
gtk_widget_get_preferred_height (box->image,
&image_minimum, &image_natural);
*minimum_height = MAX (*minimum_height, image_minimum);
*natural_height = MAX (*natural_height, image_natural);
}
}
static void
pika_message_box_get_preferred_width_for_height (GtkWidget *widget,
gint height,
gint *minimum_width,
gint *natural_width)
{
GTK_WIDGET_GET_CLASS (widget)->get_preferred_width (widget,
minimum_width,
natural_width);
}
static void
pika_message_box_get_preferred_height_for_width (GtkWidget *widget,
gint width,
gint *minimum_height,
gint *natural_height)
{
GTK_WIDGET_GET_CLASS (widget)->get_preferred_height (widget,
minimum_height,
natural_height);
}
static void
pika_message_box_size_allocate (GtkWidget *widget,
GtkAllocation *allocation)
{
PikaMessageBox *box = PIKA_MESSAGE_BOX (widget);
gint width = 0;
gboolean rtl;
rtl = (gtk_widget_get_direction (widget) == GTK_TEXT_DIR_RTL);
if (box->image && gtk_widget_get_visible (box->image))
{
GtkRequisition child_requisition;
GtkAllocation child_allocation;
gint height;
gtk_widget_get_preferred_size (box->image, &child_requisition, NULL);
width = MIN (allocation->width,
child_requisition.width + PIKA_MESSAGE_BOX_SPACING);
width = MAX (1, width);
height = allocation->height;
if (rtl)
child_allocation.x = allocation->width - child_requisition.width;
else
child_allocation.x = allocation->x;
child_allocation.y = allocation->y;
child_allocation.width = width;
child_allocation.height = height;
gtk_widget_size_allocate (box->image, &child_allocation);
}
allocation->x += rtl ? 0 : width;
allocation->width -= width;
GTK_WIDGET_CLASS (parent_class)->size_allocate (widget, allocation);
allocation->x -= rtl ? 0 : width;
allocation->width += width;
gtk_widget_set_allocation (widget, allocation);
}
static void
pika_message_box_forall (GtkContainer *container,
gboolean include_internals,
GtkCallback callback,
gpointer callback_data)
{
if (include_internals)
{
PikaMessageBox *box = PIKA_MESSAGE_BOX (container);
if (box->image)
(* callback) (box->image, callback_data);
}
GTK_CONTAINER_CLASS (parent_class)->forall (container, include_internals,
callback, callback_data);
}
static void
pika_message_box_set_label_text (PikaMessageBox *box,
gint n,
const gchar *format,
va_list args)
{
GtkWidget *label = box->label[n];
if (format)
{
gchar *text = g_strdup_vprintf (format, args);
gchar *utf8 = pika_any_to_utf8 (text, -1, "Cannot convert text to utf8.");
gtk_label_set_text (GTK_LABEL (label), utf8);
gtk_widget_show (label);
g_free (utf8);
g_free (text);
}
else
{
gtk_widget_hide (label);
gtk_label_set_text (GTK_LABEL (label), NULL);
}
}
static void
pika_message_box_set_label_markup (PikaMessageBox *box,
gint n,
const gchar *format,
va_list args)
{
GtkWidget *label = box->label[n];
if (format)
{
gchar *text = g_markup_vprintf_escaped (format, args);
gtk_label_set_markup (GTK_LABEL (label), text);
gtk_widget_show (label);
g_free (text);
}
else
{
gtk_widget_hide (label);
gtk_label_set_text (GTK_LABEL (label), NULL);
}
}
static gboolean
pika_message_box_update (gpointer data)
{
PikaMessageBox *box = data;
gchar *message;
box->idle_id = 0;
message = g_strdup_printf (ngettext ("Message repeated once.",
"Message repeated %d times.",
box->repeat),
box->repeat);
if (box->label[2])
{
gtk_label_set_text (GTK_LABEL (box->label[2]), message);
}
else
{
GtkWidget *label = box->label[2] = gtk_label_new (message);
gtk_label_set_xalign (GTK_LABEL (label), 0.0);
pika_label_set_attributes (GTK_LABEL (label),
PANGO_ATTR_STYLE, PANGO_STYLE_OBLIQUE,
-1);
gtk_box_pack_end (GTK_BOX (box), label, FALSE, FALSE, 0);
gtk_widget_show (label);
}
g_free (message);
return G_SOURCE_REMOVE;
}
/* public functions */
GtkWidget *
pika_message_box_new (const gchar *icon_name)
{
return g_object_new (PIKA_TYPE_MESSAGE_BOX,
"icon-name", icon_name,
NULL);
}
void
pika_message_box_set_primary_text (PikaMessageBox *box,
const gchar *format,
...)
{
va_list args;
g_return_if_fail (PIKA_IS_MESSAGE_BOX (box));
va_start (args, format);
pika_message_box_set_label_text (box, 0, format, args);
va_end (args);
}
void
pika_message_box_set_text (PikaMessageBox *box,
const gchar *format,
...)
{
va_list args;
g_return_if_fail (PIKA_IS_MESSAGE_BOX (box));
va_start (args, format);
pika_message_box_set_label_text (box, 1, format, args);
va_end (args);
}
void
pika_message_box_set_markup (PikaMessageBox *box,
const gchar *format,
...)
{
va_list args;
g_return_if_fail (PIKA_IS_MESSAGE_BOX (box));
va_start (args, format);
pika_message_box_set_label_markup (box, 1,format, args);
va_end (args);
}
gint
pika_message_box_repeat (PikaMessageBox *box)
{
g_return_val_if_fail (PIKA_IS_MESSAGE_BOX (box), 0);
box->repeat++;
if (box->idle_id == 0)
{
/* When a same message is repeated dozens of thousands of times in
* a short span of time, updating the GUI at each increment is
* extremely slow (like really really slow, your GUI gets stuck
* for 10 minutes). So let's just delay GUI update as a low
* priority idle task.
*/
box->idle_id = g_idle_add_full (G_PRIORITY_LOW,
pika_message_box_update,
box, NULL);
}
return box->repeat;
}