aboutsummaryrefslogtreecommitdiffstats
path: root/calendar/gui/gncal-full-day.c
diff options
context:
space:
mode:
Diffstat (limited to 'calendar/gui/gncal-full-day.c')
-rw-r--r--calendar/gui/gncal-full-day.c150
1 files changed, 126 insertions, 24 deletions
diff --git a/calendar/gui/gncal-full-day.c b/calendar/gui/gncal-full-day.c
index 398472a576..dc61018ecd 100644
--- a/calendar/gui/gncal-full-day.c
+++ b/calendar/gui/gncal-full-day.c
@@ -786,9 +786,10 @@ paint_back (GncalFullDay *fullday, GdkRectangle *area)
GdkRectangle rect, dest, aarea;
struct drag_info *di;
int x1, y1, width, height;
- int labels_width, division_x;
- int rows, row_height;
+ int labels_width;
+ int f_rows, row_height;
int i, y;
+ GdkGC *gc;
struct tm tm;
char buf[256];
@@ -808,20 +809,25 @@ paint_back (GncalFullDay *fullday, GdkRectangle *area)
width = widget->allocation.width - 2 * x1;
height = widget->allocation.height - 2 * y1;
- /* Clear and paint frame shadow */
+ di = fullday->drag_info;
+ labels_width = calc_labels_width (fullday);
+ row_height = calc_row_height (fullday);
+ get_tm_range (fullday, fullday->lower, fullday->upper, &tm, NULL, NULL, &f_rows);
- gdk_window_clear_area (widget->window, area->x, area->y, area->width, area->height);
+ /* Frame shadow */
gtk_widget_draw_focus (widget);
- /* Clear space for labels */
-
- labels_width = calc_labels_width (fullday);
+ /* Area for labels before selection */
rect.x = x1;
rect.y = y1;
rect.width = 2 * TEXT_BORDER + labels_width;
- rect.height = height;
+
+ if (di->sel_rows_used == 0)
+ rect.height = height;
+ else
+ rect.height = row_height * di->sel_start_row;
if (gdk_rectangle_intersect (&rect, area, &dest))
gdk_draw_rectangle (widget->window,
@@ -830,13 +836,19 @@ paint_back (GncalFullDay *fullday, GdkRectangle *area)
dest.x, dest.y,
dest.width, dest.height);
- /* Selected region */
+ /* Blank area before selection */
- di = fullday->drag_info;
+ rect.x += rect.width + widget->style->klass->xthickness;
+ rect.width = width - (rect.x - x1);
- get_tm_range (fullday, fullday->lower, fullday->upper, &tm, NULL, NULL, &rows);
+ if (gdk_rectangle_intersect (&rect, area, &dest))
+ gdk_draw_rectangle (widget->window,
+ widget->style->bg_gc[GTK_STATE_PRELIGHT],
+ TRUE,
+ dest.x, dest.y,
+ dest.width, dest.height);
- row_height = calc_row_height (fullday);
+ /* Selection area */
if (di->sel_rows_used != 0) {
rect.x = x1;
@@ -849,24 +861,52 @@ paint_back (GncalFullDay *fullday, GdkRectangle *area)
widget->style->bg_gc[GTK_STATE_SELECTED],
TRUE,
dest.x, dest.y,
- dest.width, dest.height);
+ dest.width, dest.height);
}
- /* Vertical division */
+ /* Areas under selection */
+
+ if (di->sel_rows_used != 0) {
+ /* Area for labels */
+
+ rect.x = x1;
+ rect.y = y1 + row_height * (di->sel_start_row + di->sel_rows_used);
+ rect.width = 2 * TEXT_BORDER + labels_width;
+ rect.height = height - rect.y;
+
+ if (gdk_rectangle_intersect (&rect, area, &dest))
+ gdk_draw_rectangle (widget->window,
+ widget->style->bg_gc[GTK_STATE_NORMAL],
+ TRUE,
+ dest.x, dest.y,
+ dest.width, dest.height);
- division_x = x1 + 2 * TEXT_BORDER + labels_width;
+ /* Blank area */
+
+ rect.x += rect.width + widget->style->klass->xthickness;
+ rect.width = width - (rect.x - x1);
+
+ if (gdk_rectangle_intersect (&rect, area, &dest))
+ gdk_draw_rectangle (widget->window,
+ widget->style->bg_gc[GTK_STATE_PRELIGHT],
+ TRUE,
+ dest.x, dest.y,
+ dest.width, dest.height);
+ }
+
+ /* Vertical division */
gtk_draw_vline (widget->style, widget->window,
GTK_STATE_NORMAL,
y1,
y1 + height - 1,
- division_x);
+ x1 + 2 * TEXT_BORDER + labels_width);
/* Horizontal divisions */
y = y1 + row_height - 1;
- for (i = 1; i < rows; i++) {
+ for (i = 1; i < f_rows; i++) {
gdk_draw_line (widget->window,
widget->style->black_gc,
x1, y,
@@ -884,15 +924,22 @@ paint_back (GncalFullDay *fullday, GdkRectangle *area)
rect.width = 2 * TEXT_BORDER + labels_width;
rect.height = row_height - 1;
- for (i = 0; i < rows; i++) {
+ for (i = 0; i < f_rows; i++) {
mktime (&tm);
if (gdk_rectangle_intersect (&rect, area, &dest)) {
strftime (buf, 256, "%X", &tm);
+ if ((di->sel_rows_used != 0)
+ && (i >= di->sel_start_row)
+ && (i < (di->sel_start_row + di->sel_rows_used)))
+ gc = widget->style->fg_gc[GTK_STATE_SELECTED];
+ else
+ gc = widget->style->fg_gc[GTK_STATE_NORMAL];
+
gdk_draw_string (widget->window,
widget->style->font,
- widget->style->fg_gc[GTK_STATE_NORMAL],
+ gc,
x1 + TEXT_BORDER,
y + widget->style->font->ascent,
buf);
@@ -906,6 +953,29 @@ paint_back (GncalFullDay *fullday, GdkRectangle *area)
}
static void
+paint_back_rows (GncalFullDay *fullday, int start_row, int rows_used)
+{
+ int row_height;
+ int xthickness, ythickness;
+ GtkWidget *widget;
+ GdkRectangle area;
+
+ widget = GTK_WIDGET (fullday);
+
+ row_height = calc_row_height (fullday);
+
+ xthickness = widget->style->klass->xthickness;
+ ythickness = widget->style->klass->ythickness;
+
+ area.x = xthickness;
+ area.y = ythickness + start_row * row_height;
+ area.width = widget->allocation.width - 2 * xthickness;
+ area.height = rows_used * row_height;
+
+ paint_back (fullday, &area);
+}
+
+static void
gncal_full_day_draw (GtkWidget *widget, GdkRectangle *area)
{
GncalFullDay *fullday;
@@ -1105,6 +1175,9 @@ gncal_full_day_button_press (GtkWidget *widget, GdkEventButton *event)
struct drag_info *di;
gint y;
int row_height;
+ int old_start_row, old_rows_used;
+ int old_max;
+ int paint_start_row, paint_rows_used;
g_return_val_if_fail (widget != NULL, FALSE);
g_return_val_if_fail (GNCAL_IS_FULL_DAY (widget), FALSE);
@@ -1124,6 +1197,9 @@ gncal_full_day_button_press (GtkWidget *widget, GdkEventButton *event)
di->drag_mode = DRAG_SELECT;
+ old_start_row = di->sel_start_row;
+ old_rows_used = di->sel_rows_used;
+
di->sel_click_row = get_row_from_y (fullday, event->y, FALSE);
di->sel_start_row = di->sel_click_row;
di->sel_rows_used = 1;
@@ -1138,7 +1214,16 @@ gncal_full_day_button_press (GtkWidget *widget, GdkEventButton *event)
NULL,
event->time);
- paint_back (fullday, NULL);
+ if (old_rows_used == 0) {
+ paint_start_row = di->sel_start_row;
+ paint_rows_used = di->sel_rows_used;
+ } else {
+ paint_start_row = MIN (old_start_row, di->sel_start_row);
+ old_max = old_start_row + old_rows_used - 1;
+ paint_rows_used = MAX (old_max, di->sel_start_row) - paint_start_row + 1;
+ }
+
+ paint_back_rows (fullday, paint_start_row, paint_rows_used);
} else {
/* Clicked on a child? */
@@ -1301,7 +1386,7 @@ gncal_full_day_button_release (GtkWidget *widget, GdkEventButton *event)
gdk_pointer_ungrab (event->time);
- paint_back (fullday, NULL);
+ paint_back_rows (fullday, di->sel_start_row, MAX (di->sel_rows_used, 1));
break;
@@ -1333,6 +1418,9 @@ gncal_full_day_motion (GtkWidget *widget, GdkEventMotion *event)
GncalFullDay *fullday;
struct drag_info *di;
gint y;
+ int old_min, old_max;
+ int new_min, new_max;
+ int new_start_row, new_rows_used;
g_return_val_if_fail (widget != NULL, FALSE);
g_return_val_if_fail (GNCAL_IS_FULL_DAY (widget), FALSE);
@@ -1348,8 +1436,18 @@ gncal_full_day_motion (GtkWidget *widget, GdkEventMotion *event)
break;
case DRAG_SELECT:
+ old_min = di->sel_start_row;
+ old_max = di->sel_start_row + di->sel_rows_used - 1;
+
recompute_motion (fullday, y);
- paint_back (fullday, NULL);
+
+ new_min = di->sel_start_row;
+ new_max = di->sel_start_row + di->sel_rows_used - 1;
+
+ new_start_row = MIN (old_min, new_min);
+ new_rows_used = MAX (old_max, new_max) - new_start_row + 1;
+
+ paint_back_rows (fullday, new_start_row, new_rows_used);
break;
@@ -1516,14 +1614,14 @@ gncal_full_day_update (GncalFullDay *fullday)
child_map (fullday, children->data);
}
- /* FIXME: paint or something */
-
gtk_widget_draw (GTK_WIDGET (fullday), NULL);
}
void
gncal_full_day_set_bounds (GncalFullDay *fullday, time_t lower, time_t upper)
{
+ struct drag_info *di;
+
g_return_if_fail (fullday != NULL);
g_return_if_fail (GNCAL_IS_FULL_DAY (fullday));
@@ -1531,6 +1629,10 @@ gncal_full_day_set_bounds (GncalFullDay *fullday, time_t lower, time_t upper)
fullday->lower = lower;
fullday->upper = upper;
+ di = fullday->drag_info;
+
+ di->sel_rows_used = 0; /* clear selection */
+
gncal_full_day_update (fullday);
}
}