4 * Copyright (C) 2008 - 2009 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!
17 #include "slideshow.h"
21 #include "layout_image.h"
22 #include "ui_fileops.h"
25 static void slideshow_timer_stop(SlideShowData *ss);
28 void slideshow_free(SlideShowData *ss)
32 slideshow_timer_stop(ss);
34 if (ss->stop_func) ss->stop_func(ss, ss->stop_data);
36 if (ss->filelist) filelist_free(ss->filelist);
37 if (ss->cd) collection_unref(ss->cd);
38 file_data_unref(ss->dir_fd);
40 g_list_free(ss->list);
41 g_list_free(ss->list_done);
43 file_data_unref(ss->slide_fd);
48 static GList *generate_list(SlideShowData *ss)
52 if (ss->from_selection)
54 list = layout_selection_list_by_index(ss->layout);
59 for (i = 0; i < ss->slide_count; i++)
61 list = g_list_prepend(list, GINT_TO_POINTER(i));
63 list = g_list_reverse(list);
69 static void ptr_array_add(gpointer data, GPtrArray *array)
71 g_ptr_array_add(array, data);
74 static void list_prepend(gpointer data, GList **list)
76 *list = g_list_prepend(*list, data);
79 static GPtrArray *generate_ptr_array_from_list(GList *src_list)
81 GPtrArray *arr = g_ptr_array_sized_new(g_list_length(src_list));
83 g_list_foreach(src_list, (GFunc) ptr_array_add, arr);
88 static void swap(GPtrArray *array, guint index1, guint index2)
90 gpointer temp = g_ptr_array_index(array, index1);
92 g_ptr_array_index(array, index1) = g_ptr_array_index(array, index2);
93 g_ptr_array_index(array, index2) = temp;
96 static void ptr_array_random_shuffle(GPtrArray *array)
99 for (i = 0; i < array->len; ++i)
101 guint p = (double)rand() / ((double)RAND_MAX + 1.0) * array->len;
106 static GList *generate_random_list(SlideShowData *ss)
109 GPtrArray *src_array;
112 src_list = generate_list(ss);
113 src_array = generate_ptr_array_from_list(src_list);
114 g_list_free(src_list);
116 ptr_array_random_shuffle(src_array);
117 g_ptr_array_foreach(src_array, (GFunc) list_prepend, &list);
118 g_ptr_array_free(src_array, TRUE);
123 static void slideshow_list_init(SlideShowData *ss, gint start_index)
127 g_list_free(ss->list_done);
128 ss->list_done = NULL;
131 if (ss->list) g_list_free(ss->list);
133 if (options->slideshow.random)
135 ss->list = generate_random_list(ss);
139 ss->list = generate_list(ss);
140 if (start_index >= 0)
142 /* start with specified image by skipping to it */
145 while (ss->list && i < start_index)
147 ss->list_done = g_list_prepend(ss->list_done, ss->list->data);
148 ss->list = g_list_remove(ss->list, ss->list->data);
155 gboolean slideshow_should_continue(SlideShowData *ss)
160 if (!ss) return FALSE;
162 imd_fd = image_get_fd(ss->imd);
164 if ( ((imd_fd == NULL) != (ss->slide_fd == NULL)) ||
165 (imd_fd && ss->slide_fd && imd_fd != ss->slide_fd) ) return FALSE;
167 if (ss->filelist) return TRUE;
171 if (g_list_length(ss->cd->list) == ss->slide_count)
177 if (!ss->layout) return FALSE;
178 dir_fd = ss->layout->dir_fd;
180 if (dir_fd && ss->dir_fd && dir_fd == ss->dir_fd)
182 if (ss->from_selection && ss->slide_count == layout_selection_count(ss->layout, NULL)) return TRUE;
183 if (!ss->from_selection && ss->slide_count == layout_list_count(ss->layout, NULL)) return TRUE;
189 static gboolean slideshow_step(SlideShowData *ss, gboolean forward)
193 if (!slideshow_should_continue(ss))
200 if (!ss->list) return TRUE;
202 row = GPOINTER_TO_INT(ss->list->data);
203 ss->list_done = g_list_prepend(ss->list_done, ss->list->data);
204 ss->list = g_list_remove(ss->list, ss->list->data);
208 if (!ss->list_done || !ss->list_done->next) return TRUE;
210 ss->list = g_list_prepend(ss->list, ss->list_done->data);
211 ss->list_done = g_list_remove(ss->list_done, ss->list_done->data);
212 row = GPOINTER_TO_INT(ss->list_done->data);
215 file_data_unref(ss->slide_fd);
220 ss->slide_fd = file_data_ref((FileData *)g_list_nth_data(ss->filelist, row));
221 image_change_fd(ss->imd, ss->slide_fd, image_zoom_get_default(ss->imd));
227 info = g_list_nth_data(ss->cd->list, row);
228 ss->slide_fd = file_data_ref(info->fd);
230 image_change_from_collection(ss->imd, ss->cd, info, image_zoom_get_default(ss->imd));
234 ss->slide_fd = file_data_ref(layout_list_get_fd(ss->layout, row));
236 if (ss->from_selection)
238 image_change_fd(ss->imd, ss->slide_fd, image_zoom_get_default(ss->imd));
239 layout_status_update_info(ss->layout, NULL);
243 layout_image_set_index(ss->layout, row);
247 if (!ss->list && options->slideshow.repeat)
249 slideshow_list_init(ss, -1);
259 if (options->image.enable_read_ahead)
264 if (!ss->list) return TRUE;
265 r = GPOINTER_TO_INT(ss->list->data);
269 if (!ss->list_done || !ss->list_done->next) return TRUE;
270 r = GPOINTER_TO_INT(ss->list_done->next->data);
275 image_prebuffer_set(ss->imd, g_list_nth_data(ss->filelist, r));
280 info = g_list_nth_data(ss->cd->list, r);
281 if (info) image_prebuffer_set(ss->imd, info->fd);
283 else if (ss->from_selection)
285 image_prebuffer_set(ss->imd, layout_list_get_fd(ss->layout, r));
292 static gboolean slideshow_loop_cb(gpointer data)
294 SlideShowData *ss = data;
296 if (ss->paused) return TRUE;
298 if (!slideshow_step(ss, TRUE))
308 static void slideshow_timer_stop(SlideShowData *ss)
310 if (ss->timeout_id == -1) return;
312 g_source_remove(ss->timeout_id);
316 static void slideshow_timer_reset(SlideShowData *ss)
318 if (options->slideshow.delay < 1) options->slideshow.delay = 1;
320 if (ss->timeout_id != -1) g_source_remove(ss->timeout_id);
321 ss->timeout_id = g_timeout_add(options->slideshow.delay * 1000 / SLIDESHOW_SUBSECOND_PRECISION,
322 slideshow_loop_cb, ss);
325 static void slideshow_move(SlideShowData *ss, gboolean forward)
329 if (!slideshow_step(ss, forward))
335 slideshow_timer_reset(ss);
338 void slideshow_next(SlideShowData *ss)
340 slideshow_move(ss, TRUE);
343 void slideshow_prev(SlideShowData *ss)
345 slideshow_move(ss, FALSE);
348 static SlideShowData *real_slideshow_start(ImageWindow *imd, LayoutWindow *lw,
349 GList *filelist, gint start_point,
350 CollectionData *cd, CollectInfo *start_info,
351 void (*stop_func)(SlideShowData *, gpointer), gpointer stop_data)
354 gint start_index = -1;
356 if (!filelist && !cd && layout_list_count(lw, NULL) < 1) return NULL;
358 ss = g_new0(SlideShowData, 1);
361 ss->filelist = filelist;
368 ss->slide_count = g_list_length(ss->filelist);
372 collection_ref(ss->cd);
373 ss->slide_count = g_list_length(ss->cd->list);
374 if (!options->slideshow.random && start_info)
376 start_index = g_list_index(ss->cd->list, start_info);
383 ss->slide_count = layout_selection_count(ss->layout, NULL);
384 ss->dir_fd = file_data_ref(ss->layout->dir_fd);
385 if (ss->slide_count < 2)
387 ss->slide_count = layout_list_count(ss->layout, NULL);
388 if (!options->slideshow.random && start_point >= 0 && (guint) start_point < ss->slide_count)
390 start_index = start_point;
395 ss->from_selection = TRUE;
399 slideshow_list_init(ss, start_index);
401 ss->slide_fd = file_data_ref(image_get_fd(ss->imd));
402 if (slideshow_step(ss, TRUE))
404 slideshow_timer_reset(ss);
406 ss->stop_func = stop_func;
407 ss->stop_data = stop_data;
418 SlideShowData *slideshow_start_from_filelist(ImageWindow *imd, GList *list,
419 void (*stop_func)(SlideShowData *, gpointer), gpointer stop_data)
421 return real_slideshow_start(imd, NULL, list, -1, NULL, NULL, stop_func, stop_data);
424 SlideShowData *slideshow_start_from_collection(ImageWindow *imd, CollectionData *cd,
425 void (*stop_func)(SlideShowData *, gpointer), gpointer stop_data,
426 CollectInfo *start_info)
428 return real_slideshow_start(imd, NULL, NULL, -1, cd, start_info, stop_func, stop_data);
431 SlideShowData *slideshow_start(ImageWindow *imd, LayoutWindow *lw, gint start_point,
432 void (*stop_func)(SlideShowData *, gpointer), gpointer stop_data)
434 return real_slideshow_start(imd, lw, NULL, start_point, NULL, NULL, stop_func, stop_data);
437 gboolean slideshow_paused(SlideShowData *ss)
439 if (!ss) return FALSE;
444 void slideshow_pause_set(SlideShowData *ss, gboolean paused)
451 gboolean slideshow_pause_toggle(SlideShowData *ss)
453 slideshow_pause_set(ss, !slideshow_paused(ss));
454 return slideshow_paused(ss);
456 /* vim: set shiftwidth=8 softtabstop=0 cindent cinoptions={1s: */