aboutsummaryrefslogtreecommitdiffstats
diff options
context:
space:
mode:
-rw-r--r--calendar/ChangeLog14
-rw-r--r--calendar/gncal-full-day.c263
-rw-r--r--calendar/gui/gncal-full-day.c263
3 files changed, 336 insertions, 204 deletions
diff --git a/calendar/ChangeLog b/calendar/ChangeLog
index 9f6e7440f1..b6231297a1 100644
--- a/calendar/ChangeLog
+++ b/calendar/ChangeLog
@@ -1,3 +1,17 @@
+1998-05-18 Miguel de Icaza <miguel@nuclecu.unam.mx>
+
+ * gncal-full-day.c (layout_children): Implemented ultra-cool
+ layout for the events that share the same time range. Gals and
+ guys you can now drop Outlook on the recycle bin.
+
+ Which reminds me. We do not have a recycle bin. How could that
+ happen in a project as cool as this one? Someone explain this to
+ me.
+
+1998-05-18 Federico Mena <federico@nuclecu.unam.mx>
+
+ * gncal-full-day.c (paint_back): Eliminated unnecessary border repainting.
+
Sun May 17 17:55:03 1998 Havoc Pennington <hp@pobox.com>
* gncal-todo.c (simple_todo_editor): Close dialog when return is pressed.
diff --git a/calendar/gncal-full-day.c b/calendar/gncal-full-day.c
index 1ac7613dbf..41d217cb6e 100644
--- a/calendar/gncal-full-day.c
+++ b/calendar/gncal-full-day.c
@@ -192,13 +192,16 @@ child_set_text_pos (Child *child)
{
GtkAllocation allocation;
int has_focus;
-
+ int handle_size;
+
has_focus = GTK_WIDGET_HAS_FOCUS (child->widget);
- allocation.x = HANDLE_SIZE;
- allocation.y = has_focus ? HANDLE_SIZE : 0;
- allocation.width = child->width - HANDLE_SIZE - child->decor_width;
- allocation.height = child->height - (has_focus ? (2 * HANDLE_SIZE) : 0);
+ handle_size = (child->ico->recur) ? 0 : HANDLE_SIZE;
+
+ allocation.x = handle_size;
+ allocation.y = has_focus ? handle_size : 0;
+ allocation.width = child->width - handle_size - child->decor_width;
+ allocation.height = child->height - (has_focus ? (2 * handle_size) : 0);
gtk_widget_size_request (child->widget, &child->widget->requisition); /* FIXME: is this needed? */
gtk_widget_size_allocate (child->widget, &allocation);
@@ -556,7 +559,7 @@ child_set_size (Child *child)
width = child->width;
height = child->rows_used * row_height;
- if (GTK_WIDGET_HAS_FOCUS (child->widget)) {
+ if (GTK_WIDGET_HAS_FOCUS (child->widget) && !child->ico->recur) {
y -= HANDLE_SIZE;
height += 2 * HANDLE_SIZE;
}
@@ -572,6 +575,7 @@ child_focus_in (GtkWidget *widget, GdkEventFocus *event, gpointer data)
child = data;
child_set_size (child);
+ gdk_window_raise (child->window);
return FALSE;
}
@@ -735,83 +739,6 @@ child_destroy (GncalFullDay *fullday, Child *child)
g_free (child);
}
-static struct layout_row *
-layout_get_rows (GncalFullDay *fullday, int *rowcount)
-{
- struct layout_row *rows;
- int max_i;
- int f_rows;
- GList *children;
- Child *child;
- int i, n;
-
- get_tm_range (fullday, fullday->lower, fullday->upper, NULL, NULL, NULL, &f_rows);
-
- *rowcount = f_rows;
- rows = g_new0 (struct layout_row, f_rows);
- max_i = 0;
-
- for (children = fullday->children; children; children = children->next) {
- child = children->data;
-
- for (i = 0; i < child->rows_used; i++) {
- n = child->lower_row + i;
-
- rows[n].intersections++;
-
- if (rows[n].intersections > max_i)
- max_i = rows[n].intersections;
- }
- }
-
- for (i = 0; i < f_rows; i++)
- rows[i].slots = g_new0 (int, max_i);
-
- return rows;
-}
-
-static void
-layout_free_rows (struct layout_row *rows, int f_rows)
-{
- int i;
-
- for (i = 0; i < f_rows; i++)
- g_free (rows[i].slots);
-
- g_free (rows);
-}
-
-static void
-layout_get_child_intersections (Child *child, struct layout_row *rows, int *min, int *max)
-{
- int i, n;
- int imin, imax;
-
- imax = 0;
-
- for (i = 0; i < child->rows_used; i++) {
- n = child->lower_row + i;
-
- if (rows[n].intersections > imax)
- imax = rows[n].intersections;
- }
-
- imin = imax;
-
- for (i = 0; i < child->rows_used; i++) {
- n = child->lower_row + i;
-
- if (rows[n].intersections < imin)
- imin = rows[n].intersections;
- }
-
- if (min)
- *min = imin;
-
- if (max)
- *max = imax;
-}
-
static int
calc_labels_width (GncalFullDay *fullday)
{
@@ -846,42 +773,145 @@ calc_labels_width (GncalFullDay *fullday)
return max_w;
}
-static void
-layout_child (GncalFullDay *fullday, Child *child, struct layout_row *rows, int left_x)
+#define MAX_CHILDREN_ON_ROW 32
+
+#define xy(a,x,y) (a[((y) * MAX_CHILDREN_ON_ROW) + (x)])
+
+static int
+range_empty (char *array, int slot, int lower, int count)
{
- GtkWidget *widget;
+ int i;
- widget = GTK_WIDGET (fullday);
+ for (i = 0; i < count; i++)
+ if (xy (array, slot, lower+i) != 0)
+ return 0;
+ return 1;
+}
- child->x = left_x;
+static void
+range_allocate (char *array, int slot, int lower, int count, int val)
+{
+ int i;
- /* FIXME: for now, the children overlap. Make it layout them nicely. */
+ for (i = 0; i < count; i++)
+ xy (array, slot, lower+i) = val;
+}
- child->width = widget->allocation.width - (widget->style->klass->xthickness + left_x);
+static int
+can_expand (char *array, int *allocations, int top_slot, int val, int lower, int count)
+{
+ int slot, i;
+ int cols = 0;
- /* Position child */
+ for (slot = allocations [val] + 1; slot < top_slot; slot++){
+ for (i = 0; i < count; i++)
+ if (xy (array, slot, lower+i))
+ return cols;
+ cols++;
+ }
+ return cols;
+}
- child_set_size (child);
+static void
+expand_space (char *array, int *allocations, int val, int lower, int count, int cols)
+{
+ int j, i, slot;
+
+ for (i = 0; i < count; i++){
+ slot = allocations [val] + 1;
+ for (j = 0; j < cols; j++)
+ xy (array, slot, lower+i) = val;
+ }
}
static void
layout_children (GncalFullDay *fullday)
{
- struct layout_row *rows;
- GList *children;
GtkWidget *widget;
- int left_x, rowcount;
-
- rows = layout_get_rows (fullday, &rowcount);
+ int lines = (24 * 60) / fullday->interval;
+ char *array = g_malloc0 (sizeof (char) * lines * MAX_CHILDREN_ON_ROW);
+ GList *children;
+ int val, slot;
+ int *allocations, *columns;
+ int top_slot = 0;
+ int left_x, cols;
+ int pixels_per_col, extra_pixels, extra, usable_pixels;
+ int child_count;
+
+ if (!fullday->children)
+ return;
+
+ /* initial allocation */
+ child_count = g_list_length (fullday->children) + 2;
+ allocations = g_malloc0 (sizeof (int) * child_count);
+ columns = g_malloc0 (sizeof (int) * child_count);
+ val = 1;
+ for (children = fullday->children; children; children = children->next, val++){
+ Child *child = children->data;
+
+ allocations [val] = 0;
+ columns [val] = 1;
+ for (slot = 0; slot < MAX_CHILDREN_ON_ROW; slot++){
+ if (range_empty (array, slot, child->lower_row, child->rows_used)){
+ /*
+ printf ("Child %d uses %d-%d allocates slot=%d\n", val, child->lower_row,
+ child->lower_row + child->rows_used, slot);
+ */
+
+ range_allocate (array, slot, child->lower_row, child->rows_used, val);
+ allocations [val] = slot;
+ columns [val] = 1;
+ if (slot+1 > top_slot)
+ top_slot = slot+1;
+ break;
+ }
+ }
+ }
+#if DEBUGME
+ for (val = 0; val < 48; val++){
+ int j;
+
+ printf ("%d: ", val);
+ for (j = 0; j < top_slot; j++){
+ printf (" %d", xy (array, j, val));
+ }
+ printf ("\n");
+ }
+#endif
+ /* Expand */
+ val = 1;
+ for (children = fullday->children; children; children = children->next, val++){
+ Child *child = children->data;
+
+ cols = can_expand (array, allocations, top_slot, val, child->lower_row, child->rows_used);
+ /* printf ("Can expand regresa: %d\n", cols); */
+ if (!cols)
+ continue;
+ expand_space (array, allocations, val, child->lower_row, child->rows_used, cols);
+ columns [val] += cols;
+ }
+ /* Assign the spaces */
widget = GTK_WIDGET (fullday);
-
left_x = 2 * (widget->style->klass->xthickness + TEXT_BORDER) + calc_labels_width (fullday);
- for (children = fullday->children; children; children = children->next)
- layout_child (fullday, children->data, rows, left_x);
+ usable_pixels = widget->allocation.width-left_x - widget->style->klass->xthickness;
+ pixels_per_col = usable_pixels / top_slot;
+ extra_pixels = usable_pixels % top_slot;
+
+ val = 1;
+ for (children = fullday->children; children; children = children->next, val++){
+ Child *child = children->data;
+
+ child->x = left_x + pixels_per_col * allocations [val];
+ extra = (allocations [val] + columns [val] == top_slot) ? extra_pixels : 0;
+ child->width = pixels_per_col * columns [val] + extra;
+ child_set_size (child);
- layout_free_rows (rows, rowcount);
+ /* printf ("Setting child %d to %d for %d pixels\n", val, allocations [val], columns [val]); */
+ }
+ g_free (allocations);
+ g_free (columns);
}
guint
@@ -1236,6 +1266,7 @@ paint_back (GncalFullDay *fullday, GdkRectangle *area)
int i;
GdkRectangle rect, dest, aarea;
int f_rows;
+ int draw_focus;
p.widget = GTK_WIDGET (fullday);
p.di = fullday->drag_info;
@@ -1261,7 +1292,35 @@ paint_back (GncalFullDay *fullday, GdkRectangle *area)
/* Frame shadow */
- gtk_widget_draw_focus (p.widget);
+ rect.x = 0;
+ rect.y = 0;
+ rect.width = p.widget->allocation.width;
+ rect.height = p.widget->style->klass->ythickness;
+
+ draw_focus = gdk_rectangle_intersect (&rect, area, &dest);
+
+ if (!draw_focus) {
+ rect.y = p.widget->allocation.height - rect.height;
+
+ draw_focus = gdk_rectangle_intersect (&rect, area, &dest);
+ }
+
+ if (!draw_focus) {
+ rect.y = p.widget->style->klass->ythickness;
+ rect.width = p.widget->style->klass->xthickness;
+ rect.height = p.widget->allocation.height - 2 * rect.y;
+
+ draw_focus = gdk_rectangle_intersect (&rect, area, &dest);
+ }
+
+ if (!draw_focus) {
+ rect.x = p.widget->allocation.width - rect.width;
+
+ draw_focus = gdk_rectangle_intersect (&rect, area, &dest);
+ }
+
+ if (draw_focus)
+ gtk_widget_draw_focus (p.widget);
/* Rows */
diff --git a/calendar/gui/gncal-full-day.c b/calendar/gui/gncal-full-day.c
index 1ac7613dbf..41d217cb6e 100644
--- a/calendar/gui/gncal-full-day.c
+++ b/calendar/gui/gncal-full-day.c
@@ -192,13 +192,16 @@ child_set_text_pos (Child *child)
{
GtkAllocation allocation;
int has_focus;
-
+ int handle_size;
+
has_focus = GTK_WIDGET_HAS_FOCUS (child->widget);
- allocation.x = HANDLE_SIZE;
- allocation.y = has_focus ? HANDLE_SIZE : 0;
- allocation.width = child->width - HANDLE_SIZE - child->decor_width;
- allocation.height = child->height - (has_focus ? (2 * HANDLE_SIZE) : 0);
+ handle_size = (child->ico->recur) ? 0 : HANDLE_SIZE;
+
+ allocation.x = handle_size;
+ allocation.y = has_focus ? handle_size : 0;
+ allocation.width = child->width - handle_size - child->decor_width;
+ allocation.height = child->height - (has_focus ? (2 * handle_size) : 0);
gtk_widget_size_request (child->widget, &child->widget->requisition); /* FIXME: is this needed? */
gtk_widget_size_allocate (child->widget, &allocation);
@@ -556,7 +559,7 @@ child_set_size (Child *child)
width = child->width;
height = child->rows_used * row_height;
- if (GTK_WIDGET_HAS_FOCUS (child->widget)) {
+ if (GTK_WIDGET_HAS_FOCUS (child->widget) && !child->ico->recur) {
y -= HANDLE_SIZE;
height += 2 * HANDLE_SIZE;
}
@@ -572,6 +575,7 @@ child_focus_in (GtkWidget *widget, GdkEventFocus *event, gpointer data)
child = data;
child_set_size (child);
+ gdk_window_raise (child->window);
return FALSE;
}
@@ -735,83 +739,6 @@ child_destroy (GncalFullDay *fullday, Child *child)
g_free (child);
}
-static struct layout_row *
-layout_get_rows (GncalFullDay *fullday, int *rowcount)
-{
- struct layout_row *rows;
- int max_i;
- int f_rows;
- GList *children;
- Child *child;
- int i, n;
-
- get_tm_range (fullday, fullday->lower, fullday->upper, NULL, NULL, NULL, &f_rows);
-
- *rowcount = f_rows;
- rows = g_new0 (struct layout_row, f_rows);
- max_i = 0;
-
- for (children = fullday->children; children; children = children->next) {
- child = children->data;
-
- for (i = 0; i < child->rows_used; i++) {
- n = child->lower_row + i;
-
- rows[n].intersections++;
-
- if (rows[n].intersections > max_i)
- max_i = rows[n].intersections;
- }
- }
-
- for (i = 0; i < f_rows; i++)
- rows[i].slots = g_new0 (int, max_i);
-
- return rows;
-}
-
-static void
-layout_free_rows (struct layout_row *rows, int f_rows)
-{
- int i;
-
- for (i = 0; i < f_rows; i++)
- g_free (rows[i].slots);
-
- g_free (rows);
-}
-
-static void
-layout_get_child_intersections (Child *child, struct layout_row *rows, int *min, int *max)
-{
- int i, n;
- int imin, imax;
-
- imax = 0;
-
- for (i = 0; i < child->rows_used; i++) {
- n = child->lower_row + i;
-
- if (rows[n].intersections > imax)
- imax = rows[n].intersections;
- }
-
- imin = imax;
-
- for (i = 0; i < child->rows_used; i++) {
- n = child->lower_row + i;
-
- if (rows[n].intersections < imin)
- imin = rows[n].intersections;
- }
-
- if (min)
- *min = imin;
-
- if (max)
- *max = imax;
-}
-
static int
calc_labels_width (GncalFullDay *fullday)
{
@@ -846,42 +773,145 @@ calc_labels_width (GncalFullDay *fullday)
return max_w;
}
-static void
-layout_child (GncalFullDay *fullday, Child *child, struct layout_row *rows, int left_x)
+#define MAX_CHILDREN_ON_ROW 32
+
+#define xy(a,x,y) (a[((y) * MAX_CHILDREN_ON_ROW) + (x)])
+
+static int
+range_empty (char *array, int slot, int lower, int count)
{
- GtkWidget *widget;
+ int i;
- widget = GTK_WIDGET (fullday);
+ for (i = 0; i < count; i++)
+ if (xy (array, slot, lower+i) != 0)
+ return 0;
+ return 1;
+}
- child->x = left_x;
+static void
+range_allocate (char *array, int slot, int lower, int count, int val)
+{
+ int i;
- /* FIXME: for now, the children overlap. Make it layout them nicely. */
+ for (i = 0; i < count; i++)
+ xy (array, slot, lower+i) = val;
+}
- child->width = widget->allocation.width - (widget->style->klass->xthickness + left_x);
+static int
+can_expand (char *array, int *allocations, int top_slot, int val, int lower, int count)
+{
+ int slot, i;
+ int cols = 0;
- /* Position child */
+ for (slot = allocations [val] + 1; slot < top_slot; slot++){
+ for (i = 0; i < count; i++)
+ if (xy (array, slot, lower+i))
+ return cols;
+ cols++;
+ }
+ return cols;
+}
- child_set_size (child);
+static void
+expand_space (char *array, int *allocations, int val, int lower, int count, int cols)
+{
+ int j, i, slot;
+
+ for (i = 0; i < count; i++){
+ slot = allocations [val] + 1;
+ for (j = 0; j < cols; j++)
+ xy (array, slot, lower+i) = val;
+ }
}
static void
layout_children (GncalFullDay *fullday)
{
- struct layout_row *rows;
- GList *children;
GtkWidget *widget;
- int left_x, rowcount;
-
- rows = layout_get_rows (fullday, &rowcount);
+ int lines = (24 * 60) / fullday->interval;
+ char *array = g_malloc0 (sizeof (char) * lines * MAX_CHILDREN_ON_ROW);
+ GList *children;
+ int val, slot;
+ int *allocations, *columns;
+ int top_slot = 0;
+ int left_x, cols;
+ int pixels_per_col, extra_pixels, extra, usable_pixels;
+ int child_count;
+
+ if (!fullday->children)
+ return;
+
+ /* initial allocation */
+ child_count = g_list_length (fullday->children) + 2;
+ allocations = g_malloc0 (sizeof (int) * child_count);
+ columns = g_malloc0 (sizeof (int) * child_count);
+ val = 1;
+ for (children = fullday->children; children; children = children->next, val++){
+ Child *child = children->data;
+
+ allocations [val] = 0;
+ columns [val] = 1;
+ for (slot = 0; slot < MAX_CHILDREN_ON_ROW; slot++){
+ if (range_empty (array, slot, child->lower_row, child->rows_used)){
+ /*
+ printf ("Child %d uses %d-%d allocates slot=%d\n", val, child->lower_row,
+ child->lower_row + child->rows_used, slot);
+ */
+
+ range_allocate (array, slot, child->lower_row, child->rows_used, val);
+ allocations [val] = slot;
+ columns [val] = 1;
+ if (slot+1 > top_slot)
+ top_slot = slot+1;
+ break;
+ }
+ }
+ }
+#if DEBUGME
+ for (val = 0; val < 48; val++){
+ int j;
+
+ printf ("%d: ", val);
+ for (j = 0; j < top_slot; j++){
+ printf (" %d", xy (array, j, val));
+ }
+ printf ("\n");
+ }
+#endif
+ /* Expand */
+ val = 1;
+ for (children = fullday->children; children; children = children->next, val++){
+ Child *child = children->data;
+
+ cols = can_expand (array, allocations, top_slot, val, child->lower_row, child->rows_used);
+ /* printf ("Can expand regresa: %d\n", cols); */
+ if (!cols)
+ continue;
+ expand_space (array, allocations, val, child->lower_row, child->rows_used, cols);
+ columns [val] += cols;
+ }
+ /* Assign the spaces */
widget = GTK_WIDGET (fullday);
-
left_x = 2 * (widget->style->klass->xthickness + TEXT_BORDER) + calc_labels_width (fullday);
- for (children = fullday->children; children; children = children->next)
- layout_child (fullday, children->data, rows, left_x);
+ usable_pixels = widget->allocation.width-left_x - widget->style->klass->xthickness;
+ pixels_per_col = usable_pixels / top_slot;
+ extra_pixels = usable_pixels % top_slot;
+
+ val = 1;
+ for (children = fullday->children; children; children = children->next, val++){
+ Child *child = children->data;
+
+ child->x = left_x + pixels_per_col * allocations [val];
+ extra = (allocations [val] + columns [val] == top_slot) ? extra_pixels : 0;
+ child->width = pixels_per_col * columns [val] + extra;
+ child_set_size (child);
- layout_free_rows (rows, rowcount);
+ /* printf ("Setting child %d to %d for %d pixels\n", val, allocations [val], columns [val]); */
+ }
+ g_free (allocations);
+ g_free (columns);
}
guint
@@ -1236,6 +1266,7 @@ paint_back (GncalFullDay *fullday, GdkRectangle *area)
int i;
GdkRectangle rect, dest, aarea;
int f_rows;
+ int draw_focus;
p.widget = GTK_WIDGET (fullday);
p.di = fullday->drag_info;
@@ -1261,7 +1292,35 @@ paint_back (GncalFullDay *fullday, GdkRectangle *area)
/* Frame shadow */
- gtk_widget_draw_focus (p.widget);
+ rect.x = 0;
+ rect.y = 0;
+ rect.width = p.widget->allocation.width;
+ rect.height = p.widget->style->klass->ythickness;
+
+ draw_focus = gdk_rectangle_intersect (&rect, area, &dest);
+
+ if (!draw_focus) {
+ rect.y = p.widget->allocation.height - rect.height;
+
+ draw_focus = gdk_rectangle_intersect (&rect, area, &dest);
+ }
+
+ if (!draw_focus) {
+ rect.y = p.widget->style->klass->ythickness;
+ rect.width = p.widget->style->klass->xthickness;
+ rect.height = p.widget->allocation.height - 2 * rect.y;
+
+ draw_focus = gdk_rectangle_intersect (&rect, area, &dest);
+ }
+
+ if (!draw_focus) {
+ rect.x = p.widget->allocation.width - rect.width;
+
+ draw_focus = gdk_rectangle_intersect (&rect, area, &dest);
+ }
+
+ if (draw_focus)
+ gtk_widget_draw_focus (p.widget);
/* Rows */