2 * Copyright (C) 2006 John Ellis
3 * Copyright (C) 2008 - 2016 The Geeqie Team
7 * This program is free software; you can redistribute it and/or modify
8 * it under the terms of the GNU General Public License as published by
9 * the Free Software Foundation; either version 2 of the License, or
10 * (at your option) any later version.
12 * This program is distributed in the hope that it will be useful,
13 * but WITHOUT ANY WARRANTY; without even the implied warranty of
14 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
15 * GNU General Public License for more details.
17 * You should have received a copy of the GNU General Public License along
18 * with this program; if not, write to the Free Software Foundation, Inc.,
19 * 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA.
22 #include "pan-calendar.h"
32 #include "pan-types.h"
34 #include "pan-view-filter.h"
36 #include "pixbuf-util.h"
42 constexpr guint8 PAN_CAL_DOT_ALPHA = 128;
46 #define PAN_CAL_POPUP_COLOR 220, 220, 220
48 PAN_CAL_POPUP_ALPHA = 255,
49 PAN_CAL_POPUP_BORDER = 1
51 #define PAN_CAL_POPUP_BORDER_COLOR 0, 0, 0
52 #define PAN_CAL_POPUP_TEXT_COLOR 0, 0, 0
55 PAN_CAL_DAY_WIDTH = 100,
56 PAN_CAL_DAY_HEIGHT = 80
59 #define PAN_CAL_DAY_COLOR 255, 255, 255
61 PAN_CAL_DAY_ALPHA = 220,
62 PAN_CAL_DAY_BORDER = 2
64 #define PAN_CAL_DAY_BORDER_COLOR 0, 0, 0
65 #define PAN_CAL_DAY_TEXT_COLOR 0, 0, 0
67 #define PAN_CAL_MONTH_COLOR 255, 255, 255
69 PAN_CAL_MONTH_ALPHA = 200,
70 PAN_CAL_MONTH_BORDER = 4
72 #define PAN_CAL_MONTH_BORDER_COLOR 0, 0, 0
73 #define PAN_CAL_MONTH_TEXT_COLOR 0, 0, 0
79 #define PAN_CAL_DOT_COLOR 128, 128, 128
81 #define PAN_CAL_DAY_OF_WEEK_COLOR 128, 128, 128
84 *-----------------------------------------------------------------------------
86 *-----------------------------------------------------------------------------
89 void pan_calendar_update(PanWindow *pw, PanItem *pi_day)
108 while ((pi = pan_item_find_by_key(pw, PAN_ITEM_NONE, "day_bubble"))) pan_item_remove(pw, pi);
110 if (!pi_day || pi_day->type != PAN_ITEM_BOX ||
111 !pi_day->key || strcmp(pi_day->key, "day") != 0) return;
113 list = pan_layout_intersect(pw, pi_day->x, pi_day->y, pi_day->width, pi_day->height);
121 dot = static_cast<PanItem *>(work->data);
125 if (dot->type != PAN_ITEM_BOX || !dot->fd ||
126 !dot->key || strcmp(dot->key, "dot") != 0)
128 list = g_list_delete_link(list, node);
132 grid = static_cast<gint>(sqrt(g_list_length(list)) + 0.5);
134 x = pi_day->x + pi_day->width + 4;
137 pbox = pan_item_box_new(pw, nullptr, x, y, PAN_BOX_BORDER, PAN_BOX_BORDER,
138 PAN_CAL_POPUP_BORDER,
139 PAN_CAL_POPUP_COLOR, PAN_CAL_POPUP_ALPHA,
140 PAN_CAL_POPUP_BORDER_COLOR, PAN_CAL_POPUP_ALPHA);
141 pan_item_set_key(pbox, "day_bubble");
148 buf = pan_date_value_string(pi_day->fd->date, PAN_DATE_LENGTH_WEEK);
149 plabel = pan_item_text_new(pw, x, y, buf, static_cast<PanTextAttrType>(PAN_TEXT_ATTR_BOLD | PAN_TEXT_ATTR_HEADING),
151 PAN_CAL_POPUP_TEXT_COLOR, 255);
152 pan_item_set_key(plabel, "day_bubble");
155 pan_item_size_by_item(pbox, plabel, 0);
172 dot = static_cast<PanItem *>(work->data);
179 pimg = pan_item_thumb_new(pw, file_data_ref(dot->fd), x, y);
180 pan_item_set_key(pimg, "day_bubble");
182 pan_item_size_by_item(pbox, pimg, PAN_BOX_BORDER);
187 x += PAN_THUMB_SIZE + PAN_THUMB_GAP;
192 x = pbox->x + PAN_BOX_BORDER;
193 y += PAN_THUMB_SIZE + PAN_THUMB_GAP;
199 x1 = pi_day->x + pi_day->width - 8;
202 y2 = pbox->y + MIN(42, pbox->height);
204 y3 = MAX(pbox->y, y2 - 30);
205 util_clip_triangle(x1, y1, x2, y2, x3, y3,
208 pi = pan_item_tri_new(pw, nullptr, x, y, w, h,
209 x1, y1, x2, y2, x3, y3,
210 PAN_CAL_POPUP_COLOR, PAN_CAL_POPUP_ALPHA);
211 pan_item_tri_border(pi, PAN_BORDER_1 | PAN_BORDER_3, PAN_CAL_POPUP_BORDER_COLOR, PAN_CAL_POPUP_ALPHA);
212 pan_item_set_key(pi, "day_bubble");
213 pan_item_added(pw, pi);
215 pan_item_box_shadow(pbox, PAN_SHADOW_OFFSET * 2, PAN_SHADOW_FADE * 2);
216 pan_item_added(pw, pbox);
218 pan_layout_resize(pw);
221 void pan_calendar_compute(PanWindow *pw, FileData *dir_fd, gint *width, gint *height)
237 list = pan_list_tree(dir_fd, SORT_NONE, TRUE, TRUE, pw->ignore_symlinks);
238 pan_filter_fd_list(&list, pw->filter_ui->filter_elements, pw->filter_ui->filter_classes);
240 if (pw->cache_list && pw->exif_date_enable)
242 pw->cache_list = pan_cache_sort(pw->cache_list, SORT_NAME, TRUE, TRUE);
243 list = filelist_sort(list, SORT_NAME, TRUE, TRUE);
244 pan_cache_sync_date(pw, list);
247 pw->cache_list = pan_cache_sort(pw->cache_list, SORT_TIME, TRUE, TRUE);
248 list = filelist_sort(list, SORT_TIME, TRUE, TRUE);
258 fd = static_cast<FileData *>(work->data);
261 if (!pan_date_compare(fd->date, tc, PAN_DATE_LENGTH_DAY))
269 if (day_max < count) day_max = count;
273 DEBUG_1("biggest day contains %d images", day_max);
275 grid = static_cast<gint>(sqrt(static_cast<gdouble>(day_max)) + 0.5) * (PAN_THUMB_SIZE + PAN_SHADOW_OFFSET * 2 + PAN_THUMB_GAP);
279 auto fd = static_cast<FileData *>(list->data);
281 year = pan_date_value(fd->date, PAN_DATE_LENGTH_YEAR);
282 month = pan_date_value(fd->date, PAN_DATE_LENGTH_MONTH);
285 work = g_list_last(list);
288 auto fd = static_cast<FileData *>(work->data);
289 end_year = pan_date_value(fd->date, PAN_DATE_LENGTH_YEAR);
290 end_month = pan_date_value(fd->date, PAN_DATE_LENGTH_MONTH);
293 *width = PAN_BOX_BORDER * 2;
294 *height = PAN_BOX_BORDER * 2;
300 while (work && (year < end_year || (year == end_year && month <= end_month)))
304 PanItem *pi_day_number;
311 /* figure last second of this month */
312 dt = pan_date_to_time((month == 12) ? year + 1 : year, (month == 12) ? 1 : month + 1, 1);
315 /* anything to show this month? */
316 if (!pan_date_compare((static_cast<FileData *>(work->data))->date, dt, PAN_DATE_LENGTH_MONTH))
327 days = pan_date_value(dt, PAN_DATE_LENGTH_DAY);
328 dt = pan_date_to_time(year, month, 1);
329 col = pan_date_value(dt, PAN_DATE_LENGTH_WEEK);
330 col = col - (date_get_first_day_of_week() - 1);
331 if (col < 0) col = col + 7;
335 pi_month = pan_item_box_new(pw, nullptr, x, y, PAN_CAL_DAY_WIDTH * 7, PAN_CAL_DAY_HEIGHT / 4,
336 PAN_CAL_MONTH_BORDER,
337 PAN_CAL_MONTH_COLOR, PAN_CAL_MONTH_ALPHA,
338 PAN_CAL_MONTH_BORDER_COLOR, PAN_CAL_MONTH_ALPHA);
339 buf = pan_date_value_string(dt, PAN_DATE_LENGTH_MONTH);
340 pi_text = pan_item_text_new(pw, x, y, buf,
341 static_cast<PanTextAttrType>(PAN_TEXT_ATTR_BOLD | PAN_TEXT_ATTR_HEADING),
343 PAN_CAL_MONTH_TEXT_COLOR, 255);
345 pi_text->x = pi_month->x + (pi_month->width - pi_text->width) / 2;
347 pi_month->height = pi_text->y + pi_text->height - pi_month->y;
349 x = PAN_BOX_BORDER + col * PAN_CAL_DAY_WIDTH;
350 y = pi_month->y + pi_month->height + PAN_BOX_BORDER;
352 for (day = 1; day <= days; day++)
361 dt = pan_date_to_time(year, month, day);
364 * Create a FileData entry that represents the given day.
365 * It does not correspond to any real file
368 g_snprintf(fake_path, sizeof(fake_path), "//%04d-%02d-%02d", year, month, day);
369 fd = file_data_new_no_grouping(fake_path);
371 pi_day = pan_item_box_new(pw, fd, x, y, PAN_CAL_DAY_WIDTH, PAN_CAL_DAY_HEIGHT,
373 PAN_CAL_DAY_COLOR, PAN_CAL_DAY_ALPHA,
374 PAN_CAL_DAY_BORDER_COLOR, PAN_CAL_DAY_ALPHA);
375 pan_item_set_key(pi_day, "day");
377 dx = x + PAN_CAL_DOT_GAP * 2;
378 dy = y + PAN_CAL_DOT_GAP * 2;
380 fd = static_cast<FileData *>((work) ? work->data : nullptr);
381 while (fd && pan_date_compare(fd->date, dt, PAN_DATE_LENGTH_DAY))
385 pi = pan_item_box_new(pw, fd, dx, dy, PAN_CAL_DOT_SIZE, PAN_CAL_DOT_SIZE,
387 PAN_CAL_DOT_COLOR, PAN_CAL_DOT_ALPHA,
389 pan_item_set_key(pi, "dot");
391 dx += PAN_CAL_DOT_SIZE + PAN_CAL_DOT_GAP;
392 if (dx + PAN_CAL_DOT_SIZE > pi_day->x + pi_day->width - PAN_CAL_DOT_GAP * 2)
394 dx = x + PAN_CAL_DOT_GAP * 2;
395 dy += PAN_CAL_DOT_SIZE + PAN_CAL_DOT_GAP;
397 if (dy + PAN_CAL_DOT_SIZE > pi_day->y + pi_day->height - PAN_CAL_DOT_GAP * 2)
399 /* must keep all dots within respective day even if it gets ugly */
400 dy = y + PAN_CAL_DOT_GAP * 2;
406 fd = static_cast<FileData *>((work) ? work->data : nullptr);
413 pi_day->color_r = MAX(pi_day->color_r - 61 - n * 3, 80);
414 pi_day->color_g = pi_day->color_r;
416 buf = g_strdup_printf("( %d )", n);
417 pi = pan_item_text_new(pw, x, y, buf, PAN_TEXT_ATTR_NONE,
419 PAN_CAL_DAY_TEXT_COLOR, 255);
422 pi->x = pi_day->x + (pi_day->width - pi->width) / 2;
423 pi->y = pi_day->y + (pi_day->height - pi->height) / 2;
426 buf = g_strdup_printf("%d", day);
427 pi_day_number = pan_item_text_new(pw, x + 4, y + 4, buf, static_cast<PanTextAttrType>(PAN_TEXT_ATTR_BOLD | PAN_TEXT_ATTR_HEADING),
429 PAN_CAL_DAY_TEXT_COLOR, 255);
432 day_of_week = date_get_first_day_of_week() + col;
433 if (day_of_week > 7) day_of_week = day_of_week - 7;
435 buf = date_get_abbreviated_day_name(day_of_week);
436 pan_item_text_new(pw, x + 4 + pi_day_number->width + 4, y + 4, buf, PAN_TEXT_ATTR_NONE,
438 PAN_CAL_DAY_OF_WEEK_COLOR, 255);
441 pan_item_size_coordinates(pi_day, PAN_BOX_BORDER, width, height);
448 y += PAN_CAL_DAY_HEIGHT;
452 x += PAN_CAL_DAY_WIDTH;
456 if (col > 0) y += PAN_CAL_DAY_HEIGHT;
457 y += PAN_BOX_BORDER * 2;
468 *height = MAX(*height, grid + PAN_BOX_BORDER * 2 * 2);
472 /* vim: set shiftwidth=8 softtabstop=0 cindent cinoptions={1s: */