Trim trailing white spaces on empty lines.
[geeqie.git] / src / slideshow.c
index fb9629b..1d7cb97 100644 (file)
 /*
- * GQview image viewer
- * (C)1999 John Ellis
+ * Geeqie
+ * (C) 2004 John Ellis
+ * Copyright (C) 2008 - 2012 The Geeqie Team
  *
  * Author: John Ellis
  *
+ * This software is released under the GNU General Public License (GNU GPL).
+ * Please read the included file COPYING for more information.
+ * This software comes with no warranty of any kind, use at your own risk!
  */
 
-#include "gqview.h"
 
-static GList *slide_list = NULL;
-static gchar *slide_img = NULL;
-static gchar *slide_path = NULL;
-static gint slide_count = 0;
-static gint slide_active = FALSE;
-static gint slide_sel_list = FALSE;
-static gint slide_timeout_id = -1;
+#include "main.h"
+#include "collect.h"
+#include "image.h"
+#include "slideshow.h"
+#include "filedata.h"
 
-static void slideshow_free_all()
+#include "layout.h"
+#include "layout_image.h"
+#include "ui_fileops.h"
+
+
+static void slideshow_timer_stop(SlideShowData *ss);
+
+
+void slideshow_free(SlideShowData *ss)
 {
-       slide_active = FALSE;
+       if (!ss) return;
+
+       slideshow_timer_stop(ss);
 
-       g_list_free(slide_list);
-       slide_list = NULL;
+       if (ss->stop_func) ss->stop_func(ss, ss->stop_data);
 
-       g_free(slide_path);
-       slide_path = NULL;
+       if (ss->filelist) filelist_free(ss->filelist);
+       if (ss->cd) collection_unref(ss->cd);
+       file_data_unref(ss->dir_fd);
 
-       g_free(slide_img);
-       slide_img = NULL;
+       g_list_free(ss->list);
+       g_list_free(ss->list_done);
 
-       slide_count = 0;
-       slide_sel_list = FALSE;
+       file_data_unref(ss->slide_fd);
+
+       g_free(ss);
 }
 
-static GList *generate_list()
+static GList *generate_list(SlideShowData *ss)
 {
        GList *list = NULL;
 
-       if (file_selection_count() < 2)
+       if (ss->from_selection)
                {
-               gint i;
-               gint c = file_count();
-               for(i = 0; i < c; i++)
-                       {
-                       list = g_list_prepend(list, GINT_TO_POINTER(i));
-                       }
-               slide_sel_list = FALSE;
+               list = layout_selection_list_by_index(ss->lw);
                }
        else
                {
-               GList *work = GTK_CLIST(file_clist)->selection;
-               while(work)
+               guint i;
+               for (i = 0; i < ss->slide_count; i++)
                        {
-                       list = g_list_prepend(list, work->data);
-                       work = work->next;
+                       list = g_list_prepend(list, GINT_TO_POINTER(i));
                        }
-               slide_sel_list = TRUE;
+               list = g_list_reverse(list);
                }
-       list = g_list_reverse(list);
 
        return list;
 }
 
-static GList *generate_random_list()
+static void ptr_array_add(gpointer data, GPtrArray *array)
 {
-       GList *src_list = NULL;
-       GList *list = NULL;
-       GList *work;
+       g_ptr_array_add(array, data);
+}
 
-       src_list = generate_list();
+static void list_prepend(gpointer data, GList **list)
+{
+       *list = g_list_prepend(*list, data);
+}
 
-       while(src_list)
+static GPtrArray *generate_ptr_array_from_list(GList *src_list)
+{
+       GPtrArray *arr = g_ptr_array_sized_new(g_list_length(src_list));
+
+       g_list_foreach(src_list, (GFunc) ptr_array_add, arr);
+
+       return arr;
+}
+
+static void swap(GPtrArray *array, guint index1, guint index2)
+{
+       gpointer temp = g_ptr_array_index(array, index1);
+
+       g_ptr_array_index(array, index1) = g_ptr_array_index(array, index2);
+       g_ptr_array_index(array, index2) = temp;
+}
+
+static void ptr_array_random_shuffle(GPtrArray *array)
+{
+       guint i;
+       for (i = 0; i < array->len; ++i)
                {
-               gint p = (float)rand() / RAND_MAX * g_list_length(src_list);
-               work = g_list_nth(src_list, p);
-               list = g_list_prepend(list, work->data);
-               src_list = g_list_remove(src_list, work->data);
+               guint p = (double)rand() / ((double)RAND_MAX + 1.0) * array->len;
+               swap(array, i, p);
                }
+}
+
+static GList *generate_random_list(SlideShowData *ss)
+{
+       GList *src_list;
+       GPtrArray *src_array;
+       GList *list = NULL;
+
+       src_list = generate_list(ss);
+       src_array = generate_ptr_array_from_list(src_list);
+       g_list_free(src_list);
+
+       ptr_array_random_shuffle(src_array);
+       g_ptr_array_foreach(src_array, (GFunc) list_prepend, &list);
+       g_ptr_array_free(src_array, TRUE);
 
        return list;
 }
 
