diff options
Diffstat (limited to 'src/nautilus-container-max-width.c')
-rw-r--r-- | src/nautilus-container-max-width.c | 301 |
1 files changed, 301 insertions, 0 deletions
diff --git a/src/nautilus-container-max-width.c b/src/nautilus-container-max-width.c new file mode 100644 index 0000000..ef6fb87 --- /dev/null +++ b/src/nautilus-container-max-width.c @@ -0,0 +1,301 @@ +#include "nautilus-container-max-width.h" + +#define DEFAULT_MAX_SIZE 120 + +struct _NautilusContainerMaxWidth +{ + GtkBin parent_instance; + guint max_width; + + gboolean width_maximized; + guint change_width_maximized_idle_id; +}; + +G_DEFINE_TYPE (NautilusContainerMaxWidth, nautilus_container_max_width, GTK_TYPE_BIN) + +enum +{ + PROP_0, + PROP_MAX_WIDTH, + PROP_WIDTH_MAXIMIZED, + N_PROPS +}; + +void +nautilus_container_max_width_set_max_width (NautilusContainerMaxWidth *self, + guint max_width) +{ + self->max_width = max_width; + gtk_widget_queue_allocate (GTK_WIDGET (self)); +} + +guint +nautilus_container_max_width_get_max_width (NautilusContainerMaxWidth *self) +{ + return self->max_width; +} + +gboolean +nautilus_container_max_width_get_width_maximized (NautilusContainerMaxWidth *self) +{ + return self->width_maximized; +} + +NautilusContainerMaxWidth * +nautilus_container_max_width_new (void) +{ + return g_object_new (NAUTILUS_TYPE_CONTAINER_MAX_WIDTH, NULL); +} + +static void +nautilus_container_max_width_finalize (GObject *object) +{ + NautilusContainerMaxWidth *self = NAUTILUS_CONTAINER_MAX_WIDTH (object); + + if (self->change_width_maximized_idle_id != 0) + { + g_source_remove (self->change_width_maximized_idle_id); + } + + G_OBJECT_CLASS (nautilus_container_max_width_parent_class)->finalize (object); +} + +static void +nautilus_container_max_width_get_property (GObject *object, + guint prop_id, + GValue *value, + GParamSpec *pspec) +{ + NautilusContainerMaxWidth *self = NAUTILUS_CONTAINER_MAX_WIDTH (object); + + switch (prop_id) + { + case PROP_MAX_WIDTH: + { + g_value_set_int (value, self->max_width); + } + break; + + case PROP_WIDTH_MAXIMIZED: + { + g_value_set_boolean (value, self->width_maximized); + } + break; + + default: + { + G_OBJECT_WARN_INVALID_PROPERTY_ID (object, prop_id, pspec); + } + break; + } +} + +static void +nautilus_container_max_width_set_property (GObject *object, + guint prop_id, + const GValue *value, + GParamSpec *pspec) +{ + NautilusContainerMaxWidth *self = NAUTILUS_CONTAINER_MAX_WIDTH (object); + + switch (prop_id) + { + case PROP_MAX_WIDTH: + { + nautilus_container_max_width_set_max_width (self, g_value_get_int (value)); + } + break; + + default: + { + G_OBJECT_WARN_INVALID_PROPERTY_ID (object, prop_id, pspec); + } + break; + } +} + +static void +get_preferred_width (GtkWidget *widget, + gint *minimum_size, + gint *natural_size) +{ + GtkWidget *child; + NautilusContainerMaxWidth *self; + GtkStyleContext *style_context; + GtkBorder padding; + + self = NAUTILUS_CONTAINER_MAX_WIDTH (widget); + child = gtk_bin_get_child (GTK_BIN (self)); + + *natural_size = 0; + *minimum_size = 0; + gtk_widget_get_preferred_width (child, minimum_size, natural_size); + + if (self->max_width != -1 && *minimum_size > self->max_width) + { + g_critical ("NautilusContainerMaxWidth's child requested %d while set maximum width is %d", + *minimum_size, self->max_width); + } + *natural_size = self->max_width == -1 ? *natural_size : + MAX (*minimum_size, MIN (self->max_width, *natural_size)); + + style_context = gtk_widget_get_style_context (widget); + gtk_style_context_get_padding (style_context, + gtk_widget_get_state_flags (widget), + &padding); + *minimum_size += padding.left + padding.right; + *natural_size += padding.left + padding.right; +} + +static void +get_preferred_height (GtkWidget *widget, + gint *minimum_size, + gint *natural_size) +{ + GtkWidget *child; + NautilusContainerMaxWidth *self; + gint minimum_width = 0; + gint natural_width = 0; + GtkStyleContext *style_context; + GtkBorder padding; + + self = NAUTILUS_CONTAINER_MAX_WIDTH (widget); + child = gtk_bin_get_child (GTK_BIN (self)); + + get_preferred_width (widget, &minimum_width, &natural_width); + natural_width = self->max_width == -1 ? natural_width : MIN (self->max_width, natural_width); + + gtk_widget_get_preferred_height_for_width (child, natural_width, minimum_size, natural_size); + + style_context = gtk_widget_get_style_context (widget); + gtk_style_context_get_padding (style_context, + gtk_widget_get_state_flags (widget), + &padding); + *minimum_size += padding.top + padding.bottom; + *natural_size += padding.top + padding.bottom; +} + +static void +get_preferred_height_for_width (GtkWidget *widget, + gint width, + gint *minimum_size, + gint *natural_size) +{ + get_preferred_height (widget, minimum_size, natural_size); +} + +static void +size_allocate (GtkWidget *widget, + GtkAllocation *allocation) +{ + GTK_WIDGET_CLASS (nautilus_container_max_width_parent_class)->size_allocate (widget, allocation); +} + +static void +get_preferred_width_for_height (GtkWidget *widget, + gint height, + gint *minimum_size, + gint *natural_size) +{ + get_preferred_width (widget, minimum_size, natural_size); +} + +static gboolean +change_width_maximized_idle_callback (gpointer userdata) +{ + NautilusContainerMaxWidth *self = userdata; + + self->change_width_maximized_idle_id = 0; + + self->width_maximized = !self->width_maximized; + g_object_notify (G_OBJECT (self), "width-maximized"); + + return G_SOURCE_REMOVE; +} + +static void +on_size_allocate (GtkWidget *widget, + GdkRectangle *allocation, + gpointer userdata) +{ + NautilusContainerMaxWidth *self = NAUTILUS_CONTAINER_MAX_WIDTH (widget); + gboolean is_width_maximized; + + is_width_maximized = self->max_width == -1 ? FALSE : allocation->width >= self->max_width; + + if (self->width_maximized != is_width_maximized) + { + /* The handlers of the "notify::width-maximized" signal may trigger + * a reallocation, which shouldn't happen at this point because we are + * still in size_allocate phase. So, change the property on idle*/ + if (self->change_width_maximized_idle_id == 0) + { + self->change_width_maximized_idle_id = g_idle_add (change_width_maximized_idle_callback, self); + } + } + else if (self->change_width_maximized_idle_id != 0) + { + /* This was going to change self->width_maximized, let's cancel it. */ + g_source_remove (self->change_width_maximized_idle_id); + self->change_width_maximized_idle_id = 0; + } +} + +static void +constructed (GObject *obj) +{ + NautilusContainerMaxWidth *self = NAUTILUS_CONTAINER_MAX_WIDTH (obj); + + G_OBJECT_CLASS (nautilus_container_max_width_parent_class)->constructed (obj); + + /* We want our parent to gives our preferred width */ + gtk_widget_set_halign (GTK_WIDGET (self), GTK_ALIGN_CENTER); + self->max_width = -1; + + /* We want to know when the container has grown to its max width */ + self->width_maximized = FALSE; + g_signal_connect (GTK_WIDGET (self), + "size-allocate", + G_CALLBACK (on_size_allocate), + NULL); +} + +static void +nautilus_container_max_width_class_init (NautilusContainerMaxWidthClass *klass) +{ + GObjectClass *object_class = G_OBJECT_CLASS (klass); + GtkWidgetClass *widget_class = GTK_WIDGET_CLASS (klass); + + object_class->finalize = nautilus_container_max_width_finalize; + object_class->get_property = nautilus_container_max_width_get_property; + object_class->set_property = nautilus_container_max_width_set_property; + object_class->constructed = constructed; + + widget_class->get_preferred_width = get_preferred_width; + widget_class->get_preferred_width_for_height = get_preferred_width_for_height; + widget_class->get_preferred_height = get_preferred_height; + widget_class->get_preferred_height_for_width = get_preferred_height_for_width; + widget_class->size_allocate = size_allocate; + + g_object_class_install_property (object_class, + PROP_MAX_WIDTH, + g_param_spec_int ("max-width", + "Max width", + "The max width of the container", + G_MININT, + G_MAXINT, + 0, + G_PARAM_READWRITE)); + + g_object_class_install_property (object_class, + PROP_WIDTH_MAXIMIZED, + g_param_spec_boolean ("width-maximized", + "Width maximized", + "Whether the container is at the max width", + FALSE, + G_PARAM_READABLE)); +} +static void +nautilus_container_max_width_init (NautilusContainerMaxWidth *self) +{ +} |