aboutsummaryrefslogtreecommitdiffstats
path: root/calendar/gui/e-day-view.c
diff options
context:
space:
mode:
Diffstat (limited to 'calendar/gui/e-day-view.c')
-rw-r--r--calendar/gui/e-day-view.c497
1 files changed, 330 insertions, 167 deletions
diff --git a/calendar/gui/e-day-view.c b/calendar/gui/e-day-view.c
index 99a9817a33..2ec44e2ac6 100644
--- a/calendar/gui/e-day-view.c
+++ b/calendar/gui/e-day-view.c
@@ -85,6 +85,7 @@ static void e_day_view_style_set (GtkWidget *widget,
GtkStyle *previous_style);
static void e_day_view_size_allocate (GtkWidget *widget,
GtkAllocation *allocation);
+static gboolean e_day_view_update_scroll_regions (EDayView *day_view);
static gint e_day_view_focus_in (GtkWidget *widget,
GdkEventFocus *event);
static gint e_day_view_focus_out (GtkWidget *widget,
@@ -111,6 +112,7 @@ static gboolean e_day_view_on_main_canvas_button_press (GtkWidget *widget,
static gboolean e_day_view_on_main_canvas_button_release (GtkWidget *widget,
GdkEventButton *event,
EDayView *day_view);
+static void e_day_view_update_calendar_selection_time (EDayView *day_view);
static gboolean e_day_view_on_main_canvas_motion (GtkWidget *widget,
GdkEventMotion *event,
EDayView *day_view);
@@ -166,6 +168,8 @@ static void e_day_view_on_event_right_click (EDayView *day_view,
gint day,
gint event_num);
+static void e_day_view_recalc_day_starts (EDayView *day_view,
+ time_t start_time);
static void e_day_view_recalc_num_rows (EDayView *day_view);
static EDayViewPosition e_day_view_convert_position_in_top_canvas (EDayView *day_view,
@@ -255,6 +259,10 @@ static void e_day_view_get_selection_range (EDayView *day_view,
static time_t e_day_view_convert_grid_position_to_time (EDayView *day_view,
gint col,
gint row);
+static gboolean e_day_view_convert_time_to_grid_position (EDayView *day_view,
+ time_t time,
+ gint *col,
+ gint *row);
static void e_day_view_check_auto_scroll (EDayView *day_view,
gint event_y);
@@ -411,16 +419,25 @@ e_day_view_init (EDayView *day_view)
day_view->need_reshape[day] = FALSE;
}
- /* FIXME: Initialize lower, upper, day_starts. */
+ /* These indicate that the times haven't been set. */
+ day_view->lower = 0;
+ day_view->upper = 0;
+
+ /* FIXME: Initialize day_starts. */
day_view->days_shown = 1;
day_view->mins_per_row = 30;
day_view->date_format = E_DAY_VIEW_DATE_FULL;
day_view->rows_in_top_display = 0;
+
+ /* Note that these don't work yet. It would need a few fixes to the
+ way event->start_minute and event->end_minute are used, and there
+ may be problems with events that go outside the visible times. */
day_view->first_hour_shown = 0;
day_view->first_minute_shown = 0;
day_view->last_hour_shown = 24;
day_view->last_minute_shown = 0;
+
day_view->main_gc = NULL;
e_day_view_recalc_num_rows (day_view);
@@ -791,6 +808,9 @@ e_day_view_destroy (GtkObject *object)
e_day_view_stop_auto_scroll (day_view);
+ if (day_view->large_font)
+ gdk_font_unref (day_view->large_font);
+
gdk_cursor_destroy (day_view->normal_cursor);
gdk_cursor_destroy (day_view->move_cursor);
gdk_cursor_destroy (day_view->resize_width_cursor);
@@ -841,7 +861,7 @@ e_day_view_style_set (GtkWidget *widget,
GdkFont *font;
gint top_rows, top_canvas_height;
gint month, max_month_width, max_abbr_month_width, number_width;
- gint hour, max_large_hour_width;
+ gint hour, max_large_hour_width, month_width;
gint minute, max_minute_width, i;
GDate date;
gchar buffer[128];
@@ -871,16 +891,17 @@ e_day_view_style_set (GtkWidget *widget,
g_date_clear (&date, 1);
g_date_set_dmy (&date, 20, 1, 2000);
max_month_width = 0;
+ max_abbr_month_width = 0;
for (month = 1; month <= 12; month++) {
g_date_set_month (&date, month);
g_date_strftime (buffer, 128, "%B", &date);
- max_month_width = MAX (max_month_width,
- gdk_string_width (font, buffer));
+ month_width = gdk_string_width (font, buffer);
+ max_month_width = MAX (max_month_width, month_width);
g_date_strftime (buffer, 128, "%b", &date);
- max_abbr_month_width = MAX (max_abbr_month_width,
- gdk_string_width (font, buffer));
+ month_width = gdk_string_width (font, buffer);
+ max_abbr_month_width = MAX (max_abbr_month_width, month_width);
}
number_width = gdk_string_width (font, "31 ");
day_view->long_format_width = number_width + max_month_width
@@ -889,6 +910,8 @@ e_day_view_style_set (GtkWidget *widget,
+ max_abbr_month_width + E_DAY_VIEW_DATE_X_PAD;
/* Calculate the widths of all the time strings necessary. */
+ day_view->max_small_hour_width = 0;
+ max_large_hour_width = 0;
for (hour = 0; hour < 24; hour++) {
sprintf (buffer, "%02i", hour);
day_view->small_hour_widths[hour] = gdk_string_width (font, buffer);
@@ -898,6 +921,7 @@ e_day_view_style_set (GtkWidget *widget,
}
day_view->max_large_hour_width = max_large_hour_width;
+ max_minute_width = 0;
for (minute = 0, i = 0; minute < 60; minute += 5, i++) {
sprintf (buffer, "%02i", minute);
day_view->minute_widths[i] = gdk_string_width (font, buffer);
@@ -918,12 +942,13 @@ e_day_view_size_allocate (GtkWidget *widget, GtkAllocation *allocation)
{
EDayView *day_view;
gfloat width, offset;
- gint col, day;
- gdouble old_width, old_height, new_width, new_height;
- gint scroll_y;
+ gint col, day, scroll_y;
+ gboolean need_reshape;
+ gdouble old_x2, old_y2, new_x2, new_y2;
+#if 0
g_print ("In e_day_view_size_allocate\n");
-
+#endif
day_view = E_DAY_VIEW (widget);
(*GTK_WIDGET_CLASS (parent_class)->size_allocate) (widget, allocation);
@@ -954,31 +979,14 @@ e_day_view_size_allocate (GtkWidget *widget, GtkAllocation *allocation)
/* Set the scroll region of the top canvas to its allocated size. */
gnome_canvas_get_scroll_region (GNOME_CANVAS (day_view->top_canvas),
- NULL, NULL, &old_width, &old_height);
- new_width = day_view->top_canvas->allocation.width;
- new_height = day_view->top_canvas->allocation.height;
- if (old_width != new_width || old_height != new_height)
+ NULL, NULL, &old_x2, &old_y2);
+ new_x2 = day_view->top_canvas->allocation.width - 1;
+ new_y2 = day_view->top_canvas->allocation.height - 1;
+ if (old_x2 != new_x2 || old_y2 != new_y2)
gnome_canvas_set_scroll_region (GNOME_CANVAS (day_view->top_canvas),
- 0, 0, new_width, new_height);
-
- /* Set the scroll region of the time canvas to its allocated width,
- but with the height the same as the main canvas. */
- gnome_canvas_get_scroll_region (GNOME_CANVAS (day_view->time_canvas),
- NULL, NULL, &old_width, &old_height);
- new_width = day_view->time_canvas->allocation.width;
- new_height = MAX (day_view->rows * day_view->row_height, day_view->main_canvas->allocation.height);
- if (old_width != new_width || old_height != new_height)
- gnome_canvas_set_scroll_region (GNOME_CANVAS (day_view->time_canvas),
- 0, 0, new_width, new_height);
+ 0, 0, new_x2, new_y2);
- /* Set the scroll region of the main canvas to its allocated width,
- but with the height depending on the number of rows needed. */
- gnome_canvas_get_scroll_region (GNOME_CANVAS (day_view->main_canvas),
- NULL, NULL, &old_width, &old_height);
- new_width = day_view->main_canvas->allocation.width;
- if (old_width != new_width || old_height != new_height)
- gnome_canvas_set_scroll_region (GNOME_CANVAS (day_view->main_canvas),
- 0, 0, new_width, new_height);
+ need_reshape = e_day_view_update_scroll_regions (day_view);
/* Scroll to the start of the working day, if this is the initial
allocation. */
@@ -991,9 +999,7 @@ e_day_view_size_allocate (GtkWidget *widget, GtkAllocation *allocation)
/* Flag that we need to reshape the events. Note that changes in height
don't matter, since the rows are always the same height. */
- if (old_width != new_width) {
- g_print ("Need reshape\n");
-
+ if (need_reshape) {
day_view->long_events_need_reshape = TRUE;
for (day = 0; day < E_DAY_VIEW_MAX_DAYS; day++)
day_view->need_reshape[day] = TRUE;
@@ -1010,8 +1016,6 @@ e_day_view_focus_in (GtkWidget *widget, GdkEventFocus *event)
g_return_val_if_fail (E_IS_DAY_VIEW (widget), FALSE);
g_return_val_if_fail (event != NULL, FALSE);
- g_print ("In e_day_view_focus_in\n");
-
GTK_WIDGET_SET_FLAGS (widget, GTK_HAS_FOCUS);
gtk_widget_draw_focus (widget);
@@ -1028,8 +1032,6 @@ e_day_view_focus_out (GtkWidget *widget, GdkEventFocus *event)
g_return_val_if_fail (E_IS_DAY_VIEW (widget), FALSE);
g_return_val_if_fail (event != NULL, FALSE);
- g_print ("In e_day_view_focus_out\n");
-
day_view = E_DAY_VIEW (widget);
GTK_WIDGET_UNSET_FLAGS (widget, GTK_HAS_FOCUS);
@@ -1048,6 +1050,8 @@ void
e_day_view_set_calendar (EDayView *day_view,
GnomeCalendar *calendar)
{
+ g_return_if_fail (E_IS_DAY_VIEW (day_view));
+
day_view->calendar = calendar;
/* FIXME: free current events? */
@@ -1063,7 +1067,13 @@ e_day_view_update_event (EDayView *day_view,
g_return_if_fail (E_IS_DAY_VIEW (day_view));
+#if 0
g_print ("In e_day_view_update_event\n");
+#endif
+
+ /* If our time hasn't been set yet, just return. */
+ if (day_view->lower == 0 && day_view->upper == 0)
+ return;
/* We only care about events. */
if (ico && ico->type != ICAL_EVENT)
@@ -1115,14 +1125,21 @@ e_day_view_update_event_label (EDayView *day_view,
{
EDayViewEvent *event;
gchar *text;
- gboolean free_text;
+ gboolean free_text = FALSE, editing_event = FALSE;
gint offset, start_minute, end_minute;
event = &g_array_index (day_view->events[day], EDayViewEvent,
event_num);
- if (event->start_minute % day_view->mins_per_row != 0
- || event->end_minute % day_view->mins_per_row != 0) {
+ text = event->ico->summary ? event->ico->summary : "";
+
+ if (day_view->editing_event_day == day
+ && day_view->editing_event_num == event_num)
+ editing_event = TRUE;
+
+ if (!editing_event
+ && (event->start_minute % day_view->mins_per_row != 0
+ || event->end_minute % day_view->mins_per_row != 0)) {
offset = day_view->first_hour_shown * 60
+ day_view->first_minute_shown;
start_minute = offset + event->start_minute;
@@ -1132,15 +1149,12 @@ e_day_view_update_event_label (EDayView *day_view,
start_minute % 60,
end_minute / 60,
end_minute % 60,
- event->ico->summary);
+ text);
free_text = TRUE;
- } else {
- text = event->ico->summary;
- free_text = FALSE;
}
gnome_canvas_item_set (event->canvas_item,
- "text", event->ico->summary ? event->ico->summary : "",
+ "text", text,
NULL);
if (free_text)
@@ -1243,51 +1257,102 @@ e_day_view_find_event_from_ico (EDayView *day_view,
}
-/* Note that the times must be the start and end of days. */
+/* This sets the selected time range. The EDayView will show the day or week
+ corresponding to the start time. If the start_time & end_time are not equal
+ and are both visible in the view, then the selection is set to those times,
+ otherwise it is set to 1 hour from the start of the working day. */
void
-e_day_view_set_interval (EDayView *day_view,
- time_t lower,
- time_t upper)
+e_day_view_set_selected_time_range (EDayView *day_view,
+ time_t start_time,
+ time_t end_time)
{
- time_t tmp_lower, day_starts[E_DAY_VIEW_MAX_DAYS + 1];
- gint day;
+ GDate date;
+ time_t lower;
+ gint start_row, start_col, end_row, end_col;
+ gboolean need_redraw = FALSE, start_in_grid, end_in_grid;
g_return_if_fail (E_IS_DAY_VIEW (day_view));
- g_print ("In e_day_view_set_interval\n");
+ /* Calculate the first day that should be shown, based on start_time
+ and the days_shown setting. If we are showing 1 day it is just the
+ start of the day given by start_time, otherwise it is the previous
+ Monday. */
+ if (day_view->days_shown == 1)
+ lower = time_day_begin (start_time);
+ else {
+ g_date_clear (&date, 1);
+ g_date_set_time (&date, start_time);
+ g_date_subtract_days (&date, g_date_weekday (&date) - 1);
+ lower = time_from_day (g_date_year (&date),
+ g_date_month (&date) - 1,
+ g_date_day (&date));
+ }
- if (lower == day_view->lower && upper == day_view->upper)
- return;
+ /* See if we need to change the days shown. */
+ if (lower != day_view->lower) {
+ e_day_view_recalc_day_starts (day_view, lower);
+ e_day_view_reload_events (day_view);
+ need_redraw = TRUE;
+ }
- /* Check that the first time is the start of a day. */
- tmp_lower = time_day_begin (lower);
- g_return_if_fail (lower == tmp_lower);
-
- /* Calculate the start of each day shown, and check that upper is
- valid. */
- day_starts[0] = lower;
- for (day = 1; day <= E_DAY_VIEW_MAX_DAYS; day++) {
- day_starts[day] = time_add_day (day_starts[day - 1], 1);
- /* Check if we have reached the upper time. */
- if (day_starts[day] == upper) {
- day_view->days_shown = day;
- break;
- }
+ /* Set the selection. */
+ start_in_grid = e_day_view_convert_time_to_grid_position (day_view,
+ start_time,
+ &start_col,
+ &start_row);
+ end_in_grid = e_day_view_convert_time_to_grid_position (day_view,
+ end_time - 60,
+ &end_col,
+ &end_row);
+
+ /* If either of the times isn't in the grid, or the selection covers
+ an entire day, we set the selection to 1 row from the start of the
+ working day, in the day corresponding to the start time. */
+ if (!start_in_grid || !end_in_grid
+ || (start_row == 0 && end_row == day_view->rows - 1)) {
+ end_col = start_col;
+
+ start_row = e_day_view_convert_time_to_row (day_view, day_view->work_day_start_hour, day_view->work_day_start_minute);
+ start_row = CLAMP (start_row, 0, day_view->rows - 1);
+ end_row = start_row;
+ }
- /* Check that we haven't gone past the upper time. */
- g_return_if_fail (day_starts[day] < upper);
+ if (start_row != day_view->selection_start_row
+ || start_col != day_view->selection_start_col) {
+ need_redraw = TRUE;
+ day_view->selection_in_top_canvas = FALSE;
+ day_view->selection_start_row = start_row;
+ day_view->selection_start_col = start_col;
+ }
+
+ if (end_row != day_view->selection_end_row
+ || end_col != day_view->selection_end_col) {
+ need_redraw = TRUE;
+ day_view->selection_in_top_canvas = FALSE;
+ day_view->selection_end_row = end_row;
+ day_view->selection_end_col = end_col;
+ }
+
+ if (need_redraw) {
+ gtk_widget_queue_draw (day_view->top_canvas);
+ gtk_widget_queue_draw (day_view->main_canvas);
}
+}
- /* Now that we know that lower & upper are valid, update the fields
- in the EDayView. */
- day_view->lower = lower;
- day_view->upper = upper;
- for (day = 0; day <= day_view->days_shown; day++) {
- day_view->day_starts[day] = day_starts[day];
+static void
+e_day_view_recalc_day_starts (EDayView *day_view,
+ time_t start_time)
+{
+ gint day;
+
+ day_view->day_starts[0] = start_time;
+ for (day = 1; day <= day_view->days_shown; day++) {
+ day_view->day_starts[day] = time_add_day (day_view->day_starts[day - 1], 1);
}
- e_day_view_reload_events (day_view);
+ day_view->lower = start_time;
+ day_view->upper = day_view->day_starts[day_view->days_shown];
}
@@ -1329,6 +1394,8 @@ void
e_day_view_set_mins_per_row (EDayView *day_view,
gint mins_per_row)
{
+ gint day;
+
g_return_if_fail (E_IS_DAY_VIEW (day_view));
if (mins_per_row != 5 && mins_per_row != 10 && mins_per_row != 15
@@ -1337,11 +1404,58 @@ e_day_view_set_mins_per_row (EDayView *day_view,
return;
}
- if (day_view->mins_per_row != mins_per_row) {
- day_view->mins_per_row = mins_per_row;
+ if (day_view->mins_per_row == mins_per_row)
+ return;
+
+ day_view->mins_per_row = mins_per_row;
+ e_day_view_recalc_num_rows (day_view);
+
+ /* If we aren't visible, we'll sort it out later. */
+ if (!GTK_WIDGET_VISIBLE (day_view))
+ return;
- /* FIXME: Update positions & display. */
+ for (day = 0; day < E_DAY_VIEW_MAX_DAYS; day++)
+ day_view->need_layout[day] = TRUE;
+
+ /* We must layout the events before updating the scroll region, since
+ that will result in a redraw which would crash otherwise. */
+ e_day_view_check_layout (day_view);
+ gtk_widget_queue_draw (day_view->time_canvas);
+ gtk_widget_queue_draw (day_view->main_canvas);
+
+ e_day_view_update_scroll_regions (day_view);
+}
+
+
+static gboolean
+e_day_view_update_scroll_regions (EDayView *day_view)
+{
+ gdouble old_x2, old_y2, new_x2, new_y2;
+ gboolean need_reshape = FALSE;
+
+ /* Set the scroll region of the time canvas to its allocated width,
+ but with the height the same as the main canvas. */
+ gnome_canvas_get_scroll_region (GNOME_CANVAS (day_view->time_canvas),
+ NULL, NULL, &old_x2, &old_y2);
+ new_x2 = day_view->time_canvas->allocation.width - 1;
+ new_y2 = MAX (day_view->rows * day_view->row_height,
+ day_view->main_canvas->allocation.height) - 1;
+ if (old_x2 != new_x2 || old_y2 != new_y2)
+ gnome_canvas_set_scroll_region (GNOME_CANVAS (day_view->time_canvas),
+ 0, 0, new_x2, new_y2);
+
+ /* Set the scroll region of the main canvas to its allocated width,
+ but with the height depending on the number of rows needed. */
+ gnome_canvas_get_scroll_region (GNOME_CANVAS (day_view->main_canvas),
+ NULL, NULL, &old_x2, &old_y2);
+ new_x2 = day_view->main_canvas->allocation.width - 1;
+ if (old_x2 != new_x2 || old_y2 != new_y2) {
+ need_reshape = TRUE;
+ gnome_canvas_set_scroll_region (GNOME_CANVAS (day_view->main_canvas),
+ 0, 0, new_x2, new_y2);
}
+
+ return need_reshape;
}
@@ -1360,7 +1474,9 @@ e_day_view_recalc_num_rows (EDayView *day_view)
}
-/* Converts an hour and minute to a row in the canvas. */
+/* Converts an hour and minute to a row in the canvas. Note that if we aren't
+ showing all 24 hours of the day, the returned row may be negative or
+ greater than day_view->rows. */
gint
e_day_view_convert_time_to_row (EDayView *day_view,
gint hour,
@@ -1372,8 +1488,10 @@ e_day_view_convert_time_to_row (EDayView *day_view,
start_minute = day_view->first_hour_shown * 60
+ day_view->first_minute_shown;
offset = total_minutes - start_minute;
-
- return offset / day_view->mins_per_row;
+ if (offset < 0)
+ return -1;
+ else
+ return offset / day_view->mins_per_row;
}
@@ -1402,8 +1520,6 @@ e_day_view_on_top_canvas_button_press (GtkWidget *widget,
gint event_x, event_y, scroll_x, scroll_y, day, event_num;
EDayViewPosition pos;
- g_print ("In e_day_view_on_top_canvas_button_press\n");
-
/* Convert the coords to the main canvas window, or return if the
window is not found. */
if (!e_day_view_convert_event_coords (day_view, (GdkEvent*) event,
@@ -1503,7 +1619,7 @@ e_day_view_convert_event_coords (EDayView *day_view,
*y_return = event_y;
if (event_window != window)
- g_print ("Couldn't find event window\n");
+ g_warning ("Couldn't find event window\n");
return (event_window == window) ? TRUE : FALSE;
}
@@ -1517,8 +1633,6 @@ e_day_view_on_main_canvas_button_press (GtkWidget *widget,
gint event_x, event_y, scroll_x, scroll_y, row, day, event_num;
EDayViewPosition pos;
- g_print ("In e_day_view_on_main_canvas_button_press\n");
-
/* Convert the coords to the main canvas window, or return if the
window is not found. */
if (!e_day_view_convert_event_coords (day_view, (GdkEvent*) event,
@@ -1645,8 +1759,6 @@ e_day_view_on_long_event_click (EDayView *day_view,
gint start_day, end_day, day;
gint item_x, item_y, item_w, item_h;
- g_print ("In e_day_view_on_long_event_click\n");
-
event = &g_array_index (day_view->long_events, EDayViewEvent,
event_num);
@@ -1702,7 +1814,6 @@ e_day_view_on_long_event_click (EDayView *day_view,
event_x, event_y,
&day, NULL);
day_view->drag_event_offset = day - start_day;
- g_print ("Y offset: %i\n", day_view->drag_event_offset);
}
}
@@ -1719,8 +1830,6 @@ e_day_view_on_event_click (EDayView *day_view,
EDayViewEvent *event;
gint tmp_day, row, start_row;
- g_print ("In e_day_view_on_event_click\n");
-
event = &g_array_index (day_view->events[day], EDayViewEvent,
event_num);
@@ -1778,8 +1887,6 @@ e_day_view_on_event_click (EDayView *day_view,
NULL);
start_row = event->start_minute / day_view->mins_per_row;
day_view->drag_event_offset = row - start_row;
- g_print ("Y offset: %i Row: %i Start: %i\n",
- day_view->drag_event_offset, row, start_row);
}
}
@@ -1868,7 +1975,9 @@ e_day_view_on_event_double_click (EDayView *day_view,
gint day,
gint event_num)
{
+#if 0
g_print ("In e_day_view_on_event_double_click\n");
+#endif
}
@@ -1903,8 +2012,6 @@ e_day_view_on_event_right_click (EDayView *day_view,
{ N_("New appointment..."), (GtkSignalFunc) e_day_view_on_new_appointment, NULL, TRUE }
};
- g_print ("In e_day_view_on_event_right_click\n");
-
have_selection = (day_view->selection_start_col != -1);
if (event_num == -1) {
@@ -2067,12 +2174,10 @@ e_day_view_on_top_canvas_button_release (GtkWidget *widget,
GdkEventButton *event,
EDayView *day_view)
{
-
- g_print ("In e_day_view_on_top_canvas_button_release\n");
-
if (day_view->selection_drag_pos != E_DAY_VIEW_DRAG_NONE) {
day_view->selection_drag_pos = E_DAY_VIEW_DRAG_NONE;
gdk_pointer_ungrab (event->time);
+ e_day_view_update_calendar_selection_time (day_view);
} else if (day_view->resize_drag_pos != E_DAY_VIEW_POS_NONE) {
e_day_view_finish_long_event_resize (day_view);
gdk_pointer_ungrab (event->time);
@@ -2094,13 +2199,11 @@ e_day_view_on_main_canvas_button_release (GtkWidget *widget,
GdkEventButton *event,
EDayView *day_view)
{
-
- g_print ("In e_day_view_on_main_canvas_button_release\n");
-
if (day_view->selection_drag_pos != E_DAY_VIEW_DRAG_NONE) {
day_view->selection_drag_pos = E_DAY_VIEW_DRAG_NONE;
gdk_pointer_ungrab (event->time);
e_day_view_stop_auto_scroll (day_view);
+ e_day_view_update_calendar_selection_time (day_view);
} else if (day_view->resize_drag_pos != E_DAY_VIEW_POS_NONE) {
e_day_view_finish_resize (day_view);
gdk_pointer_ungrab (event->time);
@@ -2118,6 +2221,17 @@ e_day_view_on_main_canvas_button_release (GtkWidget *widget,
}
+static void
+e_day_view_update_calendar_selection_time (EDayView *day_view)
+{
+ time_t start, end;
+
+ e_day_view_get_selection_range (day_view, &start, &end);
+ gnome_calendar_set_selected_time_range (day_view->calendar,
+ start, end);
+}
+
+
static gboolean
e_day_view_on_top_canvas_motion (GtkWidget *widget,
GdkEventMotion *mevent,
@@ -2160,8 +2274,6 @@ e_day_view_on_top_canvas_motion (GtkWidget *widget,
} else if (day_view->pressed_event_day == E_DAY_VIEW_LONG_EVENT) {
GtkTargetList *target_list;
- g_print ("Checking whether to start drag - Pressed %i,%i Canvas: %i,%i\n", day_view->drag_event_x, day_view->drag_event_y, canvas_x, canvas_y);
-
if (abs (canvas_x - day_view->drag_event_x) > E_DAY_VIEW_DRAG_START_OFFSET
|| abs (canvas_y - day_view->drag_event_y) > E_DAY_VIEW_DRAG_START_OFFSET) {
day_view->drag_event_day = day_view->pressed_event_day;
@@ -2246,8 +2358,6 @@ e_day_view_on_main_canvas_motion (GtkWidget *widget,
} else if (day_view->pressed_event_day != -1) {
GtkTargetList *target_list;
- g_print ("Checking whether to start drag - Pressed %i,%i Canvas: %i,%i\n", day_view->drag_event_x, day_view->drag_event_y, canvas_x, canvas_y);
-
if (abs (canvas_x - day_view->drag_event_x) > E_DAY_VIEW_DRAG_START_OFFSET
|| abs (canvas_y - day_view->drag_event_y) > E_DAY_VIEW_DRAG_START_OFFSET) {
day_view->drag_event_day = day_view->pressed_event_day;
@@ -2347,7 +2457,7 @@ e_day_view_update_long_event_resize (EDayView *day_view,
gint event_num;
gboolean need_reshape = FALSE;
-#if 1
+#if 0
g_print ("Updating resize Day:%i\n", day);
#endif
@@ -2429,9 +2539,6 @@ e_day_view_finish_long_event_resize (EDayView *day_view)
EDayViewEvent *event;
gint event_num;
-
- g_print ("In e_day_view_finish_long_event_resize\n");
-
event_num = day_view->resize_event_num;
event = &g_array_index (day_view->long_events, EDayViewEvent,
event_num);
@@ -2460,9 +2567,6 @@ e_day_view_finish_resize (EDayView *day_view)
EDayViewEvent *event;
gint day, event_num;
-
- g_print ("In e_day_view_finish_resize\n");
-
day = day_view->resize_event_day;
event_num = day_view->resize_event_num;
event = &g_array_index (day_view->events[day], EDayViewEvent,
@@ -2478,7 +2582,6 @@ e_day_view_finish_resize (EDayView *day_view)
gnome_canvas_item_hide (day_view->resize_bar_item);
/* Hide the horizontal bars. */
- g_print ("Hiding resize bars\n");
day_view->resize_bars_event_day = -1;
day_view->resize_bars_event_num = -1;
gnome_canvas_item_hide (day_view->main_canvas_top_resize_bar_item);
@@ -2501,8 +2604,6 @@ e_day_view_abort_resize (EDayView *day_view,
if (day_view->resize_drag_pos == E_DAY_VIEW_POS_NONE)
return;
- g_print ("In e_day_view_abort_resize\n");
-
day_view->resize_drag_pos = E_DAY_VIEW_POS_NONE;
gdk_pointer_ungrab (time);
@@ -2552,7 +2653,11 @@ e_day_view_reload_events (EDayView *day_view)
day_view);
}
+ /* We need to do this to make sure the top canvas is resized. */
+ day_view->long_events_need_layout = TRUE;
+
e_day_view_check_layout (day_view);
+ e_day_view_reshape_main_canvas_resize_bars (day_view);
gtk_widget_queue_draw (day_view->top_canvas);
gtk_widget_queue_draw (day_view->main_canvas);
@@ -2598,7 +2703,7 @@ e_day_view_add_event (iCalObject *ico,
{
EDayView *day_view;
EDayViewEvent event;
- gint day;
+ gint day, offset;
struct tm start_tm, end_tm;
day_view = E_DAY_VIEW (data);
@@ -2616,10 +2721,12 @@ e_day_view_add_event (iCalObject *ico,
event.end = end;
event.canvas_item = NULL;
- /* Calculate the start & end minute, relative to the
- top of the display. FIXME. */
- event.start_minute = start_tm.tm_hour * 60 + start_tm.tm_min;
- event.end_minute = end_tm.tm_hour * 60 + end_tm.tm_min;
+ /* Calculate the start & end minute, relative to the top of the
+ display. */
+ offset = day_view->first_hour_shown * 60
+ + day_view->first_minute_shown;
+ event.start_minute = start_tm.tm_hour * 60 + start_tm.tm_min - offset;
+ event.end_minute = end_tm.tm_hour * 60 + end_tm.tm_min - offset;
event.start_row_or_col = -1;
event.num_columns = -1;
@@ -2810,11 +2917,12 @@ e_day_view_reshape_long_event (EDayView *day_view,
gint event_num)
{
EDayViewEvent *event;
+ GdkFont *font;
gint start_day, end_day, item_x, item_y, item_w, item_h;
gint text_x, text_w, num_icons, icons_width, width, time_width;
iCalObject *ico;
- gint min_text_x, max_text_w;
- gdouble text_width;
+ gint min_text_x, max_text_w, text_width, line_len;
+ gchar *text, *end_of_line;
gboolean show_icons = TRUE, use_max_width = FALSE;
if (!e_day_view_get_long_event_position (day_view, event_num,
@@ -2840,6 +2948,7 @@ e_day_view_reshape_long_event (EDayView *day_view,
draw them on top of the resize rect. Nor when editing. */
num_icons = 0;
ico = event->ico;
+ font = GTK_WIDGET (day_view)->style->font;
if (day_view->resize_drag_pos != E_DAY_VIEW_POS_NONE
&& day_view->resize_event_day == E_DAY_VIEW_LONG_EVENT
@@ -2848,7 +2957,6 @@ e_day_view_reshape_long_event (EDayView *day_view,
if (day_view->editing_event_day == E_DAY_VIEW_LONG_EVENT
&& day_view->editing_event_num == event_num) {
- g_print ("Reshaping long event which is being edited.\n");
show_icons = FALSE;
use_max_width = TRUE;
}
@@ -2894,8 +3002,18 @@ e_day_view_reshape_long_event (EDayView *day_view,
} else {
/* Get the requested size of the label. */
gtk_object_get (GTK_OBJECT (event->canvas_item),
- "text_width", &text_width,
+ "text", &text,
NULL);
+ text_width = 0;
+ if (text) {
+ end_of_line = strchr (text, '\n');
+ if (end_of_line)
+ line_len = end_of_line - text;
+ else
+ line_len = strlen (text);
+ text_width = gdk_text_width (font, text, line_len);
+ g_free (text);
+ }
width = text_width + icons_width;
text_x = item_x + (item_w - width) / 2;
@@ -3033,6 +3151,16 @@ e_day_view_layout_day_event (EDayView *day_view,
start_row = event->start_minute / day_view->mins_per_row;
end_row = (event->end_minute - 1) / day_view->mins_per_row;
+ event->num_columns = 0;
+
+ /* If the event can't currently be seen, just return. */
+ if (start_row >= day_view->rows || end_row < 0)
+ return;
+
+ /* Make sure we don't go outside the visible times. */
+ start_row = CLAMP (start_row, 0, day_view->rows - 1);
+ end_row = CLAMP (end_row, 0, day_view->rows - 1);
+
/* Try each column until we find a free one. */
for (col = 0; col < E_DAY_VIEW_MAX_COLUMNS; col++) {
free_col = col;
@@ -3047,11 +3175,9 @@ e_day_view_layout_day_event (EDayView *day_view,
break;
}
- /* If we can't find space for the event, mark it as not displayed. */
- if (free_col == -1) {
- event->num_columns = 0;
+ /* If we can't find space for the event, just return. */
+ if (free_col == -1)
return;
- }
/* The event is assigned 1 col initially, but may be expanded later. */
event->start_row_or_col = free_col;
@@ -3143,8 +3269,6 @@ e_day_view_reshape_day_events (EDayView *day_view,
{
gint event_num;
- g_print ("In e_day_view_reshape_day_events\n");
-
for (event_num = 0; event_num < day_view->events[day]->len;
event_num++) {
e_day_view_reshape_day_event (day_view, day, event_num);
@@ -3353,8 +3477,6 @@ e_day_view_key_press (GtkWidget *widget, GdkEventKey *event)
day_view = E_DAY_VIEW (widget);
- g_print ("In e_day_view_key_press\n");
-
/* The Escape key aborts a resize operation. */
if (day_view->resize_drag_pos != E_DAY_VIEW_POS_NONE) {
if (event->keyval == GDK_Escape) {
@@ -3409,6 +3531,8 @@ e_day_view_start_editing_event (EDayView *day_view,
gchar *initial_text)
{
EDayViewEvent *event;
+ ETextEventProcessor *event_processor = NULL;
+ ETextEventProcessorCommand command;
/* If we are already editing the event, just return. */
if (day == day_view->editing_event_day
@@ -3427,13 +3551,28 @@ e_day_view_start_editing_event (EDayView *day_view,
if (!event->canvas_item)
return;
+ /* We must grab the focus before setting the initial text, since
+ grabbing the focus will result in a call to
+ e_day_view_on_editing_started(), which will reset the text to get
+ rid of the start and end times. */
+ e_canvas_item_grab_focus (event->canvas_item);
+
if (initial_text) {
gnome_canvas_item_set (event->canvas_item,
"text", initial_text,
NULL);
}
- e_canvas_item_grab_focus (event->canvas_item);
+ /* Try to move the cursor to the end of the text. */
+ gtk_object_get (GTK_OBJECT (event->canvas_item),
+ "event_processor", &event_processor,
+ NULL);
+ if (event_processor) {
+ command.action = E_TEP_MOVE;
+ command.position = E_TEP_END_OF_BUFFER;
+ gtk_signal_emit_by_name (GTK_OBJECT (event_processor),
+ "command", &command);
+ }
}
@@ -3493,8 +3632,18 @@ e_day_view_on_editing_started (EDayView *day_view,
&day, &event_num))
return;
+#if 0
g_print ("In e_day_view_on_editing_started Day:%i Event:%i\n",
day, event_num);
+#endif
+
+ /* FIXME: This is a temporary workaround for a bug which seems to stop
+ us getting focus_out signals. It is not a complete fix since if we
+ don't get focus_out signals we don't save the appointment text so
+ this may be lost. */
+ if (day_view->editing_event_day == day
+ && day_view->editing_event_num == event_num)
+ return;
day_view->editing_event_day = day;
day_view->editing_event_num = event_num;
@@ -3504,6 +3653,7 @@ e_day_view_on_editing_started (EDayView *day_view,
} else {
day_view->resize_bars_event_day = day;
day_view->resize_bars_event_num = event_num;
+ e_day_view_update_event_label (day_view, day, event_num);
e_day_view_reshape_main_canvas_resize_bars (day_view);
}
}
@@ -3528,8 +3678,10 @@ e_day_view_on_editing_stopped (EDayView *day_view,
if (day == -1)
return;
+#if 0
g_print ("In e_day_view_on_editing_stopped Day:%i Event:%i\n",
day, event_num);
+#endif
if (day == E_DAY_VIEW_LONG_EVENT) {
editing_long_event = TRUE;
@@ -3631,13 +3783,51 @@ e_day_view_convert_grid_position_to_time (EDayView *day_view,
}
+static gboolean
+e_day_view_convert_time_to_grid_position (EDayView *day_view,
+ time_t time,
+ gint *col,
+ gint *row)
+{
+ struct tm *tmp_tm;
+ gint day, minutes;
+
+ *col = *row = 0;
+
+ if (time < day_view->lower || time >= day_view->upper)
+ return FALSE;
+
+ /* We can find the column easily using the day_starts array. */
+ for (day = 1; day <= day_view->days_shown; day++) {
+ if (time < day_view->day_starts[day]) {
+ *col = day - 1;
+ break;
+ }
+ }
+
+ /* To find the row we need to convert the time to a struct tm,
+ calculate the offset in minutes from the top of the display and
+ divide it by the mins per row setting. */
+ tmp_tm = localtime (&time);
+ minutes = tmp_tm->tm_hour * 60 + tmp_tm->tm_min;
+ minutes -= day_view->first_hour_shown * 60
+ + day_view->first_minute_shown;
+
+ *row = minutes / day_view->mins_per_row;
+
+ if (*row < 0 || *row >= day_view->rows)
+ return FALSE;
+
+ return TRUE;
+}
+
+
/* This starts or stops auto-scrolling when dragging a selection or resizing
an event. */
static void
e_day_view_check_auto_scroll (EDayView *day_view,
gint event_y)
{
- g_print ("Event Y:%i\n", event_y);
if (event_y < E_DAY_VIEW_AUTO_SCROLL_OFFSET)
e_day_view_start_auto_scroll (day_view, TRUE);
else if (event_y >= day_view->main_canvas->allocation.height
@@ -3995,8 +4185,6 @@ e_day_view_on_top_canvas_drag_motion (GtkWidget *widget,
{
gint scroll_x, scroll_y;
- g_print ("In e_day_view_on_top_canvas_drag_motion\n");
-
gnome_canvas_get_scroll_offsets (GNOME_CANVAS (widget),
&scroll_x, &scroll_y);
day_view->drag_event_x = x + scroll_x;
@@ -4084,8 +4272,6 @@ e_day_view_update_top_canvas_drag (EDayView *day_view,
item_h = day_view->top_row_height - E_DAY_VIEW_TOP_CANVAS_Y_GAP;
- g_print ("Moving to %g,%g %gx%g\n", item_x, item_y, item_w, item_h);
-
/* Set the positions of the event & associated items. */
gnome_canvas_item_set (day_view->drag_long_event_rect_item,
"x1", item_x,
@@ -4137,8 +4323,6 @@ e_day_view_on_main_canvas_drag_motion (GtkWidget *widget,
{
gint scroll_x, scroll_y;
- g_print ("In e_day_view_on_main_canvas_drag_motion\n");
-
day_view->last_mouse_x = x;
day_view->last_mouse_y = y;
@@ -4233,8 +4417,6 @@ e_day_view_update_main_canvas_drag (EDayView *day_view,
item_y = row * day_view->row_height;
item_h = num_rows * day_view->row_height;
- g_print ("Moving to %g,%g %gx%g\n", item_x, item_y, item_w, item_h);
-
/* Set the positions of the event & associated items. */
gnome_canvas_item_set (day_view->drag_rect_item,
"x1", item_x + E_DAY_VIEW_BAR_WIDTH - 1,
@@ -4293,8 +4475,6 @@ e_day_view_on_top_canvas_drag_leave (GtkWidget *widget,
guint time,
EDayView *day_view)
{
- g_print ("In e_day_view_on_top_canvas_drag_leave\n");
-
day_view->drag_last_day = -1;
gnome_canvas_item_hide (day_view->drag_long_event_rect_item);
@@ -4308,8 +4488,6 @@ e_day_view_on_main_canvas_drag_leave (GtkWidget *widget,
guint time,
EDayView *day_view)
{
- g_print ("In e_day_view_on_main_canvas_drag_leave\n");
-
day_view->drag_last_day = -1;
e_day_view_stop_auto_scroll (day_view);
@@ -4335,8 +4513,6 @@ e_day_view_on_drag_begin (GtkWidget *widget,
EDayViewEvent *event;
gint day, event_num;
- g_print ("In e_day_view_on_main_canvas_drag_begin\n");
-
day = day_view->drag_event_day;
event_num = day_view->drag_event_num;
@@ -4365,8 +4541,6 @@ e_day_view_on_drag_end (GtkWidget *widget,
EDayViewEvent *event;
gint day, event_num;
- g_print ("In e_day_view_on_main_canvas_drag_end\n");
-
day = day_view->drag_event_day;
event_num = day_view->drag_event_num;
@@ -4405,8 +4579,6 @@ e_day_view_on_drag_data_get (GtkWidget *widget,
gint day, event_num;
gchar *event_uid;
- g_print ("In e_day_view_on_drag_data_get\n");
-
day = day_view->drag_event_day;
event_num = day_view->drag_event_num;
@@ -4443,11 +4615,9 @@ e_day_view_on_top_canvas_drag_data_received (GtkWidget *widget,
{
EDayViewEvent *event;
EDayViewPosition pos;
- gint day, /* row, scroll_x, scroll_y,*/ start_day, end_day, num_days;
+ gint day, start_day, end_day, num_days;
gchar *event_uid;
- g_print ("In e_day_view_on_top_canvas_drag_data_received\n");
-
if ((data->length >= 0) && (data->format == 8)) {
pos = e_day_view_convert_position_in_top_canvas (day_view,
x, y, &day,
@@ -4475,8 +4645,6 @@ e_day_view_on_top_canvas_drag_data_received (GtkWidget *widget,
event_uid = data->data;
- g_print ("Dropped Day:%i UID:%s\n", day, event_uid);
-
if (!event_uid || !event->ico->uid
|| strcmp (event_uid, event->ico->uid))
g_warning ("Unexpected event UID");
@@ -4517,8 +4685,6 @@ e_day_view_on_main_canvas_drag_data_received (GtkWidget *widget,
gint day, row, start_row, end_row, num_rows, scroll_x, scroll_y;
gchar *event_uid;
- g_print ("In e_day_view_on_main_canvas_drag_data_received\n");
-
gnome_canvas_get_scroll_offsets (GNOME_CANVAS (widget),
&scroll_x, &scroll_y);
x += scroll_x;
@@ -4547,9 +4713,6 @@ e_day_view_on_main_canvas_drag_data_received (GtkWidget *widget,
event_uid = data->data;
- g_print ("Dropped Day:%i Row:%i UID:%s\n", day, row,
- event_uid);
-
if (!event_uid || !event->ico->uid
|| strcmp (event_uid, event->ico->uid))
g_warning ("Unexpected event UID");