diff options
author | Elliot Lee <sopwith@src.gnome.org> | 1997-11-24 22:37:52 +0000 |
---|---|---|
committer | Elliot Lee <sopwith@src.gnome.org> | 1997-11-24 22:37:52 +0000 |
commit | 9508b76bd2401b6b9e289b5c8ec9fc0e08909283 (patch) | |
tree | 53c88a9e5ac09e1a027e56df33bdaa66d670901b /gtk/gtkvruler.c | |
download | gtk+-9508b76bd2401b6b9e289b5c8ec9fc0e08909283.tar.gz |
Initial revision
Diffstat (limited to 'gtk/gtkvruler.c')
-rw-r--r-- | gtk/gtkvruler.c | 282 |
1 files changed, 282 insertions, 0 deletions
diff --git a/gtk/gtkvruler.c b/gtk/gtkvruler.c new file mode 100644 index 0000000000..43fe16bb7b --- /dev/null +++ b/gtk/gtkvruler.c @@ -0,0 +1,282 @@ +/* GTK - The GIMP Toolkit + * Copyright (C) 1995-1997 Peter Mattis, Spencer Kimball and Josh MacDonald + * + * This 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. + * + * 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 + * Library General Public License for more details. + * + * You should have received a copy of the GNU Library General Public + * License along with this library; if not, write to the Free + * Software Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA. + */ +#include <math.h> +#include <stdio.h> +#include <string.h> +#include "gtkvruler.h" + + +#define RULER_WIDTH 14 +#define MINIMUM_INCR 5 +#define MAXIMUM_SUBDIVIDE 5 +#define MAXIMUM_SCALES 10 + +#define ROUND(x) ((int) ((x) + 0.5)) + + +static void gtk_vruler_class_init (GtkVRulerClass *klass); +static void gtk_vruler_init (GtkVRuler *vruler); +static gint gtk_vruler_motion_notify (GtkWidget *widget, + GdkEventMotion *event); +static void gtk_vruler_draw_ticks (GtkRuler *ruler); +static void gtk_vruler_draw_pos (GtkRuler *ruler); + + +guint +gtk_vruler_get_type () +{ + static guint vruler_type = 0; + + if (!vruler_type) + { + GtkTypeInfo vruler_info = + { + "GtkVRuler", + sizeof (GtkVRuler), + sizeof (GtkVRulerClass), + (GtkClassInitFunc) gtk_vruler_class_init, + (GtkObjectInitFunc) gtk_vruler_init, + (GtkArgFunc) NULL, + }; + + vruler_type = gtk_type_unique (gtk_ruler_get_type (), &vruler_info); + } + + return vruler_type; +} + +static void +gtk_vruler_class_init (GtkVRulerClass *klass) +{ + GtkWidgetClass *widget_class; + GtkRulerClass *ruler_class; + + widget_class = (GtkWidgetClass*) klass; + ruler_class = (GtkRulerClass*) klass; + + widget_class->motion_notify_event = gtk_vruler_motion_notify; + + ruler_class->draw_ticks = gtk_vruler_draw_ticks; + ruler_class->draw_pos = gtk_vruler_draw_pos; +} + +static void +gtk_vruler_init (GtkVRuler *vruler) +{ + GtkWidget *widget; + + widget = GTK_WIDGET (vruler); + widget->requisition.width = widget->style->klass->xthickness * 2 + RULER_WIDTH; + widget->requisition.height = widget->style->klass->ythickness * 2 + 1; +} + +GtkWidget* +gtk_vruler_new () +{ + return GTK_WIDGET (gtk_type_new (gtk_vruler_get_type ())); +} + + +static gint +gtk_vruler_motion_notify (GtkWidget *widget, + GdkEventMotion *event) +{ + GtkRuler *ruler; + gint y; + + g_return_val_if_fail (widget != NULL, FALSE); + g_return_val_if_fail (GTK_IS_VRULER (widget), FALSE); + g_return_val_if_fail (event != NULL, FALSE); + + ruler = GTK_RULER (widget); + + if (event->is_hint) + gdk_window_get_pointer (widget->window, NULL, &y, NULL); + else + y = event->y; + + ruler->position = ruler->lower + ((ruler->upper - ruler->lower) * y) / widget->allocation.height; + + /* Make sure the ruler has been allocated already */ + if (ruler->backing_store != NULL) + gtk_ruler_draw_pos (ruler); + + return FALSE; +} + +static void +gtk_vruler_draw_ticks (GtkRuler *ruler) +{ + GtkWidget *widget; + GdkGC *gc; + gint i, j; + gint width, height; + gint xthickness; + gint ythickness; + gint length; + gfloat subd_incr; + gfloat step_incr; + gfloat increment; + gfloat start, end, cur; + gchar unit_str[12]; + gchar digit_str[2] = { '\0', '\0' }; + gint text_height; + gint digit_height; + gint pos; + gint scale; + + g_return_if_fail (ruler != NULL); + g_return_if_fail (GTK_IS_VRULER (ruler)); + + if (GTK_WIDGET_DRAWABLE (ruler)) + { + widget = GTK_WIDGET (ruler); + + gc = widget->style->fg_gc[GTK_STATE_NORMAL]; + xthickness = widget->style->klass->xthickness; + ythickness = widget->style->klass->ythickness; + digit_height = widget->style->font->ascent; + + width = widget->allocation.height; + height = widget->allocation.width - ythickness * 2; + gdk_draw_line (ruler->backing_store, gc, + height + xthickness, + ythickness, + height + xthickness, + widget->allocation.height - ythickness); + + if ((ruler->upper - ruler->lower) == 0) + return; + + increment = (gfloat) width * ruler->metric->pixels_per_unit / (ruler->upper - ruler->lower); + + /* determine the scale + * use the maximum extents of the ruler to determine the largest possible + * number to be displayed. calculate the height in pixels of this displayed + * text as for the vertical ruler case. use this height to find a scale + * which leaves sufficient room for drawing the ruler. + */ + scale = ceil (ruler->max_size / ruler->metric->pixels_per_unit); + sprintf (unit_str, "%d", scale); + text_height = strlen (unit_str) * digit_height + 1; + + for (scale = 0; scale < MAXIMUM_SCALES; scale++) + if (ruler->metric->ruler_scale[scale] * increment > 2 * text_height) + break; + + if (scale == MAXIMUM_SCALES) + scale = MAXIMUM_SCALES - 1; + + for (i = 0; i < MAXIMUM_SUBDIVIDE; i++) + { + subd_incr = (gfloat) ruler->metric->ruler_scale[scale] / (gfloat) ruler->metric->subdivide[i]; + step_incr = subd_incr * increment; + if (step_incr <= MINIMUM_INCR) + break; + + start = floor ((ruler->lower / ruler->metric->pixels_per_unit) / subd_incr) * subd_incr; + end = ceil ((ruler->upper / ruler->metric->pixels_per_unit) / subd_incr) * subd_incr; + + length = height / (i + 1) - 1; + + cur = start; + while (cur <= end) + { + pos = ROUND ((cur - (ruler->lower / ruler->metric->pixels_per_unit)) * increment); + + gdk_draw_line (ruler->backing_store, gc, + height + xthickness - length, + pos, + height + xthickness, + pos); + + if (i == 0) + { + sprintf (unit_str, "%d", (int) cur); + for (j = 0; j < (int) strlen (unit_str); j++) + { + digit_str[0] = unit_str[j]; + gdk_draw_string (ruler->backing_store, widget->style->font, gc, + xthickness + 1, + pos + digit_height * (j + 1) + 1, + digit_str); + } + } + + cur += subd_incr; + } + } + } +} + +static void +gtk_vruler_draw_pos (GtkRuler *ruler) +{ + GtkWidget *widget; + GdkGC *gc; + int i; + gint x, y; + gint width, height; + gint bs_width, bs_height; + gint xthickness; + gint ythickness; + gfloat increment; + + g_return_if_fail (ruler != NULL); + g_return_if_fail (GTK_IS_VRULER (ruler)); + + if (GTK_WIDGET_DRAWABLE (ruler)) + { + widget = GTK_WIDGET (ruler); + + gc = widget->style->fg_gc[GTK_STATE_NORMAL]; + xthickness = widget->style->klass->xthickness; + ythickness = widget->style->klass->ythickness; + width = widget->allocation.width - xthickness * 2; + height = widget->allocation.height; + + bs_height = width / 2; + bs_height |= 1; /* make sure it's odd */ + bs_width = bs_height / 2 + 1; + + if ((bs_width > 0) && (bs_height > 0)) + { + /* If a backing store exists, restore the ruler */ + if (ruler->backing_store && ruler->non_gr_exp_gc) + gdk_draw_pixmap (ruler->widget.window, + ruler->non_gr_exp_gc, + ruler->backing_store, + ruler->xsrc, ruler->ysrc, + ruler->xsrc, ruler->ysrc, + bs_width, bs_height); + + increment = (gfloat) height / (ruler->upper - ruler->lower); + + x = (width + bs_width) / 2 + xthickness; + y = ROUND ((ruler->position - ruler->lower) * increment) + (ythickness - bs_height) / 2 - 1; + + for (i = 0; i < bs_width; i++) + gdk_draw_line (widget->window, gc, + x + i, y + i, + x + i, y + bs_height - 1 - i); + + ruler->xsrc = x; + ruler->ysrc = y; + } + } +} |