4 * Copyright (C) 2008 - 2012 The Geeqie Team
8 * This software is released under the GNU General Public License (GNU GPL).
9 * Please read the included file COPYING for more information.
10 * This software comes with no warranty of any kind, use at your own risk!
15 #include "pan-types.h"
17 #include <glib/gprintf.h>
21 #define PAN_CAL_POPUP_COLOR 220, 220, 220
22 #define PAN_CAL_POPUP_ALPHA 255
23 #define PAN_CAL_POPUP_BORDER 1
24 #define PAN_CAL_POPUP_BORDER_COLOR 0, 0, 0
25 #define PAN_CAL_POPUP_TEXT_COLOR 0, 0, 0
27 #define PAN_CAL_DAY_WIDTH 100
28 #define PAN_CAL_DAY_HEIGHT 80
30 #define PAN_CAL_DAY_COLOR 255, 255, 255
31 #define PAN_CAL_DAY_ALPHA 220
32 #define PAN_CAL_DAY_BORDER 2
33 #define PAN_CAL_DAY_BORDER_COLOR 0, 0, 0
34 #define PAN_CAL_DAY_TEXT_COLOR 0, 0, 0
36 #define PAN_CAL_MONTH_COLOR 255, 255, 255
37 #define PAN_CAL_MONTH_ALPHA 200
38 #define PAN_CAL_MONTH_BORDER 4
39 #define PAN_CAL_MONTH_BORDER_COLOR 0, 0, 0
40 #define PAN_CAL_MONTH_TEXT_COLOR 0, 0, 0
42 #define PAN_CAL_DOT_SIZE 3
43 #define PAN_CAL_DOT_GAP 2
44 #define PAN_CAL_DOT_COLOR 128, 128, 128
45 #define PAN_CAL_DOT_ALPHA 128
49 *-----------------------------------------------------------------------------
51 *-----------------------------------------------------------------------------
54 void pan_calendar_update(PanWindow *pw, PanItem *pi_day)
60 gint x1, y1, x2, y2, x3, y3;
65 while ((pi = pan_item_find_by_key(pw, PAN_ITEM_NONE, "day_bubble"))) pan_item_remove(pw, pi);
67 if (!pi_day || pi_day->type != PAN_ITEM_BOX ||
68 !pi_day->key || strcmp(pi_day->key, "day") != 0) return;
70 list = pan_layout_intersect(pw, pi_day->x, pi_day->y, pi_day->width, pi_day->height);
82 if (dot->type != PAN_ITEM_BOX || !dot->fd ||
83 !dot->key || strcmp(dot->key, "dot") != 0)
85 list = g_list_delete_link(list, node);
93 grid = (gint)(sqrt(g_list_length(list)) + 0.5);
95 x = pi_day->x + pi_day->width + 4;
99 if (y + grid * (PAN_THUMB_SIZE + PAN_THUMB_GAP) + PAN_BOX_BORDER * 4 > pw->pr->image_height)
101 y = pw->pr->image_height - (grid * (PAN_THUMB_SIZE + PAN_THUMB_GAP) + PAN_BOX_BORDER * 4);
105 pbox = pan_item_box_new(pw, NULL, x, y, PAN_BOX_BORDER, PAN_BOX_BORDER,
106 PAN_CAL_POPUP_BORDER,
107 PAN_CAL_POPUP_COLOR, PAN_CAL_POPUP_ALPHA,
108 PAN_CAL_POPUP_BORDER_COLOR, PAN_CAL_POPUP_ALPHA);
109 pan_item_set_key(pbox, "day_bubble");
116 buf = pan_date_value_string(pi_day->fd->date, PAN_DATE_LENGTH_WEEK);
117 plabel = pan_item_text_new(pw, x, y, buf, PAN_TEXT_ATTR_BOLD | PAN_TEXT_ATTR_HEADING,
118 PAN_TEXT_BORDER_SIZE,
119 PAN_CAL_POPUP_TEXT_COLOR, 255);
120 pan_item_set_key(plabel, "day_bubble");
123 pan_item_size_by_item(pbox, plabel, 0);
147 pimg = pan_item_thumb_new(pw, file_data_ref(dot->fd), x, y);
148 pan_item_set_key(pimg, "day_bubble");
150 pan_item_size_by_item(pbox, pimg, PAN_BOX_BORDER);
155 x += PAN_THUMB_SIZE + PAN_THUMB_GAP;
160 x = pbox->x + PAN_BOX_BORDER;
161 y += PAN_THUMB_SIZE + PAN_THUMB_GAP;
167 x1 = pi_day->x + pi_day->width - 8;
170 y2 = pbox->y + MIN(42, pbox->height);
172 y3 = MAX(pbox->y, y2 - 30);
173 util_clip_triangle(x1, y1, x2, y2, x3, y3,
176 pi = pan_item_tri_new(pw, NULL, x, y, w, h,
177 x1, y1, x2, y2, x3, y3,
178 PAN_CAL_POPUP_COLOR, PAN_CAL_POPUP_ALPHA);
179 pan_item_tri_border(pi, PAN_BORDER_1 | PAN_BORDER_3, PAN_CAL_POPUP_BORDER_COLOR, PAN_CAL_POPUP_ALPHA);
180 pan_item_set_key(pi, "day_bubble");
181 pan_item_added(pw, pi);
183 pan_item_box_shadow(pbox, PAN_SHADOW_OFFSET * 2, PAN_SHADOW_FADE * 2);
184 pan_item_added(pw, pbox);
186 pan_layout_resize(pw);
189 void pan_calendar_compute(PanWindow *pw, FileData *dir_fd, gint *width, gint *height)
205 list = pan_list_tree(dir_fd, SORT_NONE, TRUE, pw->ignore_symlinks);
207 if (pw->cache_list && pw->exif_date_enable)
209 pw->cache_list = pan_cache_sort(pw->cache_list, SORT_NAME, TRUE);
210 list = filelist_sort(list, SORT_NAME, TRUE);
211 pan_cache_sync_date(pw, list);
214 pw->cache_list = pan_cache_sort(pw->cache_list, SORT_TIME, TRUE);
215 list = filelist_sort(list, SORT_TIME, TRUE);
228 if (!pan_date_compare(fd->date, tc, PAN_DATE_LENGTH_DAY))
236 if (day_max < count) day_max = count;
240 DEBUG_1("biggest day contains %d images", day_max);
242 grid = (gint)(sqrt((gdouble)day_max) + 0.5) * (PAN_THUMB_SIZE + PAN_SHADOW_OFFSET * 2 + PAN_THUMB_GAP);
243 day_width = MAX(PAN_CAL_DAY_WIDTH, grid);
244 day_height = MAX(PAN_CAL_DAY_HEIGHT, grid);
248 FileData *fd = list->data;
250 year = pan_date_value(fd->date, PAN_DATE_LENGTH_YEAR);
251 month = pan_date_value(fd->date, PAN_DATE_LENGTH_MONTH);
254 work = g_list_last(list);
257 FileData *fd = work->data;
258 end_year = pan_date_value(fd->date, PAN_DATE_LENGTH_YEAR);
259 end_month = pan_date_value(fd->date, PAN_DATE_LENGTH_MONTH);
262 *width = PAN_BOX_BORDER * 2;
263 *height = PAN_BOX_BORDER * 2;
269 while (work && (year < end_year || (year == end_year && month <= end_month)))
280 /* figure last second of this month */
281 dt = pan_date_to_time((month == 12) ? year + 1 : year, (month == 12) ? 1 : month + 1, 1);
284 /* anything to show this month? */
285 if (!pan_date_compare(((FileData *)(work->data))->date, dt, PAN_DATE_LENGTH_MONTH))
296 days = pan_date_value(dt, PAN_DATE_LENGTH_DAY);
297 dt = pan_date_to_time(year, month, 1);
298 col = pan_date_value(dt, PAN_DATE_LENGTH_WEEK);
303 pi_month = pan_item_box_new(pw, NULL, x, y, PAN_CAL_DAY_WIDTH * 7, PAN_CAL_DAY_HEIGHT / 4,
304 PAN_CAL_MONTH_BORDER,
305 PAN_CAL_MONTH_COLOR, PAN_CAL_MONTH_ALPHA,
306 PAN_CAL_MONTH_BORDER_COLOR, PAN_CAL_MONTH_ALPHA);
307 buf = pan_date_value_string(dt, PAN_DATE_LENGTH_MONTH);
308 pi_text = pan_item_text_new(pw, x, y, buf,
309 PAN_TEXT_ATTR_BOLD | PAN_TEXT_ATTR_HEADING,
310 PAN_TEXT_BORDER_SIZE,
311 PAN_CAL_MONTH_TEXT_COLOR, 255);
313 pi_text->x = pi_month->x + (pi_month->width - pi_text->width) / 2;
315 pi_month->height = pi_text->y + pi_text->height - pi_month->y;
317 x = PAN_BOX_BORDER + col * PAN_CAL_DAY_WIDTH;
318 y = pi_month->y + pi_month->height + PAN_BOX_BORDER;
320 for (day = 1; day <= days; day++)
328 dt = pan_date_to_time(year, month, day);
331 * Create a FileData entry that represents the given day.
332 * It does not correspond to any real file
335 g_snprintf(fake_path, sizeof(fake_path), "//%04d-%02d-%02d", year, month, day);
336 fd = file_data_new_no_grouping(fake_path);
338 pi_day = pan_item_box_new(pw, fd, x, y, PAN_CAL_DAY_WIDTH, PAN_CAL_DAY_HEIGHT,
340 PAN_CAL_DAY_COLOR, PAN_CAL_DAY_ALPHA,
341 PAN_CAL_DAY_BORDER_COLOR, PAN_CAL_DAY_ALPHA);
342 pan_item_set_key(pi_day, "day");
344 dx = x + PAN_CAL_DOT_GAP * 2;
345 dy = y + PAN_CAL_DOT_GAP * 2;
347 fd = (work) ? work->data : NULL;
348 while (fd && pan_date_compare(fd->date, dt, PAN_DATE_LENGTH_DAY))
352 pi = pan_item_box_new(pw, fd, dx, dy, PAN_CAL_DOT_SIZE, PAN_CAL_DOT_SIZE,
354 PAN_CAL_DOT_COLOR, PAN_CAL_DOT_ALPHA,
356 pan_item_set_key(pi, "dot");
358 dx += PAN_CAL_DOT_SIZE + PAN_CAL_DOT_GAP;
359 if (dx + PAN_CAL_DOT_SIZE > pi_day->x + pi_day->width - PAN_CAL_DOT_GAP * 2)
361 dx = x + PAN_CAL_DOT_GAP * 2;
362 dy += PAN_CAL_DOT_SIZE + PAN_CAL_DOT_GAP;
364 if (dy + PAN_CAL_DOT_SIZE > pi_day->y + pi_day->height - PAN_CAL_DOT_GAP * 2)
366 /* must keep all dots within respective day even if it gets ugly */
367 dy = y + PAN_CAL_DOT_GAP * 2;
373 fd = (work) ? work->data : NULL;
380 pi_day->color_r = MAX(pi_day->color_r - 61 - n * 3, 80);
381 pi_day->color_g = pi_day->color_r;
383 buf = g_strdup_printf("( %d )", n);
384 pi = pan_item_text_new(pw, x, y, buf, PAN_TEXT_ATTR_NONE,
385 PAN_TEXT_BORDER_SIZE,
386 PAN_CAL_DAY_TEXT_COLOR, 255);
389 pi->x = pi_day->x + (pi_day->width - pi->width) / 2;
390 pi->y = pi_day->y + (pi_day->height - pi->height) / 2;
393 buf = g_strdup_printf("%d", day);
394 pan_item_text_new(pw, x + 4, y + 4, buf, PAN_TEXT_ATTR_BOLD | PAN_TEXT_ATTR_HEADING,
395 PAN_TEXT_BORDER_SIZE,
396 PAN_CAL_DAY_TEXT_COLOR, 255);
400 pan_item_size_coordinates(pi_day, PAN_BOX_BORDER, width, height);
408 y += PAN_CAL_DAY_HEIGHT;
412 x += PAN_CAL_DAY_WIDTH;
416 if (col > 0) y += PAN_CAL_DAY_HEIGHT;
417 y += PAN_BOX_BORDER * 2;
428 *height = MAX(*height, grid + PAN_BOX_BORDER * 2 * 2);
432 /* vim: set shiftwidth=8 softtabstop=0 cindent cinoptions={1s: */