forked from AuroraMiddleware/gtk
345 lines
14 KiB
C
345 lines
14 KiB
C
/*
|
|
* Copyright © 2015 Red Hat Inc.
|
|
*
|
|
* This library is free software; you can redistribute it and/or
|
|
* modify it under the terms of the GNU Lesser General Public
|
|
* License as published by the Free Software Foundation; either
|
|
* version 2.1 of the License, or (at your option) any later version.
|
|
*
|
|
* This library 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
|
|
* Lesser General Public License for more details.
|
|
*
|
|
* You should have received a copy of the GNU Lesser General Public
|
|
* License along with this library. If not, see <http://www.gnu.org/licenses/>.
|
|
*
|
|
* Authors: Benjamin Otte <otte@gnome.org>
|
|
*/
|
|
|
|
#include "config.h"
|
|
|
|
#include "gtkcsscustomgadgetprivate.h"
|
|
|
|
#include "gtkcssnodeprivate.h"
|
|
|
|
/*
|
|
* GtkCssCustomGadget is a subclass that lets widgets customize size
|
|
* requests, size allocation and drawing of gadgets. The gadget is passed
|
|
* to the callbacks as the first argument, and you can use gtk_css_gadget_get_owner()
|
|
* to obtain the widget. Note that the widgets style context is not saved,
|
|
* so if you want to query style properties or call gtk_render functions which
|
|
* take the style context as an argument, you should use
|
|
* gtk_style_context_save_to_node to make the gadget's CSS node take effect.
|
|
*
|
|
* The callbacks are
|
|
*
|
|
* GtkCssPreferredSizeFunc:
|
|
* @gadget: the #GtkCssCustomGadget
|
|
* @orientation: whether a width (ie horizontal) or height (ie vertical) is requested
|
|
* @for_size: the available size in the opposite direction, or -1
|
|
* @minimum: return location for the minimum size
|
|
* @natural: return location for the natural size
|
|
* @minimum_baseline: (nullable): return location for the baseline at minimum size
|
|
* @natural_baseline: (nullable): return location for the baseline at natural size
|
|
* @data: data provided when registering the callback
|
|
*
|
|
* The GtkCssPreferredSizeFunc is called to determine the content size in
|
|
* gtk_css_gadget_get_preferred_size(). @for_size is a content size (ie excluding
|
|
* CSS padding, border and margin) and the returned @minimum, @natural,
|
|
* @minimum_baseline, @natural_baseline should be content sizes excluding CSS
|
|
* padding, border and margin as well.
|
|
*
|
|
* Typically, GtkCssPreferredSizeFunc will query the size of sub-gadgets and
|
|
* child widgets that are placed relative to the gadget and determine its own
|
|
* needed size from the results. If the gadget has no sub-gadgets or child
|
|
* widgets that it needs to place, then a GtkCssPreferredSizeFunc is only
|
|
* needed if you want to enforce a minimum size independent of CSS min-width
|
|
* and min-height (e.g. if size-related style properties need to be supported
|
|
* for compatibility).
|
|
*
|
|
* GtkCssAllocateFunc:
|
|
* @gadget: the #GtkCssCustomGadget
|
|
* @allocation: the allocation
|
|
* @baseline: the baseline
|
|
* @out_clip: (out): return location for the content clip
|
|
* @data: data provided when registering the callback
|
|
*
|
|
* The GtkCssAllocateFunc is called to allocate the gadgets content in
|
|
* gtk_css_gadget_allocate(). @allocation and @baseline are content sizes
|
|
* (ie excluding CSS padding, border and margin).
|
|
*
|
|
* Typically, GtkCssAllocateFunc will allocate sub-gadgets and child widgets
|
|
* that are placed relative to the gadget, and merge their clips into the
|
|
* value returned as @out_clip. For clip handling in the main gadget of
|
|
* containers, gtk_container_get_children_clip() can be useful. Gadgets that
|
|
* don't have sub-gadgets of child widgets don't need a GtkCssAllocateFunc
|
|
* (although it is still required to call gtk_css_gadget_allocate() on them).
|
|
*
|
|
* Note that @out_clip *must* be set to meaningful values. If in doubt,
|
|
* just set it to the allocation.
|
|
*
|
|
* GtkCssDrawFunc:
|
|
* @gadget: the #GtkCssCustomGadget
|
|
* @cr: the cairo context to draw on
|
|
* @x: the x origin of the content area
|
|
* @y: the y origin of the content area
|
|
* @width: the width of the content area
|
|
* @height: the height of the content area
|
|
* @data: data provided when registering the callback
|
|
*
|
|
* The GtkCssDrawFunc is called to draw the gadgets content in
|
|
* gtk_css_gadget_draw(). It gets passed an untransformed cairo context
|
|
* and the coordinates of the area to draw the content in.
|
|
*
|
|
* Typically, GtkCssDrawFunc will draw sub-gadgets and child widgets
|
|
* that are placed relative to the gadget, as well as custom content
|
|
* such as icons, checkmarks, arrows or text.
|
|
*
|
|
* GtkCssSnapshotFunc:
|
|
* @gadget: the #GtkCssCustomGadget
|
|
* @snapshot: the snapshot to snapshot to
|
|
* @x: the x origin of the content area
|
|
* @y: the y origin of the content area
|
|
* @width: the width of the content area
|
|
* @height: the height of the content area
|
|
* @data: data provided when registering the callback
|
|
*
|
|
* The GtkCssSnapshotFunc is called to snapshot the gadget's content in
|
|
* gtk_css_gadget_snapshot(). It gets passed an untransformed cairo context
|
|
* and the coordinates of the area to draw the content in.
|
|
*
|
|
* Typically, GtkCssSnapshotFunc will draw sub-gadgets and child widgets
|
|
* that are placed relative to the gadget, as well as custom content
|
|
* such as icons, checkmarks, arrows or text.
|
|
*/
|
|
|
|
typedef struct _GtkCssCustomGadgetPrivate GtkCssCustomGadgetPrivate;
|
|
struct _GtkCssCustomGadgetPrivate {
|
|
GtkCssPreferredSizeFunc preferred_size_func;
|
|
GtkCssAllocateFunc allocate_func;
|
|
GtkCssDrawFunc draw_func;
|
|
GtkCssSnapshotFunc snapshot_func;
|
|
gpointer data;
|
|
GDestroyNotify destroy_func;
|
|
};
|
|
|
|
G_DEFINE_TYPE_WITH_CODE (GtkCssCustomGadget, gtk_css_custom_gadget, GTK_TYPE_CSS_GADGET,
|
|
G_ADD_PRIVATE (GtkCssCustomGadget))
|
|
|
|
static void
|
|
gtk_css_custom_gadget_get_preferred_size (GtkCssGadget *gadget,
|
|
GtkOrientation orientation,
|
|
gint for_size,
|
|
gint *minimum,
|
|
gint *natural,
|
|
gint *minimum_baseline,
|
|
gint *natural_baseline)
|
|
{
|
|
GtkCssCustomGadgetPrivate *priv = gtk_css_custom_gadget_get_instance_private (GTK_CSS_CUSTOM_GADGET (gadget));
|
|
|
|
if (priv->preferred_size_func)
|
|
return priv->preferred_size_func (gadget, orientation, for_size,
|
|
minimum, natural,
|
|
minimum_baseline, natural_baseline,
|
|
priv->data);
|
|
else
|
|
return GTK_CSS_GADGET_CLASS (gtk_css_custom_gadget_parent_class)->get_preferred_size (gadget, orientation, for_size,
|
|
minimum, natural,
|
|
minimum_baseline, natural_baseline);
|
|
}
|
|
|
|
static void
|
|
gtk_css_custom_gadget_allocate (GtkCssGadget *gadget,
|
|
const GtkAllocation *allocation,
|
|
int baseline,
|
|
GtkAllocation *out_clip)
|
|
{
|
|
GtkCssCustomGadgetPrivate *priv = gtk_css_custom_gadget_get_instance_private (GTK_CSS_CUSTOM_GADGET (gadget));
|
|
|
|
if (priv->allocate_func)
|
|
return priv->allocate_func (gadget, allocation, baseline, out_clip, priv->data);
|
|
else
|
|
return GTK_CSS_GADGET_CLASS (gtk_css_custom_gadget_parent_class)->allocate (gadget, allocation, baseline, out_clip);
|
|
}
|
|
|
|
static gboolean
|
|
gtk_css_custom_gadget_draw (GtkCssGadget *gadget,
|
|
cairo_t *cr,
|
|
int x,
|
|
int y,
|
|
int width,
|
|
int height)
|
|
{
|
|
GtkCssCustomGadgetPrivate *priv = gtk_css_custom_gadget_get_instance_private (GTK_CSS_CUSTOM_GADGET (gadget));
|
|
|
|
if (priv->draw_func)
|
|
return priv->draw_func (gadget, cr, x, y, width, height, priv->data);
|
|
else
|
|
return GTK_CSS_GADGET_CLASS (gtk_css_custom_gadget_parent_class)->draw (gadget, cr, x, y, width, height);
|
|
}
|
|
|
|
static gboolean
|
|
gtk_css_custom_gadget_snapshot (GtkCssGadget *gadget,
|
|
GtkSnapshot *snapshot,
|
|
int x,
|
|
int y,
|
|
int width,
|
|
int height)
|
|
{
|
|
GtkCssCustomGadgetPrivate *priv = gtk_css_custom_gadget_get_instance_private (GTK_CSS_CUSTOM_GADGET (gadget));
|
|
|
|
if (priv->snapshot_func)
|
|
return priv->snapshot_func (gadget, snapshot, x, y, width, height, priv->data);
|
|
else if (priv->draw_func)
|
|
return GTK_CSS_GADGET_CLASS (gtk_css_custom_gadget_parent_class)->snapshot (gadget, snapshot, x, y, width, height);
|
|
else
|
|
return FALSE;
|
|
}
|
|
|
|
static gboolean
|
|
gtk_css_custom_gadget_has_content (GtkCssGadget *gadget)
|
|
{
|
|
GtkCssCustomGadgetPrivate *priv = gtk_css_custom_gadget_get_instance_private (GTK_CSS_CUSTOM_GADGET (gadget));
|
|
|
|
return priv->draw_func != NULL;
|
|
}
|
|
|
|
static void
|
|
gtk_css_custom_gadget_finalize (GObject *object)
|
|
{
|
|
GtkCssCustomGadgetPrivate *priv = gtk_css_custom_gadget_get_instance_private (GTK_CSS_CUSTOM_GADGET (object));
|
|
|
|
if (priv->destroy_func)
|
|
priv->destroy_func (priv->data);
|
|
|
|
G_OBJECT_CLASS (gtk_css_custom_gadget_parent_class)->finalize (object);
|
|
}
|
|
|
|
static void
|
|
gtk_css_custom_gadget_class_init (GtkCssCustomGadgetClass *klass)
|
|
{
|
|
GtkCssGadgetClass *gadget_class = GTK_CSS_GADGET_CLASS (klass);
|
|
GObjectClass *object_class = G_OBJECT_CLASS (klass);
|
|
|
|
object_class->finalize = gtk_css_custom_gadget_finalize;
|
|
|
|
gadget_class->get_preferred_size = gtk_css_custom_gadget_get_preferred_size;
|
|
gadget_class->allocate = gtk_css_custom_gadget_allocate;
|
|
gadget_class->draw = gtk_css_custom_gadget_draw;
|
|
gadget_class->snapshot = gtk_css_custom_gadget_snapshot;
|
|
gadget_class->has_content = gtk_css_custom_gadget_has_content;
|
|
}
|
|
|
|
static void
|
|
gtk_css_custom_gadget_init (GtkCssCustomGadget *custom_gadget)
|
|
{
|
|
|
|
}
|
|
|
|
/**
|
|
* gtk_css_custom_gadget_new_for_node:
|
|
* @node: the #GtkCssNode to use for the gadget
|
|
* @owner: the widget that the gadget belongs to
|
|
* @preferred_size_func: (nullable): the GtkCssPreferredSizeFunc to use
|
|
* @allocate_func: (nullable): the GtkCssAllocateFunc to use
|
|
* @draw_func: (nullable): the GtkCssDrawFunc to use
|
|
* @snapshot_func: (nullable): the GtkCssSnapshotFunc to use
|
|
* @data: (nullable): user data to pass to the callbacks
|
|
* @destroy_func: (nullable): destroy notify for @data
|
|
*
|
|
* Creates a #GtkCssCustomGadget for an existing CSS node.
|
|
* This function is typically used in the widgets init function
|
|
* to create the main gadget for the widgets main CSS node (which
|
|
* is obtained with gtk_widget_get_css_node()), as well as other
|
|
* permanent sub-gadgets. Sub-gadgets that only exist sometimes
|
|
* (e.g. depending on widget properties) should be created and
|
|
* destroyed as needed. All gadgets should be destroyed in the
|
|
* finalize (or dispose) vfunc.
|
|
*
|
|
* Returns: (transfer full): the new gadget
|
|
*/
|
|
GtkCssGadget *
|
|
gtk_css_custom_gadget_new_for_node (GtkCssNode *node,
|
|
GtkWidget *owner,
|
|
GtkCssPreferredSizeFunc preferred_size_func,
|
|
GtkCssAllocateFunc allocate_func,
|
|
GtkCssDrawFunc draw_func,
|
|
GtkCssSnapshotFunc snapshot_func,
|
|
gpointer data,
|
|
GDestroyNotify destroy_func)
|
|
{
|
|
GtkCssCustomGadgetPrivate *priv;
|
|
GtkCssGadget *result;
|
|
|
|
result = g_object_new (GTK_TYPE_CSS_CUSTOM_GADGET,
|
|
"node", node,
|
|
"owner", owner,
|
|
NULL);
|
|
|
|
priv = gtk_css_custom_gadget_get_instance_private (GTK_CSS_CUSTOM_GADGET (result));
|
|
|
|
priv->preferred_size_func = preferred_size_func;
|
|
priv->allocate_func = allocate_func;
|
|
priv->draw_func = draw_func;
|
|
priv->snapshot_func = snapshot_func;
|
|
priv->data = data;
|
|
priv->destroy_func = destroy_func;
|
|
|
|
return result;
|
|
}
|
|
|
|
/**
|
|
* gtk_css_custom_gadget_new:
|
|
* @name: the name for the CSS node
|
|
* @owner: the widget that the gadget belongs to
|
|
* @parent: the gadget that has the parent CSS node
|
|
* @next_sibling: the gadget that has the sibling CSS node
|
|
* @preferred_size_func: (nullable): the GtkCssPreferredSizeFunc to use
|
|
* @allocate_func: (nullable): the GtkCssAllocateFunc to use
|
|
* @draw_func: (nullable): the GtkCssDrawFunc to use
|
|
* @snapshot_func: (nullable): the GtkCssSnapshotFunc to use
|
|
* @data: (nullable): user data to pass to the callbacks
|
|
* @destroy_func: (nullable): destroy notify for @data
|
|
*
|
|
* Creates a #GtkCssCustomGadget with a new CSS node which gets
|
|
* placed below the @parent's and before the @next_sibling's CSS node.
|
|
*
|
|
* Returns: (transfer full): the new gadget
|
|
*/
|
|
GtkCssGadget *
|
|
gtk_css_custom_gadget_new (const char *name,
|
|
GtkWidget *owner,
|
|
GtkCssGadget *parent,
|
|
GtkCssGadget *next_sibling,
|
|
GtkCssPreferredSizeFunc preferred_size_func,
|
|
GtkCssAllocateFunc allocate_func,
|
|
GtkCssDrawFunc draw_func,
|
|
GtkCssSnapshotFunc snapshot_func,
|
|
gpointer data,
|
|
GDestroyNotify destroy_func)
|
|
{
|
|
GtkCssNode *node;
|
|
GtkCssGadget *result;
|
|
|
|
node = gtk_css_node_new ();
|
|
gtk_css_node_set_name (node, g_intern_string (name));
|
|
if (parent)
|
|
gtk_css_node_insert_before (gtk_css_gadget_get_node (parent),
|
|
node,
|
|
next_sibling ? gtk_css_gadget_get_node (next_sibling) : NULL);
|
|
|
|
result = gtk_css_custom_gadget_new_for_node (node,
|
|
owner,
|
|
preferred_size_func,
|
|
allocate_func,
|
|
draw_func,
|
|
snapshot_func,
|
|
data,
|
|
destroy_func);
|
|
|
|
g_object_unref (node);
|
|
|
|
return result;
|
|
}
|