]> git.draconx.ca Git - aspectbin.git/blobdiff - aspectbin.c
Implement container's child_type method.
[aspectbin.git] / aspectbin.c
index 339c8aec916a7ca610d71b827c664c7d375f6821..7e9095fd70c87119dab63a03cc08437e50e6350c 100644 (file)
@@ -5,25 +5,32 @@ static void aspect_bin_size_request(GtkWidget *, GtkRequisition *);
 static void aspect_bin_size_allocate(GtkWidget *, GtkAllocation *);
 static void aspect_bin_remove(GtkContainer *, GtkWidget *);
 static void aspect_bin_forall(GtkContainer *, gboolean, GtkCallback, gpointer);
+static GType aspect_bin_child_type(GtkContainer *);
 
-G_DEFINE_TYPE(AspectBin, aspect_bin, GTK_TYPE_BIN)
+G_DEFINE_TYPE(AspectBin, aspect_bin, GTK_TYPE_CONTAINER)
 
-static void aspect_bin_init(AspectBin *ab)
+static void aspect_bin_init(AspectBin *abin)
 {
-       ab->child = NULL;
-       ab->ratio = 1;
+       GTK_WIDGET_SET_FLAGS(abin, GTK_NO_WINDOW);
+
+       abin->body      = NULL;
+       abin->side      = NULL;
+       abin->ratio     = 1;
+       abin->align     = 0;
+       abin->constrain = FALSE;
 }
 
-static void aspect_bin_class_init(AspectBinClass *abc)
+static void aspect_bin_class_init(AspectBinClass *class)
 {
-       GtkWidgetClass    *widget_class    = GTK_WIDGET_CLASS(abc);
-       GtkContainerClass *container_class = GTK_CONTAINER_CLASS(abc);
+       GtkWidgetClass    *widget_class    = GTK_WIDGET_CLASS(class);
+       GtkContainerClass *container_class = GTK_CONTAINER_CLASS(class);
 
        widget_class->size_request  = aspect_bin_size_request;
        widget_class->size_allocate = aspect_bin_size_allocate;
 
-       container_class->remove = aspect_bin_remove;
-       container_class->forall = aspect_bin_forall;
+       container_class->remove     = aspect_bin_remove;
+       container_class->forall     = aspect_bin_forall;
+       container_class->child_type = aspect_bin_child_type;
 }
 
 GtkWidget *aspect_bin_new(void)
