/*
- * GQview image viewer
- * (C)2000 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 GList *past_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_list_free(past_slide_list);
- past_slide_list = NULL;
+ if (ss->filelist) filelist_free(ss->filelist);
+ if (ss->cd) collection_unref(ss->cd);
+ file_data_unref(ss->dir_fd);
- g_free(slide_path);
- slide_path = NULL;
+ g_list_free(ss->list);
+ g_list_free(ss->list_done);
- g_free(slide_img);
- slide_img = NULL;
+ file_data_unref(ss->slide_fd);
- slide_count = 0;
- slide_sel_list = FALSE;
+ 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);
+}
+
+static GPtrArray *generate_ptr_array_from_list(GList *src_list)
+{
+ GPtrArray *arr = g_ptr_array_sized_new(g_list_length(src_list));
- while(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 (past_slide_list)
- {
- g_list_free(past_slide_list);
- past_slide_list = NULL;
- }
+ if (ss->list) g_list_free(ss->list);
- if (slideshow_random)
+ 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 void slideshow_move_list(gint forward)
+gboolean slideshow_should_continue(SlideShowData *ss)
{
- if (forward)
- {
- if (slide_list)
- {
- past_slide_list = g_list_prepend (past_slide_list, slide_list->data);
- slide_list = g_list_remove(slide_list, slide_list->data);
- }
- }
+ 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)
{
- if (past_slide_list)
- {
- slide_list = g_list_prepend(slide_list, past_slide_list->data);
- past_slide_list = g_list_remove(past_slide_list, past_slide_list->data);
- }
+ if (g_list_length(ss->cd->list) == ss->slide_count)
+ return TRUE;
+ else
+ return FALSE;
}
-}
-static gint slideshow_should_continue()
-{
- 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)
+ dir_fd = ss->lw->dir_fd;
+
+ if (dir_fd && ss->dir_fd && dir_fd == ss->dir_fd)
{
- return FALSE;
+ 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 TRUE;
+ return FALSE;
}
-static gint real_slideshow_prev()
+static gboolean slideshow_step(SlideShowData *ss, gboolean forward)
{
gint row;
- gchar *buf;
-
- if (!slide_active) return FALSE;
- if (!past_slide_list || !past_slide_list->next) return TRUE;
- if (!slideshow_should_continue())
+ if (!slideshow_should_continue(ss))
{
- slideshow_free_all();
- slide_timeout_id = -1;
return FALSE;
}
- slideshow_move_list(FALSE);
+ if (forward)
+ {
+ if (!ss->list) return TRUE;
+
+ 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
+ {
+ if (!ss->list_done || !ss->list_done->next) return TRUE;
- row = GPOINTER_TO_INT(past_slide_list->data);
+ 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);
+ }
- g_free(slide_img);
- slide_img = NULL;
- buf = file_get_path(row);
+ file_data_unref(ss->slide_fd);
+ ss->slide_fd = NULL;
- if (slide_sel_list)
+ 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);
+ }
+ }
+
+ if (!ss->list && options->slideshow.repeat)
+ {
+ slideshow_list_init(ss, -1);
}
- slide_img = buf;
+ if (!ss->list)
+ {
+ 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;
}
-/* the return is TRUE if slideshow should continue */
-static gint real_slideshow_next()
+static gboolean slideshow_loop_cb(gpointer data)
{
- gint row;
- gchar *buf;
+ SlideShowData *ss = data;
- if (!slide_active) return FALSE;
+ if (ss->paused) return TRUE;
- if (!slideshow_should_continue())
+ if (!slideshow_step(ss, TRUE))
{
- slideshow_free_all();
- slide_timeout_id = -1;
+ ss->timeout_id = 0;
+ slideshow_free(ss);
return FALSE;
}
- row = GPOINTER_TO_INT(slide_list->data);
+ return TRUE;
+}
+
+static void slideshow_timer_stop(SlideShowData *ss)
+{
+ if (!ss->timeout_id) return;
- g_free(slide_img);
- slide_img = NULL;
- buf = file_get_path(row);
- slideshow_move_list(TRUE);
+ g_source_remove(ss->timeout_id);
+ ss->timeout_id = 0;
+}
- if (!slide_list && slideshow_repeat)
- {
- slideshow_init_list();
- }
+static void slideshow_timer_reset(SlideShowData *ss)
+{
+ if (options->slideshow.delay < 1) options->slideshow.delay = 1;
- if (slide_sel_list)
- {
- image_change_to(buf);
- update_status_label(NULL);
- }
- else
- {
- file_image_change_to(row);
- }
+ 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);
+}
- slide_img = buf;
+static void slideshow_move(SlideShowData *ss, gboolean forward)
+{
+ if (!ss) return;
- if (!slide_list)
+ if (!slideshow_step(ss, forward))
{
- slideshow_free_all();
- slide_timeout_id = -1;
- return FALSE;
+ slideshow_free(ss);
+ return;
}
- return TRUE;
+ slideshow_timer_reset(ss);
}
-static gint slideshow_loop_cb(gpointer data)
+void slideshow_next(SlideShowData *ss)
{
- return real_slideshow_next();
+ slideshow_move(ss, TRUE);
}
-void slideshow_start()
+void slideshow_prev(SlideShowData *ss)
{
- gint row;
- gchar *buf;
-
- if (slide_active) return;
+ slideshow_move(ss, FALSE);
+}
- if (file_count() < 2) return;
+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;
- slideshow_init_list();
- if (!slide_list) return;
+ if (!filelist && !cd && layout_list_count(target_lw, NULL) < 1) return NULL;
- row = GPOINTER_TO_INT(slide_list->data);
+ ss = g_new0(SlideShowData, 1);
- 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);
- slideshow_move_list(TRUE);
+ 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 (slide_sel_list)
+ if (ss->filelist)
{
- image_change_to(buf);
- update_status_label(NULL);
+ ss->slide_count = g_list_length(ss->filelist);
}
- else
+ else if (ss->cd)
{
- file_image_change_to(row);
+ 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 */
- slide_img = buf;
+ 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;
+ }
+ }
- slide_timeout_id = gtk_timeout_add(slideshow_delay * 1000, slideshow_loop_cb, NULL);
-}
+ slideshow_list_init(ss, start_index);
-void slideshow_stop()
-{
- if (!slide_active) return;
+ 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));
- slideshow_free_all();
- if (slide_timeout_id != -1)
+ if (slideshow_step(ss, TRUE))
{
- gtk_timeout_remove(slide_timeout_id);
- slide_timeout_id = -1;
- }
- update_status_label(NULL);
-}
+ slideshow_timer_reset(ss);
-static void slideshow_reset_timeout(gint reset)
-{
- if (reset)
- {
- if (slide_timeout_id != -1) gtk_timeout_remove(slide_timeout_id);
- slide_timeout_id = gtk_timeout_add(slideshow_delay * 1000, slideshow_loop_cb, NULL);
+ ss->stop_func = stop_func;
+ ss->stop_data = stop_data;
}
else
{
- if (slide_timeout_id != -1)
- {
- gtk_timeout_remove(slide_timeout_id);
- slide_timeout_id = -1;
- }
+ slideshow_free(ss);
+ ss = NULL;
}
+
+ return ss;
}
-void slideshow_next()
+SlideShowData *slideshow_start_from_filelist(LayoutWindow *target_lw, ImageWindow *imd, GList *list,
+ void (*stop_func)(SlideShowData *, gpointer), gpointer stop_data)
{
- if (!slide_active) return;
- slideshow_reset_timeout(real_slideshow_next());
+ return real_slideshow_start(target_lw, imd, list, -1, NULL, NULL, stop_func, stop_data);
}
-void slideshow_prev()
+SlideShowData *slideshow_start_from_collection(LayoutWindow *target_lw, ImageWindow *imd, CollectionData *cd,
+ void (*stop_func)(SlideShowData *, gpointer), gpointer stop_data,
+ CollectInfo *start_info)
{
- if (!slide_active) return;
- slideshow_reset_timeout(real_slideshow_prev());
+ return real_slideshow_start(target_lw, imd, NULL, -1, cd, start_info, stop_func, stop_data);
}
-void slideshow_toggle()
+SlideShowData *slideshow_start(LayoutWindow *lw, gint start_point,
+ void (*stop_func)(SlideShowData *, gpointer), gpointer stop_data)
{
- if (!slide_active)
- {
- slideshow_start();
- }
- else
- {
- slideshow_stop();
- }
+ return real_slideshow_start(lw, NULL, NULL, start_point, NULL, NULL, stop_func, stop_data);
}
-gint slideshow_is_running()
+gboolean slideshow_paused(SlideShowData *ss)
{
- if (!slide_active) return FALSE;
+ if (!ss) return FALSE;
- if (!slideshow_should_continue())
- {
- slideshow_free_all();
- if (slide_timeout_id != -1)
- {
- gtk_timeout_remove(slide_timeout_id);
- slide_timeout_id = -1;
- }
- return FALSE;
- }
+ return ss->paused;
+}
- return TRUE;
+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: */