-static void slideshow_init_list()
+static void slideshow_list_init(SlideShowData *ss, gint start_index)
 {
-       if (slide_list)
+       if (ss->list_done)
                {
-               g_list_free(slide_list);
+               g_list_free(ss->list_done);
+               ss->list_done = NULL;
                }
 
-       if (slideshow_random)
+       if (ss->list) g_list_free(ss->list);
+
+       if (options->slideshow.random)
                {
-               slide_list = generate_random_list();
+               ss->list = generate_random_list(ss);
                }
        else
                {
-               slide_list = generate_list();
+               ss->list = generate_list(ss);
+               if (start_index >= 0)
+                       {
+                       /* start with specified image by skipping to it */
+                       gint i = 0;
+
+                       while (ss->list && i < start_index)
+                               {
+                               ss->list_done = g_list_prepend(ss->list_done, ss->list->data);
+                               ss->list = g_list_remove(ss->list, ss->list->data);
+                               i++;
+                               }
+                       }
                }
 }
 
-static gint slideshow_should_continue()
+gboolean slideshow_should_continue(SlideShowData *ss)
 {
-       if (!slide_active || !slide_list || !slide_path ||
-           slide_count != file_count() ||
-           (slide_img && image_get_path() && strcmp(image_get_path(), slide_img) != 0) ||
-           current_path == NULL ||
-           strcmp(current_path, slide_path) != 0)
+       FileData *imd_fd;
+       FileData *dir_fd;
+
+       if (!ss) return FALSE;
+
+       if (ss->lw)
+               imd_fd = layout_image_get_fd(ss->lw);
+       else
+               imd_fd = image_get_fd(ss->imd);
+
+       if ( ((imd_fd == NULL) != (ss->slide_fd == NULL)) ||
+           (imd_fd && ss->slide_fd && imd_fd != ss->slide_fd) ) return FALSE;
+
+       if (ss->filelist) return TRUE;
+
+       if (ss->cd)
                {
-               return FALSE;
+               if (g_list_length(ss->cd->list) == ss->slide_count)
+                       return TRUE;
+               else
+                       return FALSE;
                }
 
-       return TRUE;
+       dir_fd = ss->lw->dir_fd;
+
+       if (dir_fd && ss->dir_fd && dir_fd == ss->dir_fd)
+               {
+               if (ss->from_selection && ss->slide_count == layout_selection_count(ss->lw, NULL)) return TRUE;
+               if (!ss->from_selection && ss->slide_count == layout_list_count(ss->lw, NULL)) return TRUE;
+               }
+
+       return FALSE;
 }
 
