medit/moo/libfoocanvas/foo-canvas-widget.c
2007-05-26 02:43:55 -05:00

570 lines
14 KiB
C

/*
* Copyright (C) 1997, 1998, 1999, 2000 Free Software Foundation
* All rights reserved.
*
* This file is part of the Gnome Library.
*
* The Gnome Library is free software; you can redistribute it and/or
* modify it under the terms of the GNU Library General Public License as
* published by the Free Software Foundation; either version 2 of the
* License, or (at your option) any later version.
*
* The Gnome 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
* Library General Public License for more details.
*
* You should have received a copy of the GNU Library General Public
* License along with the Gnome Library; see the file COPYING.LIB. If not,
* write to the Free Software Foundation, Inc., 59 Temple Place - Suite 330,
* Boston, MA 02111-1307, USA.
*/
/*
@NOTATION@
*/
/* Widget item type for FooCanvas widget
*
* FooCanvas is basically a port of the Tk toolkit's most excellent canvas widget. Tk is
* copyrighted by the Regents of the University of California, Sun Microsystems, and other parties.
*
*
* Author: Federico Mena <federico@nuclecu.unam.mx>
*/
#include <config.h>
#include <math.h>
#include <gtk/gtksignal.h>
#include "foo-canvas-widget.h"
enum {
PROP_0,
PROP_WIDGET,
PROP_X,
PROP_Y,
PROP_WIDTH,
PROP_HEIGHT,
PROP_ANCHOR,
PROP_SIZE_PIXELS
};
static void foo_canvas_widget_class_init (FooCanvasWidgetClass *class);
static void foo_canvas_widget_init (FooCanvasWidget *witem);
static void foo_canvas_widget_destroy (GtkObject *object);
static void foo_canvas_widget_get_property (GObject *object,
guint param_id,
GValue *value,
GParamSpec *pspec);
static void foo_canvas_widget_set_property (GObject *object,
guint param_id,
const GValue *value,
GParamSpec *pspec);
static void foo_canvas_widget_update (FooCanvasItem *item, double i2w_dx, double i2w_dy, FooCanvasUpdateFlags flags);
static double foo_canvas_widget_point (FooCanvasItem *item, double x, double y,
int cx, int cy, FooCanvasItem **actual_item);
static void foo_canvas_widget_translate (FooCanvasItem *item, double dx, double dy);
static void foo_canvas_widget_bounds (FooCanvasItem *item, double *x1, double *y1, double *x2, double *y2);
static void foo_canvas_widget_map (FooCanvasItem *item);
static void foo_canvas_widget_unmap (FooCanvasItem *item);
G_DEFINE_TYPE (FooCanvasWidget, foo_canvas_widget, FOO_TYPE_CANVAS_ITEM)
static void
foo_canvas_widget_class_init (FooCanvasWidgetClass *class)
{
GObjectClass *gobject_class;
GtkObjectClass *object_class;
FooCanvasItemClass *item_class;
gobject_class = (GObjectClass *) class;
object_class = (GtkObjectClass *) class;
item_class = (FooCanvasItemClass *) class;
gobject_class->set_property = foo_canvas_widget_set_property;
gobject_class->get_property = foo_canvas_widget_get_property;
g_object_class_install_property
(gobject_class,
PROP_WIDGET,
g_param_spec_object ("widget", NULL, NULL,
GTK_TYPE_WIDGET,
G_PARAM_READWRITE));
g_object_class_install_property
(gobject_class,
PROP_X,
g_param_spec_double ("x", NULL, NULL,
-G_MAXDOUBLE, G_MAXDOUBLE, 0.0,
G_PARAM_READWRITE));
g_object_class_install_property
(gobject_class,
PROP_Y,
g_param_spec_double ("y", NULL, NULL,
-G_MAXDOUBLE, G_MAXDOUBLE, 0.0,
G_PARAM_READWRITE));
g_object_class_install_property
(gobject_class,
PROP_WIDTH,
g_param_spec_double ("width", NULL, NULL,
-G_MAXDOUBLE, G_MAXDOUBLE, 0.0,
G_PARAM_READWRITE));
g_object_class_install_property
(gobject_class,
PROP_HEIGHT,
g_param_spec_double ("height", NULL, NULL,
-G_MAXDOUBLE, G_MAXDOUBLE, 0.0,
G_PARAM_READWRITE));
g_object_class_install_property
(gobject_class,
PROP_ANCHOR,
g_param_spec_enum ("anchor", NULL, NULL,
GTK_TYPE_ANCHOR_TYPE,
GTK_ANCHOR_NW,
G_PARAM_READWRITE));
g_object_class_install_property
(gobject_class,
PROP_SIZE_PIXELS,
g_param_spec_boolean ("size-pixels", NULL, NULL,
FALSE,
G_PARAM_READWRITE));
object_class->destroy = foo_canvas_widget_destroy;
item_class->update = foo_canvas_widget_update;
item_class->point = foo_canvas_widget_point;
item_class->translate = foo_canvas_widget_translate;
item_class->bounds = foo_canvas_widget_bounds;
item_class->map = foo_canvas_widget_map;
item_class->unmap = foo_canvas_widget_unmap;
}
static void
foo_canvas_widget_init (FooCanvasWidget *witem)
{
witem->x = 0.0;
witem->y = 0.0;
witem->width = 0.0;
witem->height = 0.0;
witem->anchor = GTK_ANCHOR_NW;
witem->size_pixels = FALSE;
}
static void
foo_canvas_widget_destroy (GtkObject *object)
{
FooCanvasWidget *witem;
g_return_if_fail (object != NULL);
g_return_if_fail (FOO_IS_CANVAS_WIDGET (object));
witem = FOO_CANVAS_WIDGET (object);
if (witem->widget && !witem->in_destroy) {
g_signal_handler_disconnect (witem->widget, witem->destroy_id);
gtk_widget_destroy (witem->widget);
witem->widget = NULL;
}
if (GTK_OBJECT_CLASS (foo_canvas_widget_parent_class)->destroy)
GTK_OBJECT_CLASS (foo_canvas_widget_parent_class)->destroy (object);
}
static void
recalc_bounds (FooCanvasWidget *witem)
{
FooCanvasItem *item;
double wx, wy;
item = FOO_CANVAS_ITEM (witem);
/* Get world coordinates */
wx = witem->x;
wy = witem->y;
foo_canvas_item_i2w (item, &wx, &wy);
/* Get canvas pixel coordinates */
foo_canvas_w2c (item->canvas, wx, wy, &witem->cx, &witem->cy);
/* Anchor widget item */
switch (witem->anchor) {
case GTK_ANCHOR_NW:
case GTK_ANCHOR_W:
case GTK_ANCHOR_SW:
break;
case GTK_ANCHOR_N:
case GTK_ANCHOR_CENTER:
case GTK_ANCHOR_S:
witem->cx -= witem->cwidth / 2;
break;
case GTK_ANCHOR_NE:
case GTK_ANCHOR_E:
case GTK_ANCHOR_SE:
witem->cx -= witem->cwidth;
break;
default:
break;
}
switch (witem->anchor) {
case GTK_ANCHOR_NW:
case GTK_ANCHOR_N:
case GTK_ANCHOR_NE:
break;
case GTK_ANCHOR_W:
case GTK_ANCHOR_CENTER:
case GTK_ANCHOR_E:
witem->cy -= witem->cheight / 2;
break;
case GTK_ANCHOR_SW:
case GTK_ANCHOR_S:
case GTK_ANCHOR_SE:
witem->cy -= witem->cheight;
break;
default:
break;
}
/* Bounds */
item->x1 = witem->cx;
item->y1 = witem->cy;
item->x2 = witem->cx + witem->cwidth;
item->y2 = witem->cy + witem->cheight;
if (witem->widget)
gtk_layout_move (GTK_LAYOUT (item->canvas), witem->widget,
witem->cx,
witem->cy);
}
static void
do_destroy (G_GNUC_UNUSED GtkObject *object, gpointer data)
{
FooCanvasWidget *witem;
witem = data;
witem->in_destroy = TRUE;
gtk_object_destroy (data);
}
static void
foo_canvas_widget_set_property (GObject *object,
guint param_id,
const GValue *value,
GParamSpec *pspec)
{
FooCanvasItem *item;
FooCanvasWidget *witem;
GObject *obj;
int update;
int calc_bounds;
g_return_if_fail (object != NULL);
g_return_if_fail (FOO_IS_CANVAS_WIDGET (object));
item = FOO_CANVAS_ITEM (object);
witem = FOO_CANVAS_WIDGET (object);
update = FALSE;
calc_bounds = FALSE;
switch (param_id) {
case PROP_WIDGET:
if (witem->widget) {
g_signal_handler_disconnect (witem->widget, witem->destroy_id);
gtk_container_remove (GTK_CONTAINER (item->canvas), witem->widget);
}
obj = g_value_get_object (value);
if (obj) {
witem->widget = GTK_WIDGET (obj);
witem->destroy_id = g_signal_connect (obj, "destroy",
G_CALLBACK (do_destroy),
witem);
gtk_layout_put (GTK_LAYOUT (item->canvas), witem->widget,
witem->cx + item->canvas->zoom_xofs,
witem->cy + item->canvas->zoom_yofs);
}
update = TRUE;
break;
case PROP_X:
if (witem->x != g_value_get_double (value))
{
witem->x = g_value_get_double (value);
calc_bounds = TRUE;
}
break;
case PROP_Y:
if (witem->y != g_value_get_double (value))
{
witem->y = g_value_get_double (value);
calc_bounds = TRUE;
}
break;
case PROP_WIDTH:
if (witem->width != fabs (g_value_get_double (value)))
{
witem->width = fabs (g_value_get_double (value));
update = TRUE;
}
break;
case PROP_HEIGHT:
if (witem->height != fabs (g_value_get_double (value)))
{
witem->height = fabs (g_value_get_double (value));
update = TRUE;
}
break;
case PROP_ANCHOR:
if (witem->anchor != (GtkAnchorType)g_value_get_enum (value))
{
witem->anchor = g_value_get_enum (value);
update = TRUE;
}
break;
case PROP_SIZE_PIXELS:
if (witem->size_pixels != g_value_get_boolean (value))
{
witem->size_pixels = g_value_get_boolean (value);
update = TRUE;
}
break;
default:
G_OBJECT_WARN_INVALID_PROPERTY_ID (object, param_id, pspec);
break;
}
if (update)
(* FOO_CANVAS_ITEM_GET_CLASS (item)->update) (item, 0, 0, 0);
if (calc_bounds)
recalc_bounds (witem);
}
static void
foo_canvas_widget_get_property (GObject *object,
guint param_id,
GValue *value,
GParamSpec *pspec)
{
FooCanvasWidget *witem;
g_return_if_fail (object != NULL);
g_return_if_fail (FOO_IS_CANVAS_WIDGET (object));
witem = FOO_CANVAS_WIDGET (object);
switch (param_id) {
case PROP_WIDGET:
g_value_set_object (value, (GObject *) witem->widget);
break;
case PROP_X:
g_value_set_double (value, witem->x);
break;
case PROP_Y:
g_value_set_double (value, witem->y);
break;
case PROP_WIDTH:
g_value_set_double (value, witem->width);
break;
case PROP_HEIGHT:
g_value_set_double (value, witem->height);
break;
case PROP_ANCHOR:
g_value_set_enum (value, witem->anchor);
break;
case PROP_SIZE_PIXELS:
g_value_set_boolean (value, witem->size_pixels);
break;
default:
G_OBJECT_WARN_INVALID_PROPERTY_ID (object, param_id, pspec);
break;
}
}
static void
foo_canvas_widget_update (FooCanvasItem *item, double i2w_dx, double i2w_dy, FooCanvasUpdateFlags flags)
{
FooCanvasWidget *witem;
witem = FOO_CANVAS_WIDGET (item);
if (FOO_CANVAS_ITEM_CLASS (foo_canvas_widget_parent_class)->update)
FOO_CANVAS_ITEM_CLASS (foo_canvas_widget_parent_class)->update (item, i2w_dx, i2w_dy, flags);
if (witem->widget) {
if (witem->size_pixels) {
witem->cwidth = (int) (witem->width + 0.5);
witem->cheight = (int) (witem->height + 0.5);
} else {
witem->cwidth = (int) (witem->width * item->canvas->pixels_per_unit + 0.5);
witem->cheight = (int) (witem->height * item->canvas->pixels_per_unit + 0.5);
}
gtk_widget_set_size_request (witem->widget, witem->cwidth, witem->cheight);
} else {
witem->cwidth = 0.0;
witem->cheight = 0.0;
}
recalc_bounds (witem);
}
static void
foo_canvas_widget_map (FooCanvasItem *item)
{
FooCanvasWidget *witem = FOO_CANVAS_WIDGET (item);
if (FOO_CANVAS_ITEM_CLASS (foo_canvas_widget_parent_class)->map)
FOO_CANVAS_ITEM_CLASS (foo_canvas_widget_parent_class)->map (item);
if (witem->widget && GTK_WIDGET_VISIBLE (witem->widget))
gtk_widget_map (witem->widget);
}
static void
foo_canvas_widget_unmap (FooCanvasItem *item)
{
FooCanvasWidget *witem = FOO_CANVAS_WIDGET (item);
if (FOO_CANVAS_ITEM_CLASS (foo_canvas_widget_parent_class)->unmap)
FOO_CANVAS_ITEM_CLASS (foo_canvas_widget_parent_class)->unmap (item);
gtk_widget_unmap (witem->widget);
}
static double
foo_canvas_widget_point (FooCanvasItem *item, double x, double y,
G_GNUC_UNUSED int cx, G_GNUC_UNUSED int cy,
FooCanvasItem **actual_item)
{
FooCanvasWidget *witem;
double x1, y1, x2, y2;
double dx, dy;
witem = FOO_CANVAS_WIDGET (item);
*actual_item = item;
foo_canvas_c2w (item->canvas, witem->cx, witem->cy, &x1, &y1);
x2 = x1 + (witem->cwidth - 1) / item->canvas->pixels_per_unit;
y2 = y1 + (witem->cheight - 1) / item->canvas->pixels_per_unit;
/* Is point inside widget bounds? */
if ((x >= x1) && (y >= y1) && (x <= x2) && (y <= y2))
return 0.0;
/* Point is outside widget bounds */
if (x < x1)
dx = x1 - x;
else if (x > x2)
dx = x - x2;
else
dx = 0.0;
if (y < y1)
dy = y1 - y;
else if (y > y2)
dy = y - y2;
else
dy = 0.0;
return sqrt (dx * dx + dy * dy);
}
static void
foo_canvas_widget_translate (FooCanvasItem *item, double dx, double dy)
{
FooCanvasWidget *witem;
witem = FOO_CANVAS_WIDGET (item);
witem->x += dx;
witem->y += dy;
}
static void
foo_canvas_widget_bounds (FooCanvasItem *item, double *x1, double *y1, double *x2, double *y2)
{
FooCanvasWidget *witem;
witem = FOO_CANVAS_WIDGET (item);
*x1 = witem->x;
*y1 = witem->y;
switch (witem->anchor) {
case GTK_ANCHOR_NW:
case GTK_ANCHOR_W:
case GTK_ANCHOR_SW:
break;
case GTK_ANCHOR_N:
case GTK_ANCHOR_CENTER:
case GTK_ANCHOR_S:
*x1 -= witem->width / 2.0;
break;
case GTK_ANCHOR_NE:
case GTK_ANCHOR_E:
case GTK_ANCHOR_SE:
*x1 -= witem->width;
break;
default:
break;
}
switch (witem->anchor) {
case GTK_ANCHOR_NW:
case GTK_ANCHOR_N:
case GTK_ANCHOR_NE:
break;
case GTK_ANCHOR_W:
case GTK_ANCHOR_CENTER:
case GTK_ANCHOR_E:
*y1 -= witem->height / 2.0;
break;
case GTK_ANCHOR_SW:
case GTK_ANCHOR_S:
case GTK_ANCHOR_SE:
*y1 -= witem->height;
break;
default:
break;
}
*x2 = *x1 + witem->width;
*y2 = *y1 + witem->height;
}