slideshow_next(), slideshow_prev(): move code to common slideshow_move(), reducing...
[geeqie.git] / src / slideshow.c
1 /*
2  * Geeqie
3  * (C) 2004 John Ellis
4  * Copyright (C) 2008 The Geeqie Team
5  *
6  * Author: John Ellis
7  *
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!
11  */
12
13
14 #include "main.h"
15 #include "collect.h"
16 #include "image.h"
17 #include "slideshow.h"
18 #include "filedata.h"
19
20 #include "layout.h"
21 #include "layout_image.h"
22 #include "ui_fileops.h"
23
24
25 static void slideshow_timer_stop(SlideShowData *ss);
26
27
28 void slideshow_free(SlideShowData *ss)
29 {
30         if (!ss) return;
31
32         slideshow_timer_stop(ss);
33
34         if (ss->stop_func) ss->stop_func(ss, ss->stop_data);
35
36         if (ss->filelist) filelist_free(ss->filelist);
37         if (ss->cd) collection_unref(ss->cd);
38         g_free(ss->layout_path);
39
40         g_list_free(ss->list);
41         g_list_free(ss->list_done);
42
43         file_data_unref(ss->slide_fd);
44
45         g_free(ss);
46 }
47
48 static GList *generate_list(SlideShowData *ss)
49 {
50         GList *list = NULL;
51
52         if (ss->from_selection)
53                 {
54                 list = layout_selection_list_by_index(ss->layout);
55                 }
56         else
57                 {
58                 guint i;
59                 for (i = 0; i < ss->slide_count; i++)
60                         {
61                         list = g_list_prepend(list, GINT_TO_POINTER(i));
62                         }
63                 list = g_list_reverse(list);
64                 }
65
66         return list;
67 }
68
69 static GList *generate_random_list(SlideShowData *ss)
70 {
71         GList *src_list = NULL;
72         GList *list = NULL;
73         GList *work;
74
75         src_list = generate_list(ss);
76
77         while (src_list)
78                 {
79                 gint p = (double)rand() / ((double)RAND_MAX + 1.0) * g_list_length(src_list);
80                 work = g_list_nth(src_list, p);
81                 list = g_list_prepend(list, work->data);
82                 src_list = g_list_remove(src_list, work->data);
83                 }
84
85         return list;
86 }
87
88 static void slideshow_list_init(SlideShowData *ss, gint start_index)
89 {
90         if (ss->list_done)
91                 {
92                 g_list_free(ss->list_done);
93                 ss->list_done = NULL;
94                 }
95
96         if (ss->list) g_list_free(ss->list);
97
98         if (options->slideshow.random)
99                 {
100                 ss->list = generate_random_list(ss);
101                 }
102         else
103                 {
104                 ss->list = generate_list(ss);
105                 if (start_index >= 0)
106                         {
107                         /* start with specified image by skipping to it */
108                         gint i = 0;
109
110                         while (ss->list && i < start_index)
111                                 {
112                                 ss->list_done = g_list_prepend(ss->list_done, ss->list->data);
113                                 ss->list = g_list_remove(ss->list, ss->list->data);
114                                 i++;
115                                 }
116                         }
117                 }
118 }
119
120 gint slideshow_should_continue(SlideShowData *ss)
121 {
122         FileData *imd_fd;
123         const gchar *path;
124
125         if (!ss) return FALSE;
126
127         imd_fd = image_get_fd(ss->imd);
128
129         if ( ((imd_fd == NULL) != (ss->slide_fd == NULL)) ||
130             (imd_fd && ss->slide_fd && imd_fd != ss->slide_fd) ) return FALSE;
131
132         if (ss->filelist) return TRUE;
133
134         if (ss->cd)
135                 {
136                 if (g_list_length(ss->cd->list) == ss->slide_count)
137                         return TRUE;
138                 else
139                         return FALSE;
140                 }
141
142         if (!ss->layout) return FALSE;
143         path = layout_get_path(ss->layout);
144
145         if (path && ss->layout_path &&
146             strcmp(path, ss->layout_path) == 0)
147                 {
148                 if (ss->from_selection && ss->slide_count == layout_selection_count(ss->layout, NULL)) return TRUE;
149                 if (!ss->from_selection && ss->slide_count == layout_list_count(ss->layout, NULL)) return TRUE;
150                 }
151
152         return FALSE;
153 }
154
155 static gint slideshow_step(SlideShowData *ss, gboolean forward)
156 {
157         gint row;
158
159         if (!slideshow_should_continue(ss))
160                 {
161                 return FALSE;
162                 }
163
164         if (forward)
165                 {
166                 if (!ss->list) return TRUE;
167
168                 row = GPOINTER_TO_INT(ss->list->data);
169                 ss->list_done = g_list_prepend(ss->list_done, ss->list->data);
170                 ss->list = g_list_remove(ss->list, ss->list->data);
171                 }
172         else
173                 {
174                 if (!ss->list_done || !ss->list_done->next) return TRUE;
175
176                 ss->list = g_list_prepend(ss->list, ss->list_done->data);
177                 ss->list_done = g_list_remove(ss->list_done, ss->list_done->data);
178                 row = GPOINTER_TO_INT(ss->list_done->data);
179                 }
180
181         file_data_unref(ss->slide_fd);
182         ss->slide_fd = NULL;
183
184         if (ss->filelist)
185                 {
186                 ss->slide_fd = file_data_ref((FileData *)g_list_nth_data(ss->filelist, row));
187                 image_change_fd(ss->imd, ss->slide_fd, image_zoom_get_default(ss->imd, options->image.zoom_mode));
188                 }
189         else if (ss->cd)
190                 {
191                 CollectInfo *info;
192
193                 info = g_list_nth_data(ss->cd->list, row);
194                 ss->slide_fd = file_data_ref(info->fd);
195
196                 image_change_from_collection(ss->imd, ss->cd, info, image_zoom_get_default(ss->imd, options->image.zoom_mode));
197                 }
198         else
199                 {
200                 ss->slide_fd = file_data_ref(layout_list_get_fd(ss->layout, row));
201
202                 if (ss->from_selection)
203                         {
204                         image_change_fd(ss->imd, ss->slide_fd, image_zoom_get_default(ss->imd, options->image.zoom_mode));
205                         layout_status_update_info(ss->layout, NULL);
206                         }
207                 else
208                         {
209                         layout_image_set_index(ss->layout, row);
210                         }
211                 }
212
213         if (!ss->list && options->slideshow.repeat)
214                 {
215                 slideshow_list_init(ss, -1);
216                 }
217
218         if (!ss->list)
219                 {
220                 return FALSE;
221                 }
222
223         /* read ahead */
224
225         if (options->image.enable_read_ahead)
226                 {
227                 gint r;
228                 if (forward)
229                         {
230                         if (!ss->list) return TRUE;
231                         r = GPOINTER_TO_INT(ss->list->data);
232                         }
233                 else
234                         {
235                         if (!ss->list_done || !ss->list_done->next) return TRUE;
236                         r = GPOINTER_TO_INT(ss->list_done->next->data);
237                         }
238
239                 if (ss->filelist)
240                         {
241                         image_prebuffer_set(ss->imd, g_list_nth_data(ss->filelist, r));
242                         }
243                 else if (ss->cd)
244                         {
245                         CollectInfo *info;
246                         info = g_list_nth_data(ss->cd->list, r);
247                         if (info) image_prebuffer_set(ss->imd, info->fd);
248                         }
249                 else if (ss->from_selection)
250                         {
251                         image_prebuffer_set(ss->imd, layout_list_get_fd(ss->layout, r));
252                         }
253                 }
254
255         return TRUE;
256 }
257
258 static gint slideshow_loop_cb(gpointer data)
259 {
260         SlideShowData *ss = data;
261
262         if (ss->paused) return TRUE;
263
264         if (!slideshow_step(ss, TRUE))
265                 {
266                 ss->timeout_id = -1;
267                 slideshow_free(ss);
268                 return FALSE;
269                 }
270
271         return TRUE;
272 }
273
274 static void slideshow_timer_stop(SlideShowData *ss)
275 {
276         if (ss->timeout_id == -1) return;
277
278         g_source_remove(ss->timeout_id);
279         ss->timeout_id = -1;
280 }
281
282 static void slideshow_timer_reset(SlideShowData *ss)
283 {
284         if (options->slideshow.delay < 1) options->slideshow.delay = 1;
285
286         if (ss->timeout_id != -1) g_source_remove(ss->timeout_id);
287         ss->timeout_id = g_timeout_add(options->slideshow.delay * 1000 / SLIDESHOW_SUBSECOND_PRECISION,
288                                        slideshow_loop_cb, ss);
289 }
290
291 static void slideshow_move(SlideShowData *ss, gboolean forward)
292 {
293         if (!ss) return;
294
295         if (!slideshow_step(ss, forward))
296                 {
297                 slideshow_free(ss);
298                 return;
299                 }
300
301         slideshow_timer_reset(ss);
302 }
303
304 void slideshow_next(SlideShowData *ss)
305 {
306         slideshow_move(ss, TRUE);
307 }
308
309 void slideshow_prev(SlideShowData *ss)
310 {
311         slideshow_move(ss, FALSE);
312 }
313
314 static SlideShowData *real_slideshow_start(ImageWindow *imd, LayoutWindow *lw,
315                                            GList *filelist, gint start_point,
316                                            CollectionData *cd, CollectInfo *start_info,
317                                            void (*stop_func)(SlideShowData *, gpointer), gpointer stop_data)
318 {
319         SlideShowData *ss;
320         gint start_index = -1;
321
322         if (!filelist && !cd && layout_list_count(lw, NULL) < 1) return NULL;
323
324         ss = g_new0(SlideShowData, 1);
325
326         ss->imd = imd;
327
328         ss->filelist = filelist;
329         ss->cd = cd;
330         ss->layout = lw;
331         ss->layout_path = NULL;
332
333         ss->list = NULL;
334         ss->list_done = NULL;
335
336         ss->from_selection = FALSE;
337
338         ss->stop_func = NULL;
339
340         ss->timeout_id = -1;
341         ss->paused = FALSE;
342
343         if (ss->filelist)
344                 {
345                 ss->slide_count = g_list_length(ss->filelist);
346                 }
347         else if (ss->cd)
348                 {
349                 collection_ref(ss->cd);
350                 ss->slide_count = g_list_length(ss->cd->list);
351                 if (!options->slideshow.random && start_info)
352                         {
353                         start_index = g_list_index(ss->cd->list, start_info);
354                         }
355                 }
356         else
357                 {
358                 /* layout method */
359
360                 ss->slide_count = layout_selection_count(ss->layout, NULL);
361                 ss->layout_path = g_strdup(layout_get_path(ss->layout));
362                 if (ss->slide_count < 2)
363                         {
364                         ss->slide_count = layout_list_count(ss->layout, NULL);
365                         if (!options->slideshow.random && start_point >= 0 && (guint) start_point < ss->slide_count)
366                                 {
367                                 start_index = start_point;
368                                 }
369                         }
370                 else
371                         {
372                         ss->from_selection = TRUE;
373                         }
374                 }
375
376         slideshow_list_init(ss, start_index);
377
378         ss->slide_fd = file_data_ref(image_get_fd(ss->imd));
379         if (slideshow_step(ss, TRUE))
380                 {
381                 slideshow_timer_reset(ss);
382
383                 ss->stop_func = stop_func;
384                 ss->stop_data = stop_data;
385                 }
386         else
387                 {
388                 slideshow_free(ss);
389                 ss = NULL;
390                 }
391
392         return ss;
393 }
394
395 SlideShowData *slideshow_start_from_filelist(ImageWindow *imd, GList *list,
396                                               void (*stop_func)(SlideShowData *, gpointer), gpointer stop_data)
397 {
398         return real_slideshow_start(imd, NULL, list, -1, NULL, NULL, stop_func, stop_data);
399 }
400
401 SlideShowData *slideshow_start_from_collection(ImageWindow *imd, CollectionData *cd,
402                                                void (*stop_func)(SlideShowData *, gpointer), gpointer stop_data,
403                                                CollectInfo *start_info)
404 {
405         return real_slideshow_start(imd, NULL, NULL, -1, cd, start_info, stop_func, stop_data);
406 }
407
408 SlideShowData *slideshow_start(ImageWindow *imd, LayoutWindow *lw, gint start_point,
409                                void (*stop_func)(SlideShowData *, gpointer), gpointer stop_data)
410 {
411         return real_slideshow_start(imd, lw, NULL, start_point, NULL, NULL, stop_func, stop_data);
412 }
413
414 gint slideshow_paused(SlideShowData *ss)
415 {
416         if (!ss) return FALSE;
417
418         return ss->paused;
419 }
420
421 void slideshow_pause_set(SlideShowData *ss, gint paused)
422 {
423         if (!ss) return;
424
425         ss->paused = paused;
426 }
427
428 gint slideshow_pause_toggle(SlideShowData *ss)
429 {
430         slideshow_pause_set(ss, !slideshow_paused(ss));
431         return slideshow_paused(ss);
432 }