-static gint slideshow_loop_cb(gpointer data)
+static gboolean slideshow_step(SlideShowData *ss, gboolean forward)
 {
        gint row;
-       gchar *buf;
 
-       if (!slideshow_should_continue())
+       if (!slideshow_should_continue(ss))
                {
-               slideshow_free_all();
-               slide_timeout_id = -1;
                return FALSE;
                }
 
-       row = GPOINTER_TO_INT(slide_list->data);
-
-       g_free(slide_img);
-       slide_img = NULL;
-       buf = file_get_path(row);
-       slide_list = g_list_remove(slide_list, slide_list->data);
+       if (forward)
+               {
+               if (!ss->list) return TRUE;
 
-       if (!slide_list && slideshow_repeat)
+               row = GPOINTER_TO_INT(ss->list->data);
+               ss->list_done = g_list_prepend(ss->list_done, ss->list->data);
+               ss->list = g_list_remove(ss->list, ss->list->data);
+               }
+       else
                {
-               slideshow_init_list();
+               if (!ss->list_done || !ss->list_done->next) return TRUE;
+
+               ss->list = g_list_prepend(ss->list, ss->list_done->data);
+               ss->list_done = g_list_remove(ss->list_done, ss->list_done->data);
+               row = GPOINTER_TO_INT(ss->list_done->data);
                }
 
-       if (slide_sel_list)
+       file_data_unref(ss->slide_fd);
+       ss->slide_fd = NULL;
+
+       if (ss->filelist)
                {
-               image_change_to(buf);
-               update_status_label(NULL);
+               ss->slide_fd = file_data_ref((FileData *)g_list_nth_data(ss->filelist, row));
+               if (ss->lw)
+                       layout_set_fd(ss->lw, ss->slide_fd);
+               else
+                       image_change_fd(ss->imd, ss->slide_fd, image_zoom_get_default(ss->imd));
+               }
+       else if (ss->cd)
+               {
+               CollectInfo *info;
+
+               info = g_list_nth_data(ss->cd->list, row);
+               ss->slide_fd = file_data_ref(info->fd);
+
+               if (ss->lw)
+                       image_change_from_collection(ss->lw->image, ss->cd, info, image_zoom_get_default(ss->lw->image));
+               else
+                       image_change_from_collection(ss->imd, ss->cd, info, image_zoom_get_default(ss->imd));
                }
        else
                {
-               file_image_change_to(row);
+               ss->slide_fd = file_data_ref(layout_list_get_fd(ss->lw, row));
+
+               if (ss->from_selection)
+                       {
+                       layout_set_fd(ss->lw, ss->slide_fd);
+                       layout_status_update_info(ss->lw, NULL);
+                       }
+               else
+                       {
+                       layout_image_set_index(ss->lw, row);
+                       }
                }
 
-       slide_img = buf;
+       if (!ss->list && options->slideshow.repeat)
+               {
+               slideshow_list_init(ss, -1);
+               }
 
-       if (!slide_list)
+       if (!ss->list)
                {
-               slideshow_free_all();
-               slide_timeout_id = -1;
                return FALSE;
                }
 
+       /* read ahead */
+       if (options->image.enable_read_ahead && (!ss->lw || ss->from_selection))
+               {
+               gint r;
+               if (forward)
+                       {
+                       if (!ss->list) return TRUE;
+                       r = GPOINTER_TO_INT(ss->list->data);
+                       }
+               else
+                       {
+                       if (!ss->list_done || !ss->list_done->next) return TRUE;
+                       r = GPOINTER_TO_INT(ss->list_done->next->data);
+                       }
+
+               if (ss->filelist)
+                       {
+                       image_prebuffer_set(ss->imd, g_list_nth_data(ss->filelist, r));
+                       }
+               else if (ss->cd)
+                       {
+                       CollectInfo *info;
+                       info = g_list_nth_data(ss->cd->list, r);
+                       if (info) image_prebuffer_set(ss->imd, info->fd);
+                       }
+               else if (ss->from_selection)
+                       {
+                       image_prebuffer_set(ss->lw->image, layout_list_get_fd(ss->lw, r));
+                       }
+               }
+
        return TRUE;
 }
 
-void slideshow_start()
+static gboolean slideshow_loop_cb(gpointer data)
 {
-       gint row;
-       gchar *buf;
+       SlideShowData *ss = data;
 
-       if (slide_active) return;
+       if (ss->paused) return TRUE;
 
-       if (file_count() < 2) return;
+       if (!slideshow_step(ss, TRUE))
+               {
+               ss->timeout_id = 0;
+               slideshow_free(ss);
+               return FALSE;
+               }
 
-       slideshow_init_list();
-       if (!slide_list) return;
+       return TRUE;
+}
 
-       row = GPOINTER_TO_INT(slide_list->data);
+static void slideshow_timer_stop(SlideShowData *ss)
+{
+       if (!ss->timeout_id) return;
 
-       slide_active = TRUE;
-       slide_path = g_strdup(current_path);
-       slide_count = file_count();
-       g_free(slide_img);
-       slide_img = NULL;
-       buf = file_get_path(row);
-       slide_list = g_list_remove(slide_list, slide_list->data);
+       g_source_remove(ss->timeout_id);
+       ss->timeout_id = 0;
+}
 
-       if (slide_sel_list)
-               {
-               image_change_to(buf);
-               update_status_label(NULL);
-               }
-       else
+static void slideshow_timer_reset(SlideShowData *ss)
+{
+       if (options->slideshow.delay < 1) options->slideshow.delay = 1;
+
+       if (ss->timeout_id) g_source_remove(ss->timeout_id);
+       ss->timeout_id = g_timeout_add(options->slideshow.delay * 1000 / SLIDESHOW_SUBSECOND_PRECISION,
+                                      slideshow_loop_cb, ss);
+}
+
+static void slideshow_move(SlideShowData *ss, gboolean forward)
+{
+       if (!ss) return;
+
+       if (!slideshow_step(ss, forward))
                {
-               file_image_change_to(row);
+               slideshow_free(ss);
+               return;
                }
 
-       slide_img = buf;
+       slideshow_timer_reset(ss);
+}
 
-       slide_timeout_id = gtk_timeout_add(slideshow_delay * 1000, slideshow_loop_cb, NULL);
+void slideshow_next(SlideShowData *ss)
+{
+       slideshow_move(ss, TRUE);
 }
 
-void slideshow_stop()
+void slideshow_prev(SlideShowData *ss)
 {
-       if (!slide_active) return;
+       slideshow_move(ss, FALSE);
+}
 