@@ -35,11 +42,10 @@ static void aspect_bin_remove(GtkContainer *container, GtkWidget *child)
 {
        AspectBin *abin = ASPECT_BIN(container);
 
-       if (abin->child == child) {
-               aspect_bin_set_body_widget(abin, NULL, 1);
-       } else {
-               GTK_CONTAINER_CLASS(aspect_bin_parent_class)->remove
-                       (container, child);
+       if (abin->body == child) {
+               aspect_bin_set_body(abin, NULL, 1);
+       } else if (abin->side == child) {
+               aspect_bin_set_side(abin, NULL);
        }
 }
 
@@ -49,29 +55,34 @@ static void aspect_bin_forall(GtkContainer *container,
                               gpointer      callback_data)
 {
        AspectBin *abin = ASPECT_BIN(container);
-       GtkBin     *bin = GTK_BIN(container);
+       g_return_if_fail(callback != NULL);
+
+       if (abin->body)
+               callback(abin->body, callback_data);
+       if (abin->side)
+               callback(abin->side, callback_data);
+}
 
-       if (bin->child)
-               callback(bin->child, callback_data);
-       
-       if (abin->child)
-               callback(abin->child, callback_data);
+static GType aspect_bin_child_type(GtkContainer *container)
+{
+       if (!ASPECT_BIN(container)->body || !ASPECT_BIN(container)->side)
+               return GTK_TYPE_WIDGET;
+       return G_TYPE_NONE;
 }
 
 static void
 aspect_bin_size_request(GtkWidget *widget, GtkRequisition *requisition)
 {
        AspectBin *abin = ASPECT_BIN(widget);
-       GtkBin    *bin  = GTK_BIN(widget);
        GtkRequisition creq = {0}, areq = {0};
 
-       if (bin->child && GTK_WIDGET_VISIBLE(bin->child)) {
-               gtk_widget_size_request(bin->child, &creq);
+       if (abin->side && GTK_WIDGET_VISIBLE(abin->side)) {
+               gtk_widget_size_request(abin->side, &creq);
        }
 
-       if (abin->child && GTK_WIDGET_VISIBLE(abin->child)) {
+       if (abin->body && GTK_WIDGET_VISIBLE(abin->body)) {
                int wtmp, htmp;
-               gtk_widget_size_request(abin->child, &areq);
+               gtk_widget_size_request(abin->body, &areq);
                wtmp = areq.height * abin->ratio + 0.5;
                htmp = areq.width  / abin->ratio + 0.5;
 
@@ -90,11 +101,11 @@ static void
 aspect_bin_size_allocate(GtkWidget *widget, GtkAllocation *allocation)
 {
        AspectBin *abin = ASPECT_BIN(widget);
-       GtkBin    *bin  = GTK_BIN(widget);
        GtkRequisition creq = {0};
        GtkAllocation csize = {0}, asize = {0};
-       
-       if (abin->child && GTK_WIDGET_VISIBLE(abin->child)) {
+
+       /* First find the best fit for the body. */
+       if (abin->body && GTK_WIDGET_VISIBLE(abin->body)) {
                asize.height = allocation->height;
                asize.width  = asize.height * abin->ratio + 0.5;
 
@@ -104,49 +115,71 @@ aspect_bin_size_allocate(GtkWidget *widget, GtkAllocation *allocation)
                }
        }
 
-       if (bin->child && GTK_WIDGET_VISIBLE(bin->child)) {
-               gtk_widget_get_child_requisition(bin->child, &creq);
+       /* Now try to fit the side. */
+       if (abin->side && GTK_WIDGET_VISIBLE(abin->side)) {
+               gtk_widget_get_child_requisition(abin->side, &creq);
 
                if (allocation->width - asize.width < creq.width) {
+                       /* It didn't fit, squish the constrained guy. */
                        asize.width  = allocation->width - creq.width;
                        asize.height = asize.width / abin->ratio + 0.5;
                }
 
-               csize.width = allocation->width - asize.width;
-               csize.height = allocation->height;
+               csize.width  = allocation->width - asize.width;
+               csize.height = MIN(allocation->height, creq.height);
+               if (abin->constrain) {
+                       csize.height = MAX(csize.height, asize.height);
+               }
                csize.x = asize.width;
-               csize.y = 0;
        }
 
-       if (bin->child)
-               gtk_widget_size_allocate(bin->child, &csize);
-       if (abin->child)
-               gtk_widget_size_allocate(abin->child, &asize);
+       csize.y = (allocation->height - csize.height) * abin->align + 0.5;
+       asize.y = (allocation->height - asize.height) * abin->align + 0.5;
+
+       if (abin->body)
+               gtk_widget_size_allocate(abin->body, &asize);
+       if (abin->side)
+               gtk_widget_size_allocate(abin->side, &csize);
 }
 
-void
-aspect_bin_set_body_widget(AspectBin *ab, GtkWidget *widget, gfloat ratio)
+static gboolean
+set_widget(GtkWidget **dest, GtkWidget *parent, GtkWidget *widget)
 {
        gboolean need_resize = FALSE;
 
-       g_return_if_fail(IS_ASPECT_BIN(ab));
-       g_return_if_fail(widget == NULL || GTK_IS_WIDGET(widget));
-       g_return_if_fail(widget == NULL || widget->parent == NULL);
+       if (*dest == widget)
+               return FALSE;
 
-       if (ab->child == widget)
-               return;
-       
-       if (ab->child) {
-               need_resize |= GTK_WIDGET_VISIBLE(ab->child);
-               gtk_widget_unparent(ab->child);
+       if (*dest) {
+               need_resize |= GTK_WIDGET_VISIBLE(*dest);
+               gtk_widget_unparent(*dest);
        }
 
-       ab->child = widget;
+       *dest = widget;
        if (widget) {
-               gtk_widget_set_parent(widget, GTK_WIDGET(ab));
+               gtk_widget_set_parent(widget, parent);
                need_resize |= GTK_WIDGET_VISIBLE(widget);
        }
 
-       if (GTK_WIDGET_VISIBLE(ab) && need_resize)
-               gtk_widget_queue_resize(GTK_WIDGET(ab));
+       return GTK_WIDGET_VISIBLE(parent) && need_resize;
+}
+
+void aspect_bin_set_body(AspectBin *abin, GtkWidget *widget, gfloat ratio)
+{
+       g_return_if_fail(IS_ASPECT_BIN(abin));
+       g_return_if_fail(widget == NULL || GTK_IS_WIDGET(widget));
+       g_return_if_fail(widget == NULL || widget->parent == NULL);
+
+       if (set_widget(&abin->body, GTK_WIDGET(abin), widget))
+               gtk_widget_queue_resize(GTK_WIDGET(abin));
+}
+
+void aspect_bin_set_side(AspectBin *abin, GtkWidget *widget)
+{
+       g_return_if_fail(IS_ASPECT_BIN(abin));
+       g_return_if_fail(widget == NULL || GTK_IS_WIDGET(widget));
+       g_return_if_fail(widget == NULL || widget->parent == NULL);
+
+       if (set_widget(&abin->side, GTK_WIDGET(abin), widget))
+               gtk_widget_queue_resize(GTK_WIDGET(abin));
 }