-       slideshow_free_all();
-       if (slide_timeout_id != -1)
+static SlideShowData *real_slideshow_start(LayoutWindow *target_lw, ImageWindow *imd,
+                                          GList *filelist, gint start_point,
+                                          CollectionData *cd, CollectInfo *start_info,
+                                          void (*stop_func)(SlideShowData *, gpointer), gpointer stop_data)
+{
+       SlideShowData *ss;
+       gint start_index = -1;
+
+       if (!filelist && !cd && layout_list_count(target_lw, NULL) < 1) return NULL;
+
+       ss = g_new0(SlideShowData, 1);
+
+       ss->lw = target_lw;
+       ss->imd = imd; /* FIXME: ss->imd is used only for img-view.c and can be dropped with it */
+       ss->filelist = filelist;
+       ss->cd = cd;
+
+       if (ss->filelist)
                {
-               gtk_timeout_remove(slide_timeout_id);
-               slide_timeout_id = -1;
+               ss->slide_count = g_list_length(ss->filelist);
                }
-       update_status_label(NULL);
-}
+       else if (ss->cd)
+               {
+               collection_ref(ss->cd);
+               ss->slide_count = g_list_length(ss->cd->list);
+               if (!options->slideshow.random && start_info)
+                       {
+                       start_index = g_list_index(ss->cd->list, start_info);
+                       }
+               }
+       else
+               {
+               /* layout method */
 
-void slideshow_toggle()
-{
-       if (!slide_active)
+               ss->slide_count = layout_selection_count(ss->lw, NULL);
+               ss->dir_fd = file_data_ref(ss->lw->dir_fd);
+               if (ss->slide_count < 2)
+                       {
+                       ss->slide_count = layout_list_count(ss->lw, NULL);
+                       if (!options->slideshow.random && start_point >= 0 && (guint) start_point < ss->slide_count)
+                               {
+                               start_index = start_point;
+                               }
+                       }
+               else
+                       {
+                       ss->from_selection = TRUE;
+                       }
+               }
+
+       slideshow_list_init(ss, start_index);
+
+       if (ss->lw)
+               ss->slide_fd = file_data_ref(layout_image_get_fd(ss->lw));
+       else
+               ss->slide_fd = file_data_ref(image_get_fd(ss->imd));
+
+       if (slideshow_step(ss, TRUE))
                {
-               slideshow_start();
+               slideshow_timer_reset(ss);
+
+               ss->stop_func = stop_func;
+               ss->stop_data = stop_data;
                }
        else
                {
-               slideshow_stop();
+               slideshow_free(ss);
+               ss = NULL;
                }
+
+       return ss;
 }
 
-gint slideshow_is_running()
+SlideShowData *slideshow_start_from_filelist(LayoutWindow *target_lw, ImageWindow *imd, GList *list,
+                                             void (*stop_func)(SlideShowData *, gpointer), gpointer stop_data)
 {
-       if (!slide_active) return FALSE;
+       return real_slideshow_start(target_lw, imd, list, -1, NULL, NULL, stop_func, stop_data);
+}
 
-       if (!slideshow_should_continue())
-               {
-               slideshow_free_all();
-               if (slide_timeout_id != -1)
-                       {
-                       gtk_timeout_remove(slide_timeout_id);
-                       slide_timeout_id = -1;
-                       }
-               return FALSE;
-               }
+SlideShowData *slideshow_start_from_collection(LayoutWindow *target_lw, ImageWindow *imd, CollectionData *cd,
+                                              void (*stop_func)(SlideShowData *, gpointer), gpointer stop_data,
+                                              CollectInfo *start_info)
+{
+       return real_slideshow_start(target_lw, imd, NULL, -1, cd, start_info, stop_func, stop_data);
+}
 
-       return TRUE;
+SlideShowData *slideshow_start(LayoutWindow *lw, gint start_point,
+                              void (*stop_func)(SlideShowData *, gpointer), gpointer stop_data)
+{
+       return real_slideshow_start(lw, NULL, NULL, start_point, NULL, NULL, stop_func, stop_data);
 }
 
+gboolean slideshow_paused(SlideShowData *ss)
+{
+       if (!ss) return FALSE;
+
+       return ss->paused;
+}
+
+void slideshow_pause_set(SlideShowData *ss, gboolean paused)
+{
+       if (!ss) return;
+
+       ss->paused = paused;
+}
+
+gboolean slideshow_pause_toggle(SlideShowData *ss)
+{
+       slideshow_pause_set(ss, !slideshow_paused(ss));
+       return slideshow_paused(ss);
+}
+/* vim: set shiftwidth=8 softtabstop=0 cindent cinoptions={1